diff --git a/Makefile b/Makefile index a53cde0a026b..6c1ed017c0f9 100644 --- a/Makefile +++ b/Makefile @@ -14,4 +14,8 @@ docs: build: uv build -prepare-release: docs build \ No newline at end of file +prepare-release: docs build + +synchronize-base-object-schemas: + cd weave && make generate_base_object_schemas && \ + cd ../weave-js && yarn generate-schemas diff --git a/dev_docs/BaseObjectClasses.md b/dev_docs/BaseObjectClasses.md new file mode 100644 index 000000000000..a571af49755b --- /dev/null +++ b/dev_docs/BaseObjectClasses.md @@ -0,0 +1,218 @@ +# BaseObjectClasses + +## Refresher on Objects and object storage + +In Weave, we have a general-purpose data storage system for objects. +The payloads themselves are completely free-form - basically anything that can be JSON-serialized. +Users can "publish" runtime objects to weave using `weave.publish`. +For example: + +```python +config = {"model_name": "my_model", "model_version": "1.0"} +ref = weave.publish(config, name="my_model_config") +``` + +This will create a new object "version" in the collection called "my_model_config". +These can then be retrieved using `weave.ref().get()`: + +```python +config = weave.ref("my_model_config").get() +``` + +Sometimes users are working with standard structured classes like `dataclasses` or `pydantic.BaseModel`. +In such cases, we have special serialization and deserialization logic that allows for cleaner serialization patterns. +For example, let's say the user does: + +```python +class ModelConfig(weave.Object): + model_name: str + model_version: str +``` + +Then the user can publish an instance of `ModelConfig` as follows: + +```python +config = ModelConfig(model_name="my_model", model_version="1.0") +ref = weave.publish(config) +``` + +This will result in an on-disk payload that looks like: + +```json +{ + "model_name": "my_model", + "model_version": "1.0", + "_type": "ModelConfig", + "_class_name": "ModelConfig", + "_bases": ["Object", "BaseModel"] +} +``` + +And additionally, the user can query for all objects of the `ModelConfig` class using the `base_object_classes` filter in `objs_query` or `POST objs/query`. +Effectively, this is like creating a virtual table for that class. + +**Terminology**: We use the term "weave Object" (capital "O") to refer to instances of classes that subclass `weave.Object`. + +**Technical note**: the "base_object_class" is the first subtype of "Object", not the _class_name. +For example, let's say the class hierarchy is: +* `A -> Object -> BaseModel`, then the `base_object_class` filter will be "A". +* `B -> A -> Object -> BaseModel`, then the `base_object_class` filter will still be "A"! + +Finally, the Weave library itself utilizes this mechanism for common objects like `Model`, `Dataset`, `Evaluation`, etc... +This allows the user to subclass these objects to add additional metadata or functionality, while categorizing them in the same virtual table. + +## Validated Base Objects + +While many Weave Objects are free-form and user-defined, there is often a need for well-defined schemas for configuration objects that are tightly defined by Weave itself. The BaseObject system provides a way to define these schemas once and use them consistently across the entire stack. + +### Key Features + +1. **Single Source of Truth**: Define your schema once using Pydantic models +2. **Full Stack Integration**: The schema is used for: + - Python SDK validation + - Server-side HTTP API validation + - Frontend UI validation with generated TypeScript types + - Future: OpenAPI schema generation + - Future: TypeScript SDK type generation + +### Usage Example + +Here's how to define and use a validated base object: + +1. **Define your schema** (in `weave/trace_server/interface/base_object_classes/your_schema.py`): + +```python +from pydantic import BaseModel +from weave.trace_server.interface.base_object_classes import base_object_def + +class NestedConfig(BaseModel): + setting_a: int + +class MyConfig(base_object_def.BaseObject): + name: str + nested: NestedConfig + reference: base_object_def.RefStr + +__all__ = ["MyConfig"] +``` + +2. **Use in Python**: +```python +# Publishing +ref = weave.publish(MyConfig(...)) + +# Fetching (maintains type) +config = ref.get() +assert isinstance(config, MyConfig) +``` + +3. **Use via HTTP API**: +```bash +# Creating +curl -X POST 'https://trace.wandb.ai/obj/create' \ + -H 'Content-Type: application/json' \ + -d '{ + "obj": { + "project_id": "user/project", + "object_id": "my_config", + "val": {...}, + "set_base_object_class": "MyConfig" + } + }' + +# Querying +curl -X POST 'https://trace.wandb.ai/objs/query' \ + -d '{ + "project_id": "user/project", + "filter": { + "base_object_classes": ["MyConfig"] + } + }' +``` + +4. **Use in React**: +```typescript +// Read with type safety +const result = useBaseObjectInstances("MyConfig", ...); + +// Write with validation +const createFn = useCreateBaseObjectInstance("MyConfig"); +createFn({...}); // TypeScript enforced schema +``` + +### Keeping Frontend Types in Sync + +Run `make synchronize-base-object-schemas` to ensure the frontend TypeScript types are up to date with your Pydantic schemas. + +### Implementation Notes + +- Base objects are pure data schemas (fields only) +- The system is designed to work independently of the weave SDK to maintain clean separation of concerns +- Server-side validation ensures data integrity +- Client-side validation (both Python and TypeScript) provides early feedback +- Generated TypeScript types ensure type safety in the frontend + +### Architecture Flow + +1. Define your schema in a python file in the `weave/trace_server/interface/base_object_classes/test_only_example.py` directory. See `weave/trace_server/interface/base_object_classes/test_only_example.py` as an example. +2. Make sure to register your schemas in `weave/trace_server/interface/base_object_classes/base_object_registry.py` by calling `register_base_object`. +3. Run `make synchronize-base-object-schemas` to generate the frontend types. + * The first step (`make generate_base_object_schemas`) will run `weave/scripts/generate_base_object_schemas.py` to generate a JSON schema in `weave/trace_server/interface/base_object_classes/generated/generated_base_object_class_schemas.json`. + * The second step (yarn `generate-schemas`) will read this file and use it to generate the frontend types located in `weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/generatedBaseObjectClasses.zod.ts`. +4. Now, each use case uses different parts: + 1. `Python Writing`. Users can directly import these classes and use them as normal Pydantic models, which get published with `weave.publish`. The python client correct builds the requisite payload. + 2. `Python Reading`. Users can `weave.ref().get()` and the weave python SDK will return the instance with the correct type. Note: we do some special handling such that the returned object is not a WeaveObject, but literally the exact pydantic class. + 3. `HTTP Writing`. In cases where the client/user does not want to add the special type information, users can publish base objects by setting the `set_base_object_class` setting on `POST obj/create` to the name of the class. The weave server will validate the object against the schema, update the metadata fields, and store the object. + 4. `HTTP Reading`. When querying for objects, the server will return the object with the correct type if the `base_object_class` metadata field is set. + 5. `Frontend`. The frontend will read the zod schema from `weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/generatedBaseObjectClasses.zod.ts` and use that to provide compile time type safety when using `useBaseObjectInstances` and runtime type safety when using `useCreateBaseObjectInstance`. +* Note: it is critical that all techniques produce the same digest for the same data - which is tested in the tests. This way versions are not thrashed by different clients/users. + +```mermaid +graph TD + subgraph Schema Definition + F["weave/trace_server/interface/
base_object_classes/your_schema.py"] --> |defines| P[Pydantic BaseObject] + P --> |register_base_object| R["base_object_registry.py"] + end + + subgraph Schema Generation + M["make synchronize-base-object-schemas"] --> G["make generate_base_object_schemas"] + G --> |runs| S["weave/scripts/
generate_base_object_schemas.py"] + R --> |import registered classes| S + S --> |generates| J["generated_base_object_class_schemas.json"] + M --> |yarn generate-schemas| Z["generatedBaseObjectClasses.zod.ts"] + J --> Z + end + + subgraph "Trace Server" + subgraph "HTTP API" + R --> |validates using| HW["POST obj/create
set_base_object_class"] + HW --> DB[(Weave Object Store)] + HR["POST objs/query
base_object_classes"] --> |Filters base_object_class| DB + end + end + + subgraph "Python SDK" + PW[Client Code] --> |import & publish| W[weave.publish] + W --> |store| HW + R --> |validates using| W + PR["weave ref get()"] --> |queries| HR + R --> |deserializes using| PR + end + + subgraph "Frontend" + Z --> |import| UBI["useBaseObjectInstances"] + Z --> |import| UCI["useCreateBaseObjectInstance"] + UBI --> |Filters base_object_class| HR + UCI --> |set_base_object_class| HW + UI[React UI] --> UBI + UI --> UCI + end + + style F fill:#f9f,stroke:#333,stroke-width:2px + style P fill:#f9f,stroke:#333,stroke-width:2px + style R fill:#bbf,stroke:#333,stroke-width:2px + style DB fill:#dfd,stroke:#333,stroke-width:2px + style J fill:#ffd,stroke:#333,stroke-width:2px + style Z fill:#ffd,stroke:#333,stroke-width:2px + style M fill:#faa,stroke:#333,stroke-width:4px +``` diff --git a/tests/trace/test_base_object_classes.py b/tests/trace/test_base_object_classes.py new file mode 100644 index 000000000000..a264941f7b00 --- /dev/null +++ b/tests/trace/test_base_object_classes.py @@ -0,0 +1,366 @@ +""" +This test file ensures the base_object_classes behavior is as expected. Specifically: +1. We ensure that pythonic publishing and getting of objects: + a. Results in the correct base_object_class filter in the query. + b. Produces identical results. +2. We ensure that using the low-level interface: + a. Results in the correct base_object_class filter in the query. + b. Produces identical results. +3. We ensure that digests are equivalent between pythonic and interface style creation. + This is important to ensure that UI-based generation of objects is consistent with + programmatic generation. +4. We ensure that invalid schemas are properly rejected from the server. +""" + +from typing import Literal, Optional + +import pytest +from pydantic import ValidationError + +import weave +from weave.trace import base_objects +from weave.trace.refs import ObjectRef +from weave.trace.weave_client import WeaveClient +from weave.trace_server import trace_server_interface as tsi +from weave.trace_server.interface.base_object_classes.test_only_example import ( + TestOnlyNestedBaseModel, +) + + +def with_base_object_class_annotations( + val: dict, + class_name: str, + base_object_name: Optional[Literal["Object", "BaseObject"]] = None, +): + """ + When serializing pydantic objects, add additional fields to indicate the class information. This is + a utlity to perform that mapping for the purposes of testing. We want to ensure that both the client + and server agree on this structure, therefore I am adding this utility here. + """ + bases = ["BaseModel"] + if base_object_name is not None: + bases.insert(0, base_object_name) + return { + **val, + "_type": class_name, + "_class_name": class_name, + "_bases": bases, + } + + +def test_pythonic_creation(client: WeaveClient): + # First, let's use the high-level pythonic creation API. + nested_obj = base_objects.TestOnlyNestedBaseObject(b=3) + top_obj = base_objects.TestOnlyExample( + primitive=1, + nested_base_model=TestOnlyNestedBaseModel(a=2), + nested_base_object=weave.publish(nested_obj).uri(), + ) + ref = weave.publish(top_obj) + + top_obj_gotten = weave.ref(ref.uri()).get() + + assert isinstance(top_obj_gotten, base_objects.TestOnlyExample) + assert top_obj_gotten.model_dump() == top_obj.model_dump() + + objs_res = client.server.objs_query( + tsi.ObjQueryReq.model_validate( + { + "project_id": client._project_id(), + "filter": {"base_object_classes": ["TestOnlyExample"]}, + }, + ) + ) + objs = objs_res.objs + + assert len(objs) == 1 + assert ( + objs[0].val + == { + **with_base_object_class_annotations( + top_obj.model_dump(), "TestOnlyExample", "BaseObject" + ), + "nested_base_model": with_base_object_class_annotations( + top_obj.nested_base_model.model_dump(), "TestOnlyNestedBaseModel" + ), + } + == { + "_type": "TestOnlyExample", + "name": None, + "description": None, + "primitive": 1, + "nested_base_model": { + "_type": "TestOnlyNestedBaseModel", + "a": 2, + "_class_name": "TestOnlyNestedBaseModel", + "_bases": ["BaseModel"], + }, + "nested_base_object": "weave:///shawn/test-project/object/TestOnlyNestedBaseObject:JyFvHfyaJ79uCKpdZ3DD3if4NYam8QgTkzUlXQXAILI", + "_class_name": "TestOnlyExample", + "_bases": ["BaseObject", "BaseModel"], + } + ) + + objs_res = client.server.objs_query( + tsi.ObjQueryReq.model_validate( + { + "project_id": client._project_id(), + "filter": {"base_object_classes": ["TestOnlyNestedBaseObject"]}, + }, + ) + ) + objs = objs_res.objs + + assert len(objs) == 1 + assert ( + objs[0].val + == with_base_object_class_annotations( + nested_obj.model_dump(), "TestOnlyNestedBaseObject", "BaseObject" + ) + == { + "_type": "TestOnlyNestedBaseObject", + "name": None, + "description": None, + "b": 3, + "_class_name": "TestOnlyNestedBaseObject", + "_bases": ["BaseObject", "BaseModel"], + } + ) + + +def test_interface_creation(client): + # Now we will do the equivant operation using low-level interface. + nested_obj_id = "TestOnlyNestedBaseObject" + nested_obj = base_objects.TestOnlyNestedBaseObject(b=3) + nested_obj_res = client.server.obj_create( + tsi.ObjCreateReq.model_validate( + { + "obj": { + "project_id": client._project_id(), + "object_id": nested_obj_id, + "val": nested_obj.model_dump(), + "set_base_object_class": "TestOnlyNestedBaseObject", + } + } + ) + ) + nested_obj_ref = ObjectRef( + entity=client.entity, + project=client.project, + name=nested_obj_id, + _digest=nested_obj_res.digest, + ) + + top_level_obj_id = "TestOnlyExample" + top_obj = base_objects.TestOnlyExample( + primitive=1, + nested_base_model=TestOnlyNestedBaseModel(a=2), + nested_base_object=nested_obj_ref.uri(), + ) + top_obj_res = client.server.obj_create( + tsi.ObjCreateReq.model_validate( + { + "obj": { + "project_id": client._project_id(), + "object_id": top_level_obj_id, + "val": top_obj.model_dump(), + "set_base_object_class": "TestOnlyExample", + } + } + ) + ) + top_obj_ref = ObjectRef( + entity=client.entity, + project=client.project, + name=top_level_obj_id, + _digest=top_obj_res.digest, + ) + + top_obj_gotten = weave.ref(top_obj_ref.uri()).get() + + assert top_obj_gotten.model_dump() == top_obj.model_dump() + + nested_obj_gotten = weave.ref(nested_obj_ref.uri()).get() + + assert nested_obj_gotten.model_dump() == nested_obj.model_dump() + + objs_res = client.server.objs_query( + tsi.ObjQueryReq.model_validate( + { + "project_id": client._project_id(), + "filter": {"base_object_classes": ["TestOnlyExample"]}, + }, + ) + ) + + objs = objs_res.objs + assert len(objs) == 1 + assert ( + objs[0].val + == { + **with_base_object_class_annotations( + top_obj.model_dump(), "TestOnlyExample", "BaseObject" + ), + "nested_base_model": with_base_object_class_annotations( + top_obj.nested_base_model.model_dump(), "TestOnlyNestedBaseModel" + ), + } + == { + "_type": "TestOnlyExample", + "name": None, + "description": None, + "primitive": 1, + "nested_base_model": { + "_type": "TestOnlyNestedBaseModel", + "a": 2, + "_class_name": "TestOnlyNestedBaseModel", + "_bases": ["BaseModel"], + }, + "nested_base_object": "weave:///shawn/test-project/object/TestOnlyNestedBaseObject:JyFvHfyaJ79uCKpdZ3DD3if4NYam8QgTkzUlXQXAILI", + "_class_name": "TestOnlyExample", + "_bases": ["BaseObject", "BaseModel"], + } + ) + + objs_res = client.server.objs_query( + tsi.ObjQueryReq.model_validate( + { + "project_id": client._project_id(), + "filter": {"base_object_classes": ["TestOnlyNestedBaseObject"]}, + }, + ) + ) + objs = objs_res.objs + assert len(objs) == 1 + assert ( + objs[0].val + == with_base_object_class_annotations( + nested_obj.model_dump(), "TestOnlyNestedBaseObject", "BaseObject" + ) + == { + "_type": "TestOnlyNestedBaseObject", + "name": None, + "description": None, + "b": 3, + "_class_name": "TestOnlyNestedBaseObject", + "_bases": ["BaseObject", "BaseModel"], + } + ) + + +def test_digest_equality(client): + # Next, let's make sure that the digests are all equivalent + nested_obj = base_objects.TestOnlyNestedBaseObject(b=3) + nested_ref = weave.publish(nested_obj) + top_obj = base_objects.TestOnlyExample( + primitive=1, + nested_base_model=TestOnlyNestedBaseModel(a=2), + nested_base_object=nested_ref.uri(), + ) + ref = weave.publish(top_obj) + nested_pythonic_digest = nested_ref.digest + top_level_pythonic_digest = ref.digest + + # Now we will do the equivant operation using low-level interface. + nested_obj_id = "TestOnlyNestedBaseObject" + nested_obj = base_objects.TestOnlyNestedBaseObject(b=3) + nested_obj_res = client.server.obj_create( + tsi.ObjCreateReq.model_validate( + { + "obj": { + "project_id": client._project_id(), + "object_id": nested_obj_id, + "val": nested_obj.model_dump(), + "set_base_object_class": "TestOnlyNestedBaseObject", + } + } + ) + ) + nested_obj_ref = ObjectRef( + entity=client.entity, + project=client.project, + name=nested_obj_id, + _digest=nested_obj_res.digest, + ) + + nested_interface_style_digest = nested_obj_ref.digest + + assert nested_pythonic_digest == nested_interface_style_digest + + top_level_obj_id = "TestOnlyExample" + top_obj = base_objects.TestOnlyExample( + primitive=1, + nested_base_model=TestOnlyNestedBaseModel(a=2), + nested_base_object=nested_obj_ref.uri(), + ) + top_obj_res = client.server.obj_create( + tsi.ObjCreateReq.model_validate( + { + "obj": { + "project_id": client._project_id(), + "object_id": top_level_obj_id, + "val": top_obj.model_dump(), + "set_base_object_class": "TestOnlyExample", + } + } + ) + ) + + top_level_interface_style_digest = top_obj_res.digest + + assert top_level_pythonic_digest == top_level_interface_style_digest + + +def test_schema_validation(client): + # Test that we can't create an object with the wrong schema + with pytest.raises(ValidationError): + client.server.obj_create( + tsi.ObjCreateReq.model_validate( + { + "obj": { + "project_id": client._project_id(), + "object_id": "nested_obj", + # Incorrect schema, should raise! + "val": {"a": 2}, + "set_base_object_class": "TestOnlyNestedBaseObject", + } + } + ) + ) + + # Correct schema, should work + client.server.obj_create( + tsi.ObjCreateReq.model_validate( + { + "obj": { + "project_id": client._project_id(), + "object_id": "nested_obj", + "val": { + "b": 2, + "_class_name": "TestOnlyNestedBaseObject", + "_bases": ["BaseObject", "BaseModel"], + }, + "set_base_object_class": "TestOnlyNestedBaseObject", + } + } + ) + ) + + with pytest.raises(ValueError): + # Mismatching base object class, should raise + client.server.obj_create( + tsi.ObjCreateReq.model_validate( + { + "obj": { + "project_id": client._project_id(), + "object_id": "nested_obj", + "val": { + "b": 2, + "_class_name": "TestOnlyNestedBaseObject", + "_bases": ["BaseObject", "BaseModel"], + }, + "set_base_object_class": "TestOnlyExample", + } + } + ) + ) diff --git a/weave-js/package.json b/weave-js/package.json index d925f9d0a423..efd9a5100326 100644 --- a/weave-js/package.json +++ b/weave-js/package.json @@ -19,8 +19,10 @@ "generate:watch": "graphql-codegen -w", "prettier": "prettier --config .prettierrc --check \"src/**/*.ts\" \"src/**/*.tsx\"", "prettier-fix": "prettier --loglevel warn --config .prettierrc --write \"src/**/*.ts\" \"src/**/*.tsx\"", + "direct-prettier": "prettier", "lint": "yarn eslint & yarn tslint & yarn prettier & wait", - "lint-fix": "yarn eslint-fix & yarn tslint-fix & yarn prettier-fix & wait" + "lint-fix": "yarn eslint-fix & yarn tslint-fix & yarn prettier-fix & wait", + "generate-schemas": "bash scripts/generate-schemas.sh" }, "dependencies": { "@apollo/client": "^3.8.4", @@ -145,7 +147,8 @@ "wavesurfer.js": "^2.0.0", "web-tree-sitter": "^0.20.5", "yet-another-react-lightbox": "^3.17.5", - "zen-observable": "^0.10.0" + "zen-observable": "^0.10.0", + "zod": "^3.23.8" }, "devDependencies": { "@babel/core": "^7.23.2", @@ -217,11 +220,13 @@ "identity-obj-proxy": "^3.0.0", "jsdom": "^22.1.0", "json-schema-to-typescript": "^11.0.2", + "json-schema-to-zod": "^2.4.1", "less": "^2.7.3", "lodash.defaults": "^4.2.0", "nodemon": "^2.0.22", "prettier": "^2.8.7", "prettier-plugin-tailwindcss": "^0.2.1", + "quicktype": "^23.0.170", "rimraf": "^3.0.2", "rollup-plugin-visualizer": "^5.5.2", "tailwindcss": "^3.3.2", @@ -233,7 +238,8 @@ "typescript": "4.7.4", "uuid": "^9.0.0", "vite": "5.2.9", - "vitest": "^1.6.0" + "vitest": "^1.6.0", + "tsd": "^0.30.4" }, "resolutions": { "@types/react": "^17.0.26", diff --git a/weave-js/scripts/generate-schemas.sh b/weave-js/scripts/generate-schemas.sh new file mode 100644 index 000000000000..545d81e17afb --- /dev/null +++ b/weave-js/scripts/generate-schemas.sh @@ -0,0 +1,41 @@ +#!/bin/bash + +# Exit on error +set -e + +SCHEMA_INPUT_PATH="../weave/trace_server/interface/base_object_classes/generated/generated_base_object_class_schemas.json" +SCHEMA_OUTPUT_PATH="./src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/generatedBaseObjectClasses.zod.ts" + +echo "Generating schemas..." + +# Generate TypeScript-Zod types from schema +yarn quicktype -s schema "$SCHEMA_INPUT_PATH" -o "$SCHEMA_OUTPUT_PATH" --lang typescript-zod + +# Transform the schema to extract the type map +sed -i.bak ' + # Find the GeneratedBaseObjectClassesZodSchema definition and capture its contents + /export const GeneratedBaseObjectClassesZodSchema = z.object({/,/});/ { + # Replace the opening line with typeMap declaration + s/export const GeneratedBaseObjectClassesZodSchema = z.object({/export const baseObjectClassRegistry = ({/ + # Store the pattern + h + # If this is the last line (with closing brace), append the schema definition + /});/ { + p + s/.*// + x + s/.*// + i\ +\ +export const GeneratedBaseObjectClassesZodSchema = z.object(baseObjectClassRegistry) + } + } +' "$SCHEMA_OUTPUT_PATH" + +# Remove backup file +rm "${SCHEMA_OUTPUT_PATH}.bak" + +# Format the generated file +yarn direct-prettier --write "$SCHEMA_OUTPUT_PATH" + +echo "Schema generation completed successfully" \ No newline at end of file diff --git a/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/baseObjectClassQuery.test.ts b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/baseObjectClassQuery.test.ts new file mode 100644 index 000000000000..9918ae7f2857 --- /dev/null +++ b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/baseObjectClassQuery.test.ts @@ -0,0 +1,105 @@ +import {expectType} from 'tsd'; + +import { + useBaseObjectInstances, + useCreateBaseObjectInstance, +} from './baseObjectClassQuery'; +import { + TestOnlyExample, + TestOnlyExampleSchema, +} from './generatedBaseObjectClasses.zod'; +import { + TraceObjCreateReq, + TraceObjCreateRes, + TraceObjSchema, +} from './traceServerClientTypes'; +import {Loadable} from './wfDataModelHooksInterface'; + +type TypesAreEqual = [T] extends [U] + ? [U] extends [T] + ? true + : false + : false; + +describe('Type Tests', () => { + it('useCollectionObjects return type matches expected structure', () => { + type CollectionObjectsReturn = ReturnType< + typeof useBaseObjectInstances<'TestOnlyExample'> + >; + + // Define the expected type structure + type ExpectedType = Loadable< + Array> + >; + + // Type assertion tests + type AssertTypesAreEqual = TypesAreEqual< + CollectionObjectsReturn, + ExpectedType + >; + type Assert = AssertTypesAreEqual extends true ? true : never; + + // This will fail compilation if the types don't match exactly + const _assert: Assert = true; + expect(_assert).toBe(true); + + // Additional runtime sample for documentation + const sampleResult: CollectionObjectsReturn = { + loading: false, + result: [ + { + project_id: '', + object_id: '', + created_at: '', + digest: '', + version_index: 0, + is_latest: 0, + kind: 'object', + base_object_class: 'TestOnlyExample', + val: TestOnlyExampleSchema.parse({ + name: '', + description: '', + nested_base_model: { + a: 1, + }, + nested_base_object: '', + primitive: 1, + }), + }, + ], + }; + + expectType(sampleResult); + }); + + it('useCreateCollectionObject return type matches expected structure', () => { + type CreateCollectionObjectReturn = ReturnType< + typeof useCreateBaseObjectInstance<'TestOnlyExample'> + >; + + // Define the expected type structure + type ExpectedType = ( + req: TraceObjCreateReq + ) => Promise; + + // Type assertion tests + type AssertTypesAreEqual = TypesAreEqual< + CreateCollectionObjectReturn, + ExpectedType + >; + type Assert = AssertTypesAreEqual extends true ? true : never; + + // This will fail compilation if the types don't match exactly + const _assert: Assert = true; + expect(_assert).toBe(true); + + // Additional runtime sample for documentation + const sampleResult: CreateCollectionObjectReturn = async req => { + return { + digest: '', + }; + }; + + expectType(sampleResult); + }); +}); diff --git a/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/baseObjectClassQuery.ts b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/baseObjectClassQuery.ts new file mode 100644 index 000000000000..5906d52ad6fe --- /dev/null +++ b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/baseObjectClassQuery.ts @@ -0,0 +1,142 @@ +import {useDeepMemo} from '@wandb/weave/hookUtils'; +import {useEffect, useRef, useState} from 'react'; +import {z} from 'zod'; + +import {baseObjectClassRegistry} from './generatedBaseObjectClasses.zod'; +import {TraceServerClient} from './traceServerClient'; +import {useGetTraceServerClientContext} from './traceServerClientContext'; +import { + TraceObjCreateReq, + TraceObjCreateRes, + TraceObjQueryReq, + TraceObjSchema, +} from './traceServerClientTypes'; +import {Loadable} from './wfDataModelHooksInterface'; + +type BaseObjectClassRegistry = typeof baseObjectClassRegistry; +type BaseObjectClassRegistryKeys = keyof BaseObjectClassRegistry; +type BaseObjectClassType = z.infer< + BaseObjectClassRegistry[C] +>; + +export const useBaseObjectInstances = ( + baseObjectClassName: C, + req: TraceObjQueryReq +): Loadable, C>>> => { + const [objects, setObjects] = useState< + Array, C>> + >([]); + const getTsClient = useGetTraceServerClientContext(); + const client = getTsClient(); + const deepReq = useDeepMemo(req); + const currReq = useRef(deepReq); + const [loading, setLoading] = useState(true); + + useEffect(() => { + let isMounted = true; + setLoading(true); + currReq.current = deepReq; + getBaseObjectInstances(client, baseObjectClassName, deepReq).then( + collectionObjects => { + if (isMounted && currReq.current === deepReq) { + setObjects(collectionObjects); + setLoading(false); + } + } + ); + return () => { + isMounted = false; + }; + }, [client, baseObjectClassName, deepReq]); + + return {result: objects, loading}; +}; + +const getBaseObjectInstances = async ( + client: TraceServerClient, + baseObjectClassName: C, + req: TraceObjQueryReq +): Promise, C>>> => { + const knownObjectClass = baseObjectClassRegistry[baseObjectClassName]; + if (!knownObjectClass) { + console.warn(`Unknown object class: ${baseObjectClassName}`); + return []; + } + + const reqWithBaseObjectClass: TraceObjQueryReq = { + ...req, + filter: {...req.filter, base_object_classes: [baseObjectClassName]}, + }; + + const objectPromise = client.objsQuery(reqWithBaseObjectClass); + + const objects = await objectPromise; + + // We would expect that this filtering does not filter anything + // out because the backend enforces the base object class, but this + // is here as a sanity check. + return objects.objs + .map(obj => ({obj, parsed: knownObjectClass.safeParse(obj.val)})) + .filter(({parsed}) => parsed.success) + .filter(({obj}) => obj.base_object_class === baseObjectClassName) + .map( + ({obj, parsed}) => + ({...obj, val: parsed.data} as TraceObjSchema< + BaseObjectClassType, + C + >) + ); +}; + +export const useCreateBaseObjectInstance = < + C extends BaseObjectClassRegistryKeys, + T = BaseObjectClassType +>( + baseObjectClassName: C +): ((req: TraceObjCreateReq) => Promise) => { + const getTsClient = useGetTraceServerClientContext(); + const client = getTsClient(); + return (req: TraceObjCreateReq) => + createBaseObjectInstance(client, baseObjectClassName, req); +}; + +const createBaseObjectInstance = async < + C extends BaseObjectClassRegistryKeys, + T = BaseObjectClassType +>( + client: TraceServerClient, + baseObjectClassName: C, + req: TraceObjCreateReq +): Promise => { + if ( + req.obj.set_base_object_class != null && + req.obj.set_base_object_class !== baseObjectClassName + ) { + throw new Error( + `set_base_object_class must match baseObjectClassName: ${baseObjectClassName}` + ); + } + + const knownBaseObjectClass = baseObjectClassRegistry[baseObjectClassName]; + if (!knownBaseObjectClass) { + throw new Error(`Unknown object class: ${baseObjectClassName}`); + } + + const verifiedObject = knownBaseObjectClass.safeParse(req.obj.val); + + if (!verifiedObject.success) { + throw new Error( + `Invalid object: ${JSON.stringify(verifiedObject.error.errors)}` + ); + } + + const reqWithBaseObjectClass: TraceObjCreateReq = { + ...req, + obj: { + ...req.obj, + set_base_object_class: baseObjectClassName, + }, + }; + + return client.objCreate(reqWithBaseObjectClass); +}; diff --git a/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/generatedBaseObjectClasses.zod.ts b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/generatedBaseObjectClasses.zod.ts new file mode 100644 index 000000000000..fd27f0bc9339 --- /dev/null +++ b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/generatedBaseObjectClasses.zod.ts @@ -0,0 +1,39 @@ +import * as z from 'zod'; + +export const TestOnlyNestedBaseModelSchema = z.object({ + a: z.number(), +}); +export type TestOnlyNestedBaseModel = z.infer< + typeof TestOnlyNestedBaseModelSchema +>; + +export const TestOnlyNestedBaseObjectSchema = z.object({ + b: z.number(), + description: z.union([z.null(), z.string()]).optional(), + name: z.union([z.null(), z.string()]).optional(), +}); +export type TestOnlyNestedBaseObject = z.infer< + typeof TestOnlyNestedBaseObjectSchema +>; + +export const TestOnlyExampleSchema = z.object({ + description: z.union([z.null(), z.string()]).optional(), + name: z.union([z.null(), z.string()]).optional(), + nested_base_model: TestOnlyNestedBaseModelSchema, + nested_base_object: z.string(), + primitive: z.number(), +}); +export type TestOnlyExample = z.infer; + +export const baseObjectClassRegistry = { + TestOnlyExample: TestOnlyExampleSchema, + TestOnlyNestedBaseObject: TestOnlyNestedBaseObjectSchema, +}; + +export const GeneratedBaseObjectClassesZodSchema = z.object( + baseObjectClassRegistry +); + +export type GeneratedBaseObjectClassesZod = z.infer< + typeof GeneratedBaseObjectClassesZodSchema +>; diff --git a/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/traceServerClientTypes.ts b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/traceServerClientTypes.ts index 88113a37a74a..d7219af58343 100644 --- a/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/traceServerClientTypes.ts +++ b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/traceServerClientTypes.ts @@ -205,7 +205,10 @@ export type TraceObjQueryReq = { metadata_only?: boolean; }; -export interface TraceObjSchema { +export interface TraceObjSchema< + T extends any = any, + OBC extends string = string +> { project_id: string; object_id: string; created_at: string; @@ -213,12 +216,14 @@ export interface TraceObjSchema { version_index: number; is_latest: number; kind: 'op' | 'object'; - base_object_class?: string; - val: any; + base_object_class?: OBC; + val: T; } -export type TraceObjQueryRes = { - objs: TraceObjSchema[]; + +export type TraceObjQueryRes = { + objs: Array>; }; + export type TraceObjReadReq = { project_id: string; object_id: string; @@ -229,6 +234,19 @@ export type TraceObjReadRes = { obj: TraceObjSchema; }; +export type TraceObjCreateReq = { + obj: { + project_id: string; + object_id: string; + val: T; + set_base_object_class?: string; + }; +}; + +export type TraceObjCreateRes = { + digest: string; +}; + export type TraceRefsReadBatchReq = { refs: string[]; }; diff --git a/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/traceServerDirectClient.ts b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/traceServerDirectClient.ts index caaf63b7f561..25f11624740e 100644 --- a/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/traceServerDirectClient.ts +++ b/weave-js/src/components/PagePanelComponents/Home/Browse3/pages/wfReactInterface/traceServerDirectClient.ts @@ -34,6 +34,8 @@ import { TraceCallUpdateReq, TraceFileContentReadReq, TraceFileContentReadRes, + TraceObjCreateReq, + TraceObjCreateRes, TraceObjQueryReq, TraceObjQueryRes, TraceObjReadReq, @@ -231,6 +233,22 @@ export class DirectTraceServerClient { ); } + public objCreate(req: TraceObjCreateReq): Promise { + const initialObjectId = req.obj.object_id; + const sanitizedObjectId = sanitizeObjectId(initialObjectId); + if (sanitizedObjectId !== initialObjectId) { + // Caller is expected to sanitize the object id. We should be doing this + // on the server, but it is currently disabled. + throw new Error( + `Invalid object name: ${initialObjectId}, sanitized to ${sanitizedObjectId}` + ); + } + return this.makeRequest( + '/obj/create', + req + ); + } + public tableQuery(req: TraceTableQueryReq): Promise { return this.makeRequest( '/table/query', @@ -376,3 +394,30 @@ export class DirectTraceServerClient { return prom; }; } + +/** + * Sanitizes an object name by replacing non-alphanumeric characters with dashes and enforcing length limits. + * This matches the Python implementation in weave_client.py. + * + * @param name The name to sanitize + * @returns The sanitized name + * @throws Error if the resulting name would be empty + */ +export function sanitizeObjectId(name: string): string { + // Replace any non-word chars (except dots and underscores) with dashes + let res = name.replace(/[^\w._]+/g, '-'); + // Replace multiple consecutive dashes/dots/underscores with a single dash + res = res.replace(/([._-]{2,})+/g, '-'); + // Remove leading/trailing dashes and underscores + res = res.replace(/^[-_]+|[-_]+$/g, ''); + + if (!res) { + throw new Error(`Invalid object name: ${name}`); + } + + if (res.length > 128) { + res = res.slice(0, 128); + } + + return res; +} diff --git a/weave-js/yarn.lock b/weave-js/yarn.lock index 2ee205532570..1a0ce3ab43e5 100644 --- a/weave-js/yarn.lock +++ b/weave-js/yarn.lock @@ -2119,6 +2119,16 @@ resolved "https://registry.yarnpkg.com/@floating-ui/utils/-/utils-0.2.5.tgz#105c37d9d9620ce69b7f692a20c821bf1ad2cbf9" integrity sha512-sTcG+QZ6fdEUObICavU+aB3Mp8HY4n14wYHdxK4fXjPmv3PXZZeY5RaguJmGyeH/CJQhX3fqKUtS4qc1LoHwhQ== +"@glideapps/ts-necessities@2.2.3": + version "2.2.3" + resolved "https://registry.yarnpkg.com/@glideapps/ts-necessities/-/ts-necessities-2.2.3.tgz#62e25b3a1ace8b8c3f47e55e66d101a0a854eb23" + integrity sha512-gXi0awOZLHk3TbW55GZLCPP6O+y/b5X1pBXKBVckFONSwF1z1E5ND2BGJsghQFah+pW7pkkyFb2VhUQI2qhL5w== + +"@glideapps/ts-necessities@^2.2.3": + version "2.3.2" + resolved "https://registry.yarnpkg.com/@glideapps/ts-necessities/-/ts-necessities-2.3.2.tgz#3e7a07f41c8c07527757631f25599a7b67d39d8c" + integrity sha512-tOXo3SrEeLu+4X2q6O2iNPXdGI1qoXEz/KrbkElTsWiWb69tFH4GzWz2K++0nBD6O3qO2Ft1C4L4ZvUfE2QDlQ== + "@graphql-codegen/add@^5.0.0": version "5.0.0" resolved "https://registry.yarnpkg.com/@graphql-codegen/add/-/add-5.0.0.tgz#578ebaf4fa87c1e934c381cd679bcedcf79feaba" @@ -2774,6 +2784,20 @@ resolved "https://registry.yarnpkg.com/@mapbox/whoots-js/-/whoots-js-3.1.0.tgz#497c67a1cef50d1a2459ba60f315e448d2ad87fe" integrity sha512-Es6WcD0nO5l+2BOQS4uLfNPYQaNDfbot3X1XUoloz+x0mPDS3eeORZJl06HXjwBG1fOGwCRnzK88LMdxKRrd6Q== +"@mark.probst/typescript-json-schema@0.55.0": + version "0.55.0" + resolved "https://registry.yarnpkg.com/@mark.probst/typescript-json-schema/-/typescript-json-schema-0.55.0.tgz#a82c0cb8b3c9ba1a14faf2ea3fa95f26c1a6a57d" + integrity sha512-jI48mSnRgFQxXiE/UTUCVCpX8lK3wCFKLF1Ss2aEreboKNuLQGt3e0/YFqWVHe/WENxOaqiJvwOz+L/SrN2+qQ== + dependencies: + "@types/json-schema" "^7.0.9" + "@types/node" "^16.9.2" + glob "^7.1.7" + path-equal "^1.1.2" + safe-stable-stringify "^2.2.0" + ts-node "^10.9.1" + typescript "4.9.4" + yargs "^17.1.1" + "@material-ui/core@^4.12.4": version "4.12.4" resolved "https://registry.yarnpkg.com/@material-ui/core/-/core-4.12.4.tgz#4ac17488e8fcaf55eb6a7f5efb2a131e10138a73" @@ -4126,6 +4150,11 @@ resolved "https://registry.yarnpkg.com/@tsconfig/node16/-/node16-1.0.4.tgz#0b92dcc0cc1c81f6f306a381f28e31b1a56536e9" integrity sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA== +"@tsd/typescript@~5.3.3": + version "5.3.3" + resolved "https://registry.yarnpkg.com/@tsd/typescript/-/typescript-5.3.3.tgz#bc01854b6e0e746b5f70a6b48c30c7b95b81a74e" + integrity sha512-CQlfzol0ldaU+ftWuG52vH29uRoKboLinLy84wS8TQOu+m+tWoaUfk4svL4ij2V8M5284KymJBlHUusKj6k34w== + "@turf/area@^6.4.0": version "6.5.0" resolved "https://registry.yarnpkg.com/@turf/area/-/area-6.5.0.tgz#1d0d7aee01d8a4a3d4c91663ed35cc615f36ad56" @@ -4237,6 +4266,19 @@ resolved "https://registry.yarnpkg.com/@types/downloadjs/-/downloadjs-1.4.3.tgz#01c0414a9756afa2bd438b1e2ec6c50de7df316d" integrity sha512-MjJepFle/tLtT2/jmDNth6ZnwWzEhm40L+olE5HKR70ISUCfgT55eqreeHldAzFLY2HDUGsn8zgyto8KygN0CA== +"@types/eslint@^7.2.13": + version "7.29.0" + resolved "https://registry.yarnpkg.com/@types/eslint/-/eslint-7.29.0.tgz#e56ddc8e542815272720bb0b4ccc2aff9c3e1c78" + integrity sha512-VNcvioYDH8/FxaeTKkM4/TiTwt6pBV9E3OfGmvaw8tPl0rrHCJ4Ll15HRT+pMiFAf/MLQvAzC+6RzUMEL9Ceng== + dependencies: + "@types/estree" "*" + "@types/json-schema" "*" + +"@types/estree@*": + version "1.0.6" + resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.6.tgz#628effeeae2064a1b4e79f78e81d87b7e5fc7b50" + integrity sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw== + "@types/estree@1.0.5": version "1.0.5" resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.5.tgz#a6ce3e556e00fd9895dd872dd172ad0d4bd687f4" @@ -4339,6 +4381,11 @@ resolved "https://registry.yarnpkg.com/@types/js-yaml/-/js-yaml-4.0.6.tgz#4b3afd5158b8749095b1f096967b6d0f838d862f" integrity sha512-ACTuifTSIIbyksx2HTon3aFtCKWcID7/h3XEmRpDYdMCXxPbl+m9GteOJeaAkiAta/NJaSFuA7ahZ0NkwajDSw== +"@types/json-schema@*": + version "7.0.15" + resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.15.tgz#596a1747233694d50f6ad8a7869fcb6f56cf5841" + integrity sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA== + "@types/json-schema@^7.0.11", "@types/json-schema@^7.0.6", "@types/json-schema@^7.0.9": version "7.0.12" resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.12.tgz#d70faba7039d5fca54c83c7dbab41051d2b6f6cb" @@ -4383,6 +4430,11 @@ resolved "https://registry.yarnpkg.com/@types/minimatch/-/minimatch-5.1.2.tgz#07508b45797cb81ec3f273011b054cd0755eddca" integrity sha512-K0VQKziLUWkVKiRVrx4a40iPaxTUefQmjtkQofBkYRcoaaL/8rhwDWww9qWbrgicNOgnpIsMxyNIUM4+n6dUIA== +"@types/minimist@^1.2.0": + version "1.2.5" + resolved "https://registry.yarnpkg.com/@types/minimist/-/minimist-1.2.5.tgz#ec10755e871497bcd83efe927e43ec46e8c0747e" + integrity sha512-hov8bUuiLiyFPGyFPE1lwWhmzYbirOXQNNo40+y3zow8aFVTeyn3VWL0VFFfdNddA8S4Vf0Tc062rzyNr7Paag== + "@types/ms@*": version "0.7.31" resolved "https://registry.yarnpkg.com/@types/ms/-/ms-0.7.31.tgz#31b7ca6407128a3d2bbc27fe2d21b345397f6197" @@ -4398,11 +4450,21 @@ resolved "https://registry.yarnpkg.com/@types/node/-/node-14.18.53.tgz#42855629b8773535ab868238718745bf56c56219" integrity sha512-soGmOpVBUq+gaBMwom1M+krC/NNbWlosh4AtGA03SyWNDiqSKtwp7OulO1M6+mg8YkHMvJ/y0AkCeO8d1hNb7A== +"@types/node@^16.9.2": + version "16.18.116" + resolved "https://registry.yarnpkg.com/@types/node/-/node-16.18.116.tgz#138a0ce907c9f308d43f89902a9ef79fbcbc5e5c" + integrity sha512-mLigUvhoaADRewggiby+XfAAFOUOMCm/SwL5DAJ+CMUGjSLIGMsJVN7BOKftuQSHGjUmS/W7hVht8fcNbi/MRA== + "@types/node@^18.15.11": version "18.16.19" resolved "https://registry.yarnpkg.com/@types/node/-/node-18.16.19.tgz#cb03fca8910fdeb7595b755126a8a78144714eea" integrity sha512-IXl7o+R9iti9eBW4Wg2hx1xQDig183jj7YLn8F7udNceyfkbn1ZxmzZXuak20gR40D7pIkIY1kYGx5VIGbaHKA== +"@types/normalize-package-data@^2.4.0": + version "2.4.4" + resolved "https://registry.yarnpkg.com/@types/normalize-package-data/-/normalize-package-data-2.4.4.tgz#56e2cc26c397c038fab0e3a917a12d5c5909e901" + integrity sha512-37i+OaWTh9qeK4LSHPsyRC7NahnGotNuZvjLSgcPzblpHB3rrCJxAOgI5gCdKm7coonsaX1Of0ILiTcnZjbfxA== + "@types/numeral@^2.0.2": version "2.0.2" resolved "https://registry.yarnpkg.com/@types/numeral/-/numeral-2.0.2.tgz#8ea2c4f4e64c0cc948ad7da375f6f827778a7912" @@ -4997,6 +5059,13 @@ abbrev@1: resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.1.tgz#f8f2c887ad10bf67f634f005b6987fed3179aac8" integrity sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q== +abort-controller@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/abort-controller/-/abort-controller-3.0.0.tgz#eaf54d53b62bae4138e809ca225c8439a6efb392" + integrity sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg== + dependencies: + event-target-shim "^5.0.0" + abs-svg-path@^0.1.1, abs-svg-path@~0.1.1: version "0.1.1" resolved "https://registry.yarnpkg.com/abs-svg-path/-/abs-svg-path-0.1.1.tgz#df601c8e8d2ba10d4a76d625e236a9a39c2723bf" @@ -5195,6 +5264,16 @@ aria-query@^5.0.0, aria-query@^5.1.3: dependencies: dequal "^2.0.3" +array-back@^3.0.1, array-back@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/array-back/-/array-back-3.1.0.tgz#b8859d7a508871c9a7b2cf42f99428f65e96bfb0" + integrity sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q== + +array-back@^6.2.2: + version "6.2.2" + resolved "https://registry.yarnpkg.com/array-back/-/array-back-6.2.2.tgz#f567d99e9af88a6d3d2f9dfcc21db6f9ba9fd157" + integrity sha512-gUAZ7HPyb4SJczXAMUXMGAvI976JoK3qEx9v1FTmeYuJj0IBiaKttG1ydtGKdkfqWkIkouke7nG8ufGy77+Cvw== + array-bounds@^1.0.0, array-bounds@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/array-bounds/-/array-bounds-1.0.1.tgz#da11356b4e18e075a4f0c86e1f179a67b7d7ea31" @@ -5277,6 +5356,11 @@ array.prototype.tosorted@^1.1.1: es-shim-unscopables "^1.0.0" get-intrinsic "^1.1.3" +arrify@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/arrify/-/arrify-1.0.1.tgz#898508da2226f380df904728456849c1501a4b0d" + integrity sha512-3CYzex9M9FGQjCGMGyi6/31c8GJbgb0qGyrx5HWxPd0aCwh4cB2YjMb2Xf9UuoogrMrlO9cTqnB5rI5GHZTcUA== + asap@~2.0.3, asap@~2.0.6: version "2.0.6" resolved "https://registry.yarnpkg.com/asap/-/asap-2.0.6.tgz#e50347611d7e690943208bbdafebcbc2fb866d46" @@ -5562,7 +5646,7 @@ balanced-match@^1.0.0: resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== -base64-js@^1.3.1: +base64-js@^1.3.0, base64-js@^1.3.1: version "1.5.1" resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a" integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA== @@ -5645,6 +5729,11 @@ braces@^3.0.2, braces@~3.0.2: dependencies: fill-range "^7.0.1" +browser-or-node@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/browser-or-node/-/browser-or-node-3.0.0.tgz#2b11335570b28887e0bf5cd857f2e8062c6ae293" + integrity sha512-iczIdVJzGEYhP5DqQxYM9Hh7Ztpqqi+CXZpSmX8ALFs9ecXkQIeqRyM6TfxEfMVpwhl3dSuDvxdzzo9sUOIVBQ== + browserify-package-json@^1.0.0: version "1.0.1" resolved "https://registry.yarnpkg.com/browserify-package-json/-/browserify-package-json-1.0.1.tgz#98dde8aa5c561fd6d3fe49bbaa102b74b396fdea" @@ -5687,6 +5776,14 @@ buffer@^5.5.0: base64-js "^1.3.1" ieee754 "^1.1.13" +buffer@^6.0.3: + version "6.0.3" + resolved "https://registry.yarnpkg.com/buffer/-/buffer-6.0.3.tgz#2ace578459cc8fbe2a70aaa8f52ee63b6a74c6c6" + integrity sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA== + dependencies: + base64-js "^1.3.1" + ieee754 "^1.2.1" + builtin-modules@^1.1.1: version "1.1.1" resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-1.1.1.tgz#270f076c5a72c02f5b65a47df94c5fe3a278892f" @@ -5744,7 +5841,16 @@ camelcase-css@^2.0.1: resolved "https://registry.yarnpkg.com/camelcase-css/-/camelcase-css-2.0.1.tgz#ee978f6947914cc30c6b44741b6ed1df7f043fd5" integrity sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA== -camelcase@^5.0.0: +camelcase-keys@^6.2.2: + version "6.2.2" + resolved "https://registry.yarnpkg.com/camelcase-keys/-/camelcase-keys-6.2.2.tgz#5e755d6ba51aa223ec7d3d52f25778210f9dc3c0" + integrity sha512-YrwaA0vEKazPBkn0ipTiMpSajYDSe+KjQfrjhcBMxJt/znbvlHd8Pw/Vamaz5EB4Wfhs3SUR3Z9mwRu/P3s3Yg== + dependencies: + camelcase "^5.3.1" + map-obj "^4.0.0" + quick-lru "^4.0.1" + +camelcase@^5.0.0, camelcase@^5.3.1: version "5.3.1" resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-5.3.1.tgz#e3c9b31569e106811df242f715725a1f4c494320" integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg== @@ -5817,6 +5923,13 @@ chai@^4.3.10: pathval "^1.1.1" type-detect "^4.0.8" +chalk-template@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/chalk-template/-/chalk-template-0.4.0.tgz#692c034d0ed62436b9062c1707fadcd0f753204b" + integrity sha512-/ghrgmhfY8RaSdeo43hNXxpoHAtxdbskUHjPpfqUWGttFgycUhYPGx3YZBCnUCvOa7Doivn1IZec3DEGFoMgLg== + dependencies: + chalk "^4.1.2" + chalk@^2.0.0, chalk@^2.3.0, chalk@^2.4.2: version "2.4.2" resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424" @@ -5834,7 +5947,7 @@ chalk@^3.0.0: ansi-styles "^4.1.0" supports-color "^7.1.0" -chalk@^4.0.0, chalk@^4.1.0, chalk@^4.1.1: +chalk@^4.0.0, chalk@^4.1.0, chalk@^4.1.1, chalk@^4.1.2: version "4.1.2" resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== @@ -6047,6 +6160,11 @@ collapse-white-space@^1.0.2: resolved "https://registry.yarnpkg.com/collapse-white-space/-/collapse-white-space-1.0.6.tgz#e63629c0016665792060dbbeb79c42239d2c5287" integrity sha512-jEovNnrhMuqyCcjfEJA56v0Xq8SkIoPKDyaHahwo3POf4qcSXqMYuwNcOTzp74vTsR9Tn08z4MxWqAhcekogkQ== +collection-utils@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/collection-utils/-/collection-utils-1.0.1.tgz#31d14336488674f27aefc0a7c5eccacf6df78044" + integrity sha512-LA2YTIlR7biSpXkKYwwuzGjwL5rjWEZVOSnvdUc7gObvWe4WkjxOpfrdhoP7Hs09YWDVfg0Mal9BpAqLfVEzQg== + color-alpha@1.0.4: version "1.0.4" resolved "https://registry.yarnpkg.com/color-alpha/-/color-alpha-1.0.4.tgz#c141dc926e95fc3db647d0e14e5bc3651c29e040" @@ -6198,6 +6316,26 @@ comma-separated-tokens@^2.0.0: resolved "https://registry.yarnpkg.com/comma-separated-tokens/-/comma-separated-tokens-2.0.3.tgz#4e89c9458acb61bc8fef19f4529973b2392839ee" integrity sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg== +command-line-args@^5.2.1: + version "5.2.1" + resolved "https://registry.yarnpkg.com/command-line-args/-/command-line-args-5.2.1.tgz#c44c32e437a57d7c51157696893c5909e9cec42e" + integrity sha512-H4UfQhZyakIjC74I9d34fGYDwk3XpSr17QhEd0Q3I9Xq1CETHo4Hcuo87WyWHpAF1aSLjLRf5lD9ZGX2qStUvg== + dependencies: + array-back "^3.1.0" + find-replace "^3.0.0" + lodash.camelcase "^4.3.0" + typical "^4.0.0" + +command-line-usage@^7.0.1: + version "7.0.3" + resolved "https://registry.yarnpkg.com/command-line-usage/-/command-line-usage-7.0.3.tgz#6bce992354f6af10ecea2b631bfdf0c8b3bfaea3" + integrity sha512-PqMLy5+YGwhMh1wS04mVG44oqDsgyLRSKJBdOo1bnYhMKBW65gZF1dRp2OZRhiTjgUHljy99qkO7bsctLaw35Q== + dependencies: + array-back "^6.2.2" + chalk-template "^0.4.0" + table-layout "^4.1.0" + typical "^7.1.1" + commander@2, commander@^2.12.1, commander@^2.15.1, commander@^2.19.0: version "2.20.3" resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33" @@ -6437,6 +6575,13 @@ cross-fetch@^3.1.5: dependencies: node-fetch "^2.6.12" +cross-fetch@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/cross-fetch/-/cross-fetch-4.0.0.tgz#f037aef1580bb3a1a35164ea2a848ba81b445983" + integrity sha512-e4a5N8lVvuLgAWgnCrLr2PP0YyDOTHa9H/Rj54dirp61qXnNq46m82bRhNqIA5VccJtWBvPTFRV3TtvHUKPB1g== + dependencies: + node-fetch "^2.6.12" + cross-spawn@^7.0.1, cross-spawn@^7.0.2, cross-spawn@^7.0.3: version "7.0.3" resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6" @@ -6874,7 +7019,15 @@ debug@^3.1.0, debug@^3.2.6, debug@^3.2.7: dependencies: ms "^2.1.1" -decamelize@^1.2.0: +decamelize-keys@^1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/decamelize-keys/-/decamelize-keys-1.1.1.tgz#04a2d523b2f18d80d0158a43b895d56dff8d19d8" + integrity sha512-WiPxgEirIV0/eIOMcnFBA3/IJZAZqKnwAwWyvvdi4lsr1WCN22nhdf/3db3DoZcUjTV2SqfzIwNyp6y2xs3nmg== + dependencies: + decamelize "^1.1.0" + map-obj "^1.0.0" + +decamelize@^1.1.0, decamelize@^1.2.0: version "1.2.0" resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-1.2.0.tgz#f6534d15148269b20352e7bee26f501f9a191290" integrity sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA== @@ -7483,6 +7636,20 @@ eslint-config-react-app@^7.0.1: eslint-plugin-react-hooks "^4.3.0" eslint-plugin-testing-library "^5.0.1" +eslint-formatter-pretty@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/eslint-formatter-pretty/-/eslint-formatter-pretty-4.1.0.tgz#7a6877c14ffe2672066c853587d89603e97c7708" + integrity sha512-IsUTtGxF1hrH6lMWiSl1WbGaiP01eT6kzywdY1U+zLc0MP+nwEnUiS9UI8IaOTUhTeQJLlCEWIbXINBH4YJbBQ== + dependencies: + "@types/eslint" "^7.2.13" + ansi-escapes "^4.2.1" + chalk "^4.1.0" + eslint-rule-docs "^1.1.5" + log-symbols "^4.0.0" + plur "^4.0.0" + string-width "^4.2.0" + supports-hyperlinks "^2.0.0" + eslint-import-resolver-node@^0.3.7: version "0.3.7" resolved "https://registry.yarnpkg.com/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.7.tgz#83b375187d412324a1963d84fa664377a23eb4d7" @@ -7646,6 +7813,11 @@ eslint-rule-composer@^0.3.0: resolved "https://registry.yarnpkg.com/eslint-rule-composer/-/eslint-rule-composer-0.3.0.tgz#79320c927b0c5c0d3d3d2b76c8b4a488f25bbaf9" integrity sha512-bt+Sh8CtDmn2OajxvNO+BX7Wn4CIWMpTRm3MaiKPCQcnnlm0CS2mhui6QaoeQugs+3Kj2ESKEEGJUdVafwhiCg== +eslint-rule-docs@^1.1.5: + version "1.1.235" + resolved "https://registry.yarnpkg.com/eslint-rule-docs/-/eslint-rule-docs-1.1.235.tgz#be6ef1fc3525f17b3c859ae2997fedadc89bfb9b" + integrity sha512-+TQ+x4JdTnDoFEXXb3fDvfGOwnyNV7duH8fXWTPD1ieaBmB8omj7Gw/pMBBu4uI2uJCCU8APDaQJzWuXnTsH4A== + eslint-scope@5.1.1, eslint-scope@^5.1.1: version "5.1.1" resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-5.1.1.tgz#e786e59a66cb92b3f6c1fb0d508aab174848f48c" @@ -7797,7 +7969,12 @@ event-emitter@^0.3.5: d "1" es5-ext "~0.10.14" -events@^3.2.0: +event-target-shim@^5.0.0: + version "5.0.1" + resolved "https://registry.yarnpkg.com/event-target-shim/-/event-target-shim-5.0.1.tgz#5d4d3ebdf9583d63a5333ce2deb7480ab2b05789" + integrity sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ== + +events@^3.2.0, events@^3.3.0: version "3.3.0" resolved "https://registry.yarnpkg.com/events/-/events-3.3.0.tgz#31a95ad0a924e2d2c419a813aeb2c4e878ea7400" integrity sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q== @@ -8005,6 +8182,13 @@ filter-obj@^5.1.0: resolved "https://registry.yarnpkg.com/filter-obj/-/filter-obj-5.1.0.tgz#5bd89676000a713d7db2e197f660274428e524ed" integrity sha512-qWeTREPoT7I0bifpPUXtxkZJ1XJzxWtfoWWkdVGqa+eCr3SHW/Ocp89o8vLvbUuQnadybJpjOKu4V+RwO6sGng== +find-replace@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/find-replace/-/find-replace-3.0.0.tgz#3e7e23d3b05167a76f770c9fbd5258b0def68c38" + integrity sha512-6Tb2myMioCAgv5kfvP5/PkZZ/ntTpVK39fHY7WkWBgvbeE+VHd/tZuZ4mrC+bxh4cfOZeYKVPaJIZtZXV7GNCQ== + dependencies: + array-back "^3.0.1" + find-root@^1.1.0: version "1.1.0" resolved "https://registry.yarnpkg.com/find-root/-/find-root-1.1.0.tgz#abcfc8ba76f708c42a97b3d685b7e9450bfb9ce4" @@ -8349,7 +8533,7 @@ glob@7.1.6: once "^1.3.0" path-is-absolute "^1.0.0" -glob@^7.1.1, glob@^7.1.3, glob@^7.1.6: +glob@^7.1.1, glob@^7.1.3, glob@^7.1.6, glob@^7.1.7: version "7.2.3" resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b" integrity sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q== @@ -8380,7 +8564,7 @@ globalthis@^1.0.3: dependencies: define-properties "^1.1.3" -globby@^11.0.3, globby@^11.1.0: +globby@^11.0.1, globby@^11.0.3, globby@^11.1.0: version "11.1.0" resolved "https://registry.yarnpkg.com/globby/-/globby-11.1.0.tgz#bd4be98bb042f83d796f7e3811991fbe82a0d34b" integrity sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g== @@ -8585,6 +8769,13 @@ graphql@=16.6.0: resolved "https://registry.yarnpkg.com/graphql/-/graphql-16.6.0.tgz#c2dcffa4649db149f6282af726c8c83f1c7c5fdb" integrity sha512-KPIBPDlW7NxrbT/eh4qPXz5FiFdL5UbaA0XUNz2Rp3Z3hqBSkbj0GVjwFDztsWVauZUWsbKHgMg++sk8UX0bkw== +graphql@^0.11.7: + version "0.11.7" + resolved "https://registry.yarnpkg.com/graphql/-/graphql-0.11.7.tgz#e5abaa9cb7b7cccb84e9f0836bf4370d268750c6" + integrity sha512-x7uDjyz8Jx+QPbpCFCMQ8lltnQa4p4vSYHx6ADe8rVYRTdsyhCJbvSty5DAsLVmU6cGakl+r8HQYolKHxk/tiw== + dependencies: + iterall "1.1.3" + grid-index@^1.1.0: version "1.1.0" resolved "https://registry.yarnpkg.com/grid-index/-/grid-index-1.1.0.tgz#97f8221edec1026c8377b86446a7c71e79522ea7" @@ -8620,6 +8811,11 @@ har-validator@~4.2.1: ajv "^4.9.1" har-schema "^1.0.5" +hard-rejection@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/hard-rejection/-/hard-rejection-2.1.0.tgz#1c6eda5c1685c63942766d79bb40ae773cecd883" + integrity sha512-VIZB+ibDhx7ObhAe7OVtoEbuP4h/MuOTHJ+J8h/eBXotJYl0fBgR72xDFCKgIh22OJZIOVNxBMWuhAr10r8HdA== + harmony-reflect@^1.4.6: version "1.6.2" resolved "https://registry.yarnpkg.com/harmony-reflect/-/harmony-reflect-1.6.2.tgz#31ecbd32e648a34d030d86adb67d4d47547fe710" @@ -8702,6 +8898,13 @@ hasown@^2.0.0: dependencies: function-bind "^1.1.2" +hasown@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.2.tgz#003eaf91be7adc372e84ec59dc37252cedb80003" + integrity sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ== + dependencies: + function-bind "^1.1.2" + hast-util-from-parse5@^5.0.0: version "5.0.3" resolved "https://registry.yarnpkg.com/hast-util-from-parse5/-/hast-util-from-parse5-5.0.3.tgz#3089dc0ee2ccf6ec8bc416919b51a54a589e097c" @@ -8920,6 +9123,18 @@ hoist-non-react-statics@^3.0.0, hoist-non-react-statics@^3.1.0, hoist-non-react- dependencies: react-is "^16.7.0" +hosted-git-info@^2.1.4: + version "2.8.9" + resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.8.9.tgz#dffc0bf9a21c02209090f2aa69429e1414daf3f9" + integrity sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw== + +hosted-git-info@^4.0.1: + version "4.1.0" + resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz#827b82867e9ff1c8d0c4d9d53880397d2c86d224" + integrity sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA== + dependencies: + lru-cache "^6.0.0" + hsluv@^0.0.3: version "0.0.3" resolved "https://registry.yarnpkg.com/hsluv/-/hsluv-0.0.3.tgz#829107dafb4a9f8b52a1809ed02e091eade6754c" @@ -9026,7 +9241,7 @@ identity-obj-proxy@^3.0.0: dependencies: harmony-reflect "^1.4.6" -ieee754@^1.1.12, ieee754@^1.1.13: +ieee754@^1.1.12, ieee754@^1.1.13, ieee754@^1.2.1: version "1.2.1" resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352" integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA== @@ -9144,6 +9359,11 @@ invariant@^2.2.4: dependencies: loose-envify "^1.0.0" +irregular-plurals@^3.2.0: + version "3.5.0" + resolved "https://registry.yarnpkg.com/irregular-plurals/-/irregular-plurals-3.5.0.tgz#0835e6639aa8425bdc8b0d33d0dc4e89d9c01d2b" + integrity sha512-1ANGLZ+Nkv1ptFb2pa8oG8Lem4krflKuX/gINiHJHjJUKaJHk/SXk5x6K3J+39/p0h1RQ2saROclJJ+QLvETCQ== + is-absolute@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/is-absolute/-/is-absolute-1.0.0.tgz#395e1ae84b11f26ad1795e73c17378e48a301576" @@ -9246,6 +9466,13 @@ is-core-module@^2.11.0, is-core-module@^2.9.0: dependencies: has "^1.0.3" +is-core-module@^2.13.0, is-core-module@^2.5.0: + version "2.15.1" + resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.15.1.tgz#a7363a25bee942fefab0de13bf6aa372c82dcc37" + integrity sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ== + dependencies: + hasown "^2.0.2" + is-date-object@^1.0.1: version "1.0.5" resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.0.5.tgz#0841d5536e724c25597bf6ea62e1bd38298df31f" @@ -9552,6 +9779,21 @@ isstream@~0.1.2: resolved "https://registry.yarnpkg.com/isstream/-/isstream-0.1.2.tgz#47e63f7af55afa6f92e1500e690eb8b8529c099a" integrity sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g== +iterall@1.1.3: + version "1.1.3" + resolved "https://registry.yarnpkg.com/iterall/-/iterall-1.1.3.tgz#1cbbff96204056dde6656e2ed2e2226d0e6d72c9" + integrity sha512-Cu/kb+4HiNSejAPhSaN1VukdNTTi/r4/e+yykqjlG/IW+1gZH5b4+Bq3whDX4tvbYugta3r8KTMUiqT3fIGxuQ== + +jest-diff@^29.0.3: + version "29.7.0" + resolved "https://registry.yarnpkg.com/jest-diff/-/jest-diff-29.7.0.tgz#017934a66ebb7ecf6f205e84699be10afd70458a" + integrity sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw== + dependencies: + chalk "^4.0.0" + diff-sequences "^29.6.3" + jest-get-type "^29.6.3" + pretty-format "^29.7.0" + jest-diff@^29.6.2: version "29.6.2" resolved "https://registry.yarnpkg.com/jest-diff/-/jest-diff-29.6.2.tgz#c36001e5543e82a0805051d3ceac32e6825c1c46" @@ -9572,6 +9814,11 @@ jest-get-type@^29.4.3: resolved "https://registry.yarnpkg.com/jest-get-type/-/jest-get-type-29.4.3.tgz#1ab7a5207c995161100b5187159ca82dd48b3dd5" integrity sha512-J5Xez4nRRMjk8emnTpWrlkyb9pfRQQanDrvWHhsR1+VUfbwxi30eVcZFlcdGInRibU4G5LwHXpI7IRHU0CY+gg== +jest-get-type@^29.6.3: + version "29.6.3" + resolved "https://registry.yarnpkg.com/jest-get-type/-/jest-get-type-29.6.3.tgz#36f499fdcea197c1045a127319c0481723908fd1" + integrity sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw== + jest-matcher-utils@^29.6.2: version "29.6.2" resolved "https://registry.yarnpkg.com/jest-matcher-utils/-/jest-matcher-utils-29.6.2.tgz#39de0be2baca7a64eacb27291f0bd834fea3a535" @@ -9636,6 +9883,11 @@ jose@^4.11.4: resolved "https://registry.yarnpkg.com/jose/-/jose-4.14.6.tgz#94dca1d04a0ad8c6bff0998cdb51220d473cc3af" integrity sha512-EqJPEUlZD0/CSUMubKtMaYUOtWe91tZXTWMJZoKSbLk+KtdhNdcvppH8lA9XwVu2V4Ailvsj0GBZJ2ZwDjfesQ== +js-base64@^3.7.7: + version "3.7.7" + resolved "https://registry.yarnpkg.com/js-base64/-/js-base64-3.7.7.tgz#e51b84bf78fbf5702b9541e2cb7bfcb893b43e79" + integrity sha512-7rCnleh0z2CkXhH67J8K1Ytz0b2Y+yxTPL+/KOJoa20hfnVQ/3/T6W/KflYI4bRHRagNeXeU2bkNGI3v1oS/lw== + js-levenshtein@^1.1.6: version "1.1.6" resolved "https://registry.yarnpkg.com/js-levenshtein/-/js-levenshtein-1.1.6.tgz#c6cee58eb3550372df8deb85fad5ce66ce01d59d" @@ -9735,6 +9987,11 @@ json-schema-to-typescript@^11.0.2: mz "^2.7.0" prettier "^2.6.2" +json-schema-to-zod@^2.4.1: + version "2.4.1" + resolved "https://registry.yarnpkg.com/json-schema-to-zod/-/json-schema-to-zod-2.4.1.tgz#bd1e66b4ac4da4d3293e59d9a582b77d8ce9744b" + integrity sha512-aMoez9TxgnfLAIZaWTPaQ+j7rOt1K9Ew/TBI85XcnhcFlo/47b1MDgpi4r07XndLSZWOX/KsJiRJvhdzSvo2Dw== + json-schema-traverse@^0.4.1: version "0.4.1" resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" @@ -9936,6 +10193,11 @@ keyboard-key@^1.0.4: resolved "https://registry.yarnpkg.com/keyboard-key/-/keyboard-key-1.1.0.tgz#6f2e8e37fa11475bb1f1d65d5174f1b35653f5b7" integrity sha512-qkBzPTi3rlAKvX7k0/ub44sqOfXeLc/jcnGGmj5c7BJpU8eDrEVPyhCvNYAaoubbsLm9uGWwQJO1ytQK1a9/dQ== +kind-of@^6.0.3: + version "6.0.3" + resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-6.0.3.tgz#07c05034a6c349fa06e24fa35aa76db4580ce4dd" + integrity sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw== + kleur@^4.0.3: version "4.1.5" resolved "https://registry.yarnpkg.com/kleur/-/kleur-4.1.5.tgz#95106101795f7050c6c650f350c683febddb1780" @@ -10038,6 +10300,11 @@ locate-path@^6.0.0: dependencies: p-locate "^5.0.0" +lodash.camelcase@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz#b28aa6288a2b9fc651035c7711f65ab6190331a6" + integrity sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA== + lodash.debounce@^4.0.8: version "4.0.8" resolved "https://registry.yarnpkg.com/lodash.debounce/-/lodash.debounce-4.0.8.tgz#82d79bff30a67c4005ffd5e2515300ad9ca4d7af" @@ -10191,6 +10458,16 @@ map-limit@0.0.1: dependencies: once "~1.3.0" +map-obj@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/map-obj/-/map-obj-1.0.1.tgz#d933ceb9205d82bdcf4886f6742bdc2b4dea146d" + integrity sha512-7N/q3lyZ+LVCp7PzuxrJr4KMbBE2hW7BT7YNia330OFxIf4d3r5zVpicP2650l7CPN6RM9zOJRl3NGpqSiw3Eg== + +map-obj@^4.0.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/map-obj/-/map-obj-4.3.0.tgz#9304f906e93faae70880da102a9f1df0ea8bb05a" + integrity sha512-hdN1wVrZbb29eBGiGjJbeP8JbKjq1urkHJ/LIP/NY48MZ1QVXUsQBV1G1zvYFHn1XE06cwjBsOI2K3Ulnj1YXQ== + mapbox-gl@1.10.1: version "1.10.1" resolved "https://registry.yarnpkg.com/mapbox-gl/-/mapbox-gl-1.10.1.tgz#7dbd53bdf2f78e45e125c1115e94dea286ef663c" @@ -10402,6 +10679,24 @@ memoizee@^0.4.15: next-tick "^1.1.0" timers-ext "^0.1.7" +meow@^9.0.0: + version "9.0.0" + resolved "https://registry.yarnpkg.com/meow/-/meow-9.0.0.tgz#cd9510bc5cac9dee7d03c73ee1f9ad959f4ea364" + integrity sha512-+obSblOQmRhcyBt62furQqRAQpNyWXo8BuQ5bN7dG8wmwQ+vwHKp/rCFD4CrTP8CsDQD1sjoZ94K417XEUk8IQ== + dependencies: + "@types/minimist" "^1.2.0" + camelcase-keys "^6.2.2" + decamelize "^1.2.0" + decamelize-keys "^1.1.0" + hard-rejection "^2.1.0" + minimist-options "4.1.0" + normalize-package-data "^3.0.0" + read-pkg-up "^7.0.1" + redent "^3.0.0" + trim-newlines "^3.0.0" + type-fest "^0.18.0" + yargs-parser "^20.2.3" + merge-stream@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" @@ -10749,6 +11044,15 @@ minimatch@^4.2.3: dependencies: brace-expansion "^1.1.7" +minimist-options@4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/minimist-options/-/minimist-options-4.1.0.tgz#c0655713c53a8a2ebd77ffa247d342c40f010619" + integrity sha512-Q4r8ghd80yhO/0j1O3B2BjweX3fiHg9cdOwjJd2J76Q135c+NDxGCqdYKQ1SKBuFfgWbAUzBfvYjPUEeNgqN1A== + dependencies: + arrify "^1.0.1" + is-plain-obj "^1.1.0" + kind-of "^6.0.3" + minimist@^1.2.0, minimist@^1.2.5, minimist@^1.2.6: version "1.2.8" resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c" @@ -10850,6 +11154,11 @@ moment@^2.29.3: resolved "https://registry.yarnpkg.com/moment/-/moment-2.29.4.tgz#3dbe052889fe7c1b2ed966fcb3a77328964ef108" integrity sha512-5LC9SOxjSc2HF6vO2CyuTDNivEdoz2IvyJJGj6X8DJ0eFyfszE0QiEd+iXmBvUP3WHxSjFH/vIsA0EN00cgr8w== +moment@^2.30.1: + version "2.30.1" + resolved "https://registry.yarnpkg.com/moment/-/moment-2.30.1.tgz#f8c91c07b7a786e30c59926df530b4eac96974ae" + integrity sha512-uEmtNhbDOrWPFS+hdjFCBfy9f2YoyzRpwcl+DqpC6taX21FzsTLQVbMV/W7PzNSX6x/bhC1zA3c2UQ5NzH6how== + monaco-editor@^0.29.1: version "0.29.1" resolved "https://registry.yarnpkg.com/monaco-editor/-/monaco-editor-0.29.1.tgz#6ee93d8a5320704d48fd7058204deed72429c020" @@ -11074,6 +11383,26 @@ nopt@~1.0.10: dependencies: abbrev "1" +normalize-package-data@^2.5.0: + version "2.5.0" + resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-2.5.0.tgz#e66db1838b200c1dfc233225d12cb36520e234a8" + integrity sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA== + dependencies: + hosted-git-info "^2.1.4" + resolve "^1.10.0" + semver "2 || 3 || 4 || 5" + validate-npm-package-license "^3.0.1" + +normalize-package-data@^3.0.0: + version "3.0.3" + resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-3.0.3.tgz#dbcc3e2da59509a0983422884cd172eefdfa525e" + integrity sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA== + dependencies: + hosted-git-info "^4.0.1" + is-core-module "^2.5.0" + semver "^7.3.4" + validate-npm-package-license "^3.0.1" + normalize-path@^2.1.1: version "2.1.1" resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-2.1.1.tgz#1ab28b556e198363a8c1a6f7e6fa20137fe6aed9" @@ -11382,6 +11711,16 @@ package-json-versionify@^1.0.4: dependencies: browserify-package-json "^1.0.0" +pako@^0.2.5: + version "0.2.9" + resolved "https://registry.yarnpkg.com/pako/-/pako-0.2.9.tgz#f3f7522f4ef782348da8161bad9ecfd51bf83a75" + integrity sha512-NUcwaKxUxWrZLpDG+z/xZaCgQITkA/Dv4V/T6bw7VON6l1Xz/VnrBqrYjZQ12TamKHzITTfOEIYUj48y2KXImA== + +pako@^1.0.6: + version "1.0.11" + resolved "https://registry.yarnpkg.com/pako/-/pako-1.0.11.tgz#6c9599d340d54dfd3946380252a35705a6b992bf" + integrity sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw== + pako@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/pako/-/pako-2.1.0.tgz#266cc37f98c7d883545d11335c00fbd4062c9a86" @@ -11500,6 +11839,11 @@ path-case@^3.0.4: dot-case "^3.0.4" tslib "^2.0.3" +path-equal@^1.1.2: + version "1.2.5" + resolved "https://registry.yarnpkg.com/path-equal/-/path-equal-1.2.5.tgz#9fcbdd5e5daee448e96f43f3bac06c666b5e982a" + integrity sha512-i73IctDr3F2W+bsOWDyyVm/lqsXO47aY9nsFZUjTT/aljSbkxHxxCoyZ9UUrM8jK0JVod+An+rl48RCsvWM+9g== + path-exists@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz#513bdbe2d3b95d7762e8c1137efa195c6c61b5b3" @@ -11695,6 +12039,18 @@ plotly.js@^2.23.2: webgl-context "^2.2.0" world-calendars "^1.0.3" +plur@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/plur/-/plur-4.0.0.tgz#729aedb08f452645fe8c58ef115bf16b0a73ef84" + integrity sha512-4UGewrYgqDFw9vV6zNV+ADmPAUAfJPKtGvb/VdpQAx25X5f3xXdGdyOEVFwkl8Hl/tl7+xbeHqSEM+D5/TirUg== + dependencies: + irregular-plurals "^3.2.0" + +pluralize@^8.0.0: + version "8.0.0" + resolved "https://registry.yarnpkg.com/pluralize/-/pluralize-8.0.0.tgz#1a6fa16a38d12a1901e0320fa017051c539ce3b1" + integrity sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA== + point-in-polygon@^1.1.0: version "1.1.0" resolved "https://registry.yarnpkg.com/point-in-polygon/-/point-in-polygon-1.1.0.tgz#b0af2616c01bdee341cbf2894df643387ca03357" @@ -11859,6 +12215,11 @@ process-nextick-args@~2.0.0: resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2" integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag== +process@^0.11.10: + version "0.11.10" + resolved "https://registry.yarnpkg.com/process/-/process-0.11.10.tgz#7332300e840161bda3e69a1d1d91a7d4bc16f182" + integrity sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A== + promise@^7.1.1: version "7.3.1" resolved "https://registry.yarnpkg.com/promise/-/promise-7.3.1.tgz#064b72602b18f90f29192b8b1bc418ffd1ebd3bf" @@ -11960,11 +12321,76 @@ queue-microtask@^1.2.2: resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243" integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A== +quick-lru@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/quick-lru/-/quick-lru-4.0.1.tgz#5b8878f113a58217848c6482026c73e1ba57727f" + integrity sha512-ARhCpm70fzdcvNQfPoy49IaanKkTlRWF2JMzqhcJbhSFRZv7nPTvZJdcY7301IPmvW+/p0RgIWnQDLJxifsQ7g== + quickselect@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/quickselect/-/quickselect-2.0.0.tgz#f19680a486a5eefb581303e023e98faaf25dd018" integrity sha512-RKJ22hX8mHe3Y6wH/N3wCM6BWtjaxIyyUIkpHOvfFnxdI4yD4tBXEBKSbriGujF6jnSVkJrffuo6vxACiSSxIw== +quicktype-core@23.0.170: + version "23.0.170" + resolved "https://registry.yarnpkg.com/quicktype-core/-/quicktype-core-23.0.170.tgz#ecaab8091552980883dd587ebe7c91abed74866c" + integrity sha512-ZsjveG0yJUIijUx4yQshzyQ5EAXKbFSBTQJHnJ+KoSZVxcS+m3GcmDpzrdUIRYMhgLaF11ZGvLSYi5U0xcwemw== + dependencies: + "@glideapps/ts-necessities" "2.2.3" + browser-or-node "^3.0.0" + collection-utils "^1.0.1" + cross-fetch "^4.0.0" + is-url "^1.2.4" + js-base64 "^3.7.7" + lodash "^4.17.21" + pako "^1.0.6" + pluralize "^8.0.0" + readable-stream "4.5.2" + unicode-properties "^1.4.1" + urijs "^1.19.1" + wordwrap "^1.0.0" + yaml "^2.4.1" + +quicktype-graphql-input@23.0.170: + version "23.0.170" + resolved "https://registry.yarnpkg.com/quicktype-graphql-input/-/quicktype-graphql-input-23.0.170.tgz#f52bb9204a1b434b4e5f0a9003da227d50bcd9da" + integrity sha512-L0xPKdIFZFChwups9oqJuQw/vwEbRVKBvU9L5jAs0Z/aLyfdsuxDpKGMJXnNWa2yE7NhPX/UDX8ytxn8uc8hdQ== + dependencies: + collection-utils "^1.0.1" + graphql "^0.11.7" + quicktype-core "23.0.170" + +quicktype-typescript-input@23.0.170: + version "23.0.170" + resolved "https://registry.yarnpkg.com/quicktype-typescript-input/-/quicktype-typescript-input-23.0.170.tgz#13efb2f8a7846a0f685fab2852086995f8c712b2" + integrity sha512-lckhc//Mc95f/puRFKv4BFs7VpUUJXhw/psh+5ZAMiErxOWgoF87XthGusmaqoXNzjmEy1AVwGgMCG2pp/tJ/w== + dependencies: + "@mark.probst/typescript-json-schema" "0.55.0" + quicktype-core "23.0.170" + typescript "4.9.5" + +quicktype@^23.0.170: + version "23.0.170" + resolved "https://registry.yarnpkg.com/quicktype/-/quicktype-23.0.170.tgz#3a70a5d0870e327d0f7ee35e61d25b17744bb8bc" + integrity sha512-3gFyS7w36ktxrttEv1gMfuUlGairepnSpLN0cp7JVevkKX2N6Uk8AyMlDS2Puki09MY6PB6ch90plThvACtEHA== + dependencies: + "@glideapps/ts-necessities" "^2.2.3" + chalk "^4.1.2" + collection-utils "^1.0.1" + command-line-args "^5.2.1" + command-line-usage "^7.0.1" + cross-fetch "^4.0.0" + graphql "^0.11.7" + lodash "^4.17.21" + moment "^2.30.1" + quicktype-core "23.0.170" + quicktype-graphql-input "23.0.170" + quicktype-typescript-input "23.0.170" + readable-stream "^4.5.2" + stream-json "1.8.0" + string-to-stream "^3.0.1" + typescript "4.9.5" + raf@^3.4.1: version "3.4.1" resolved "https://registry.yarnpkg.com/raf/-/raf-3.4.1.tgz#0742e99a4a6552f445d73e3ee0328af0ff1ede39" @@ -12330,6 +12756,36 @@ read-cache@^1.0.0: dependencies: pify "^2.3.0" +read-pkg-up@^7.0.0, read-pkg-up@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/read-pkg-up/-/read-pkg-up-7.0.1.tgz#f3a6135758459733ae2b95638056e1854e7ef507" + integrity sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg== + dependencies: + find-up "^4.1.0" + read-pkg "^5.2.0" + type-fest "^0.8.1" + +read-pkg@^5.2.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/read-pkg/-/read-pkg-5.2.0.tgz#7bf295438ca5a33e56cd30e053b34ee7250c93cc" + integrity sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg== + dependencies: + "@types/normalize-package-data" "^2.4.0" + normalize-package-data "^2.5.0" + parse-json "^5.0.0" + type-fest "^0.6.0" + +readable-stream@4.5.2, readable-stream@^4.5.2: + version "4.5.2" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-4.5.2.tgz#9e7fc4c45099baeed934bff6eb97ba6cf2729e09" + integrity sha512-yjavECdqeZ3GLXNgRXgeQEdz9fvDDkNKyHnbHRFtOr7/LcfgBcmct7t/ET+HaCTqfh06OzoAxrkN/IfjJBVe+g== + dependencies: + abort-controller "^3.0.0" + buffer "^6.0.3" + events "^3.3.0" + process "^0.11.10" + string_decoder "^1.3.0" + "readable-stream@>=1.0.33-1 <1.1.0-0": version "1.0.34" resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-1.0.34.tgz#125820e34bc842d2f2aaafafe4c2916ee32c157c" @@ -12792,6 +13248,15 @@ resolve@^1.0.0, resolve@^1.1.10, resolve@^1.1.5, resolve@^1.1.7, resolve@^1.10.1 path-parse "^1.0.7" supports-preserve-symlinks-flag "^1.0.0" +resolve@^1.10.0: + version "1.22.8" + resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.8.tgz#b6c87a9f2aa06dfab52e3d70ac8cde321fa5a48d" + integrity sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw== + dependencies: + is-core-module "^2.13.0" + path-parse "^1.0.7" + supports-preserve-symlinks-flag "^1.0.0" + resolve@^2.0.0-next.4: version "2.0.0-next.4" resolved "https://registry.yarnpkg.com/resolve/-/resolve-2.0.0-next.4.tgz#3d37a113d6429f496ec4752d2a2e58efb1fd4660" @@ -12936,6 +13401,11 @@ safe-regex-test@^1.0.0: get-intrinsic "^1.1.3" is-regex "^1.1.4" +safe-stable-stringify@^2.2.0: + version "2.5.0" + resolved "https://registry.yarnpkg.com/safe-stable-stringify/-/safe-stable-stringify-2.5.0.tgz#4ca2f8e385f2831c432a719b108a3bf7af42a1dd" + integrity sha512-b3rppTKm9T+PsVCBEOUR46GWI7fdOs00VKZ1+9c1EWDaDMvjQc6tUwuFyIprgGgTcWoVHSKrU8H31ZHA2e0RHA== + "safer-buffer@>= 2.1.2 < 3", "safer-buffer@>= 2.1.2 < 3.0.0", safer-buffer@^2.0.2, safer-buffer@^2.1.0, safer-buffer@~2.1.0: version "2.1.2" resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" @@ -13009,6 +13479,11 @@ semantic-ui-react@^0.88.2: react-popper "^1.3.4" shallowequal "^1.1.0" +"semver@2 || 3 || 4 || 5", semver@^5.3.0, semver@^5.7.1: + version "5.7.2" + resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.2.tgz#48d55db737c3287cd4835e17fa13feace1c41ef8" + integrity sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g== + semver@7.x, semver@^7.3.5, semver@^7.3.7: version "7.5.4" resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.4.tgz#483986ec4ed38e1c6c48c34894a9182dbff68a6e" @@ -13016,16 +13491,16 @@ semver@7.x, semver@^7.3.5, semver@^7.3.7: dependencies: lru-cache "^6.0.0" -semver@^5.3.0, semver@^5.7.1: - version "5.7.2" - resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.2.tgz#48d55db737c3287cd4835e17fa13feace1c41ef8" - integrity sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g== - semver@^6.0.0, semver@^6.1.0, semver@^6.3.0, semver@^6.3.1: version "6.3.1" resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== +semver@^7.3.4: + version "7.6.3" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.3.tgz#980f7b5550bc175fb4dc09403085627f9eb33143" + integrity sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A== + semver@~7.0.0: version "7.0.0" resolved "https://registry.yarnpkg.com/semver/-/semver-7.0.0.tgz#5f3ca35761e47e05b206c6daff2cf814f0316b8e" @@ -13275,6 +13750,32 @@ spawn-command@^0.0.2-1: resolved "https://registry.yarnpkg.com/spawn-command/-/spawn-command-0.0.2-1.tgz#62f5e9466981c1b796dc5929937e11c9c6921bd0" integrity sha512-n98l9E2RMSJ9ON1AKisHzz7V42VDiBQGY6PB1BwRglz99wpVsSuGzQ+jOi6lFXBGVTCrRpltvjm+/XA+tpeJrg== +spdx-correct@^3.0.0: + version "3.2.0" + resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-3.2.0.tgz#4f5ab0668f0059e34f9c00dce331784a12de4e9c" + integrity sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA== + dependencies: + spdx-expression-parse "^3.0.0" + spdx-license-ids "^3.0.0" + +spdx-exceptions@^2.1.0: + version "2.5.0" + resolved "https://registry.yarnpkg.com/spdx-exceptions/-/spdx-exceptions-2.5.0.tgz#5d607d27fc806f66d7b64a766650fa890f04ed66" + integrity sha512-PiU42r+xO4UbUS1buo3LPJkjlO7430Xn5SVAhdpzzsPHsjbYVflnnFdATgabnLude+Cqu25p6N+g2lw/PFsa4w== + +spdx-expression-parse@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz#cf70f50482eefdc98e3ce0a6833e4a53ceeba679" + integrity sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q== + dependencies: + spdx-exceptions "^2.1.0" + spdx-license-ids "^3.0.0" + +spdx-license-ids@^3.0.0: + version "3.0.20" + resolved "https://registry.yarnpkg.com/spdx-license-ids/-/spdx-license-ids-3.0.20.tgz#e44ed19ed318dd1e5888f93325cee800f0f51b89" + integrity sha512-jg25NiDV/1fLtSgEgyvVyDunvaNHbuwF9lfNV17gSmPFAlYzdfNBlLtLzXTevwkPj7DhGbmN9VnmJIgLnhvaBw== + split-on-first@^3.0.0: version "3.0.0" resolved "https://registry.yarnpkg.com/split-on-first/-/split-on-first-3.0.0.tgz#f04959c9ea8101b9b0bbf35a61b9ebea784a23e7" @@ -13346,6 +13847,18 @@ std-env@^3.5.0: resolved "https://registry.yarnpkg.com/std-env/-/std-env-3.7.0.tgz#c9f7386ced6ecf13360b6c6c55b8aaa4ef7481d2" integrity sha512-JPbdCEQLj1w5GilpiHAx3qJvFndqybBysA3qUOnznweH4QbNYUsW/ea8QzSrnh0vNsezMMw5bcVool8lM0gwzg== +stream-chain@^2.2.5: + version "2.2.5" + resolved "https://registry.yarnpkg.com/stream-chain/-/stream-chain-2.2.5.tgz#b30967e8f14ee033c5b9a19bbe8a2cba90ba0d09" + integrity sha512-1TJmBx6aSWqZ4tx7aTpBDXK0/e2hhcNSTV8+CbFJtDjbb+I1mZ8lHit0Grw9GRT+6JbIrrDd8esncgBi8aBXGA== + +stream-json@1.8.0: + version "1.8.0" + resolved "https://registry.yarnpkg.com/stream-json/-/stream-json-1.8.0.tgz#53f486b2e3b4496c506131f8d7260ba42def151c" + integrity sha512-HZfXngYHUAr1exT4fxlbc1IOce1RYxp2ldeaf97LYCOPSoOqY/1Psp7iGvpb+6JIOgkra9zDYnPX01hGAHzEPw== + dependencies: + stream-chain "^2.2.5" + stream-parser@~0.3.1: version "0.3.1" resolved "https://registry.yarnpkg.com/stream-parser/-/stream-parser-0.3.1.tgz#1618548694420021a1182ff0af1911c129761773" @@ -13380,6 +13893,13 @@ string-split-by@^1.0.0: dependencies: parenthesis "^3.1.5" +string-to-stream@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/string-to-stream/-/string-to-stream-3.0.1.tgz#480e6fb4d5476d31cb2221f75307a5dcb6638a42" + integrity sha512-Hl092MV3USJuUCC6mfl9sPzGloA3K5VwdIeJjYIkXY/8K+mUvaeEabWJgArp+xXrsWxCajeT2pc4axbVhIZJyg== + dependencies: + readable-stream "^3.4.0" + "string-width@^1.0.2 || 2 || 3 || 4", string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3: version "4.2.3" resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" @@ -13430,7 +13950,7 @@ string.prototype.trimstart@^1.0.6: define-properties "^1.1.4" es-abstract "^1.20.4" -string_decoder@^1.1.1: +string_decoder@^1.1.1, string_decoder@^1.3.0: version "1.3.0" resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e" integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA== @@ -13563,7 +14083,7 @@ supports-color@^5.3.0, supports-color@^5.5.0: dependencies: has-flag "^3.0.0" -supports-color@^7.1.0: +supports-color@^7.0.0, supports-color@^7.1.0: version "7.2.0" resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da" integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw== @@ -13577,6 +14097,14 @@ supports-color@^8.1.0: dependencies: has-flag "^4.0.0" +supports-hyperlinks@^2.0.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/supports-hyperlinks/-/supports-hyperlinks-2.3.0.tgz#3943544347c1ff90b15effb03fc14ae45ec10624" + integrity sha512-RpsAZlpWcDwOPQA22aCH4J0t7L8JmAvsCxfOSEwm7cQs3LshN36QaTkwd70DnBOXDWGssw2eUoc8CaRWT0XunA== + dependencies: + has-flag "^4.0.0" + supports-color "^7.0.0" + supports-preserve-symlinks-flag@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz#6eda4bd344a3c94aea376d4cc31bc77311039e09" @@ -13635,6 +14163,14 @@ tabbable@^6.0.1: resolved "https://registry.yarnpkg.com/tabbable/-/tabbable-6.2.0.tgz#732fb62bc0175cfcec257330be187dcfba1f3b97" integrity sha512-Cat63mxsVJlzYvN51JmVXIgNoUokrIaT2zLclCXjRd8boZ0004U4KCs/sToJ75C6sdlByWxpYnb5Boif1VSFew== +table-layout@^4.1.0: + version "4.1.1" + resolved "https://registry.yarnpkg.com/table-layout/-/table-layout-4.1.1.tgz#0f72965de1a5c0c1419c9ba21cae4e73a2f73a42" + integrity sha512-iK5/YhZxq5GO5z8wb0bY1317uDF3Zjpha0QFFLA8/trAoiLbQD0HUbMesEaxyzUgDxi2QlcbM8IvqOlEjgoXBA== + dependencies: + array-back "^6.2.2" + wordwrapjs "^5.1.0" + tailwind-merge@^1.14.0: version "1.14.0" resolved "https://registry.yarnpkg.com/tailwind-merge/-/tailwind-merge-1.14.0.tgz#e677f55d864edc6794562c63f5001f45093cdb8b" @@ -13733,6 +14269,11 @@ timers-ext@^0.1.7: es5-ext "~0.10.46" next-tick "1" +tiny-inflate@^1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/tiny-inflate/-/tiny-inflate-1.0.3.tgz#122715494913a1805166aaf7c93467933eea26c4" + integrity sha512-pkY1fj1cKHb2seWDy0B16HeWyczlJA9/WW3u3c4z/NiWDsO3DOU5D7nhTLE9CF0yXv/QZFY7sEJmj24dK+Rrqw== + tiny-invariant@1.0.6: version "1.0.6" resolved "https://registry.yarnpkg.com/tiny-invariant/-/tiny-invariant-1.0.6.tgz#b3f9b38835e36a41c843a3b0907a5a7b3755de73" @@ -13888,6 +14429,11 @@ trim-lines@^3.0.0: resolved "https://registry.yarnpkg.com/trim-lines/-/trim-lines-3.0.1.tgz#d802e332a07df861c48802c04321017b1bd87338" integrity sha512-kRj8B+YHZCc9kQYdWfJB2/oUl9rA99qbowYYBtr4ui4mZyAQ2JpvVBd/6U2YloATfqBhBTSMhTpgBHtU0Mf3Rg== +trim-newlines@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/trim-newlines/-/trim-newlines-3.0.1.tgz#260a5d962d8b752425b32f3a7db0dcacd176c144" + integrity sha512-c1PTsA3tYrIsLGkJkzHF+w9F2EyxfXGo4UyJc4pFL++FMjnq0HJS69T3M7d//gKrFKwy429bouPescbjecU+Zw== + trim-trailing-lines@^1.0.0: version "1.1.4" resolved "https://registry.yarnpkg.com/trim-trailing-lines/-/trim-trailing-lines-1.1.4.tgz#bd4abbec7cc880462f10b2c8b5ce1d8d1ec7c2c0" @@ -13973,6 +14519,19 @@ tsconfig-paths@^3.14.1: minimist "^1.2.6" strip-bom "^3.0.0" +tsd@^0.30.4: + version "0.30.7" + resolved "https://registry.yarnpkg.com/tsd/-/tsd-0.30.7.tgz#319a0403073df6d3f572c4089378901662554ae5" + integrity sha512-oTiJ28D6B/KXoU3ww/Eji+xqHJojiuPVMwA12g4KYX1O72N93Nb6P3P3h2OAhhf92Xl8NIhb/xFmBZd5zw/xUw== + dependencies: + "@tsd/typescript" "~5.3.3" + eslint-formatter-pretty "^4.1.0" + globby "^11.0.1" + jest-diff "^29.0.3" + meow "^9.0.0" + path-exists "^4.0.0" + read-pkg-up "^7.0.0" + tslib@>=1.10.0, tslib@^2.0.0, tslib@^2.0.1, tslib@^2.1.0, tslib@^2.5.0: version "2.6.0" resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.0.tgz#b295854684dbda164e181d259a22cd779dcd7bc3" @@ -14081,6 +14640,11 @@ type-detect@^4.0.0, type-detect@^4.0.8: resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-4.0.8.tgz#7646fb5f18871cfbb7749e69bd39a6388eb7450c" integrity sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g== +type-fest@^0.18.0: + version "0.18.1" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.18.1.tgz#db4bc151a4a2cf4eebf9add5db75508db6cc841f" + integrity sha512-OIAYXk8+ISY+qTOwkHtKqzAuxchoMiD9Udx+FSGQDuiRR+PJKJHc2NJAXlbhkGwTt/4/nKZxELY1w3ReWOL8mw== + type-fest@^0.20.2: version "0.20.2" resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.20.2.tgz#1bf207f4b28f91583666cb5fbd327887301cd5f4" @@ -14091,6 +14655,16 @@ type-fest@^0.21.3: resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.21.3.tgz#d260a24b0198436e133fa26a524a6d65fa3b2e37" integrity sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w== +type-fest@^0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.6.0.tgz#8d2a2370d3df886eb5c90ada1c5bf6188acf838b" + integrity sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg== + +type-fest@^0.8.1: + version "0.8.1" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.8.1.tgz#09e249ebde851d3b1e48d27c105444667f17b83d" + integrity sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA== + type@^1.0.1: version "1.2.0" resolved "https://registry.yarnpkg.com/type/-/type-1.2.0.tgz#848dd7698dafa3e54a6c479e759c4bc3f18847a0" @@ -14144,6 +14718,26 @@ typescript@4.7.4: resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.7.4.tgz#1a88596d1cf47d59507a1bcdfb5b9dfe4d488235" integrity sha512-C0WQT0gezHuw6AdY1M2jxUO83Rjf0HP7Sk1DtXj6j1EwkQNZrHAg2XPWlq62oqEhYvONq5pkC2Y9oPljWToLmQ== +typescript@4.9.4: + version "4.9.4" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.9.4.tgz#a2a3d2756c079abda241d75f149df9d561091e78" + integrity sha512-Uz+dTXYzxXXbsFpM86Wh3dKCxrQqUcVMxwU54orwlJjOpO3ao8L7j5lH+dWfTwgCwIuM9GQ2kvVotzYJMXTBZg== + +typescript@4.9.5: + version "4.9.5" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.9.5.tgz#095979f9bcc0d09da324d58d03ce8f8374cbe65a" + integrity sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g== + +typical@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/typical/-/typical-4.0.0.tgz#cbeaff3b9d7ae1e2bbfaf5a4e6f11eccfde94fc4" + integrity sha512-VAH4IvQ7BDFYglMd7BPRDfLgxZZX4O4TFcRDA6EN5X7erNJJq+McIEp8np9aVtxrCJ6qx4GTYVfOWNjcqwZgRw== + +typical@^7.1.1: + version "7.2.0" + resolved "https://registry.yarnpkg.com/typical/-/typical-7.2.0.tgz#b3ec4b76530d144640df86c6b061dafd70e10c1e" + integrity sha512-W1+HdVRUl8fS3MZ9ogD51GOb46xMmhAZzR0WPw5jcgIZQJVvkddYzAl4YTU6g5w33Y1iRQLdIi2/1jhi2RNL0g== + ua-parser-js@^1.0.35: version "1.0.36" resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-1.0.36.tgz#a9ab6b9bd3a8efb90bb0816674b412717b7c428c" @@ -14222,11 +14816,27 @@ unicode-match-property-value-ecmascript@^2.1.0: resolved "https://registry.yarnpkg.com/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-2.1.0.tgz#cb5fffdcd16a05124f5a4b0bf7c3770208acbbe0" integrity sha512-qxkjQt6qjg/mYscYMC0XKRn3Rh0wFPlfxB0xkt9CfyTvpX1Ra0+rAmdX2QyAobptSEvuy4RtpPRui6XkV+8wjA== +unicode-properties@^1.4.1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/unicode-properties/-/unicode-properties-1.4.1.tgz#96a9cffb7e619a0dc7368c28da27e05fc8f9be5f" + integrity sha512-CLjCCLQ6UuMxWnbIylkisbRj31qxHPAurvena/0iwSVbQ2G1VY5/HjV0IRabOEbDHlzZlRdCrD4NhB0JtU40Pg== + dependencies: + base64-js "^1.3.0" + unicode-trie "^2.0.0" + unicode-property-aliases-ecmascript@^2.0.0: version "2.1.0" resolved "https://registry.yarnpkg.com/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-2.1.0.tgz#43d41e3be698bd493ef911077c9b131f827e8ccd" integrity sha512-6t3foTQI9qne+OZoVQB/8x8rk2k1eVy1gRXhV3oFQ5T6R1dqQ1xtin3XqSlx3+ATBkliTaR/hHyJBm+LVPNM8w== +unicode-trie@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/unicode-trie/-/unicode-trie-2.0.0.tgz#8fd8845696e2e14a8b67d78fa9e0dd2cad62fec8" + integrity sha512-x7bc76x0bm4prf1VLg79uhAzKw8DVboClSN5VxJuQ+LKDOVEW9CdH+VY7SP+vX7xCYQqzzgQpFqz15zeLvAtZQ== + dependencies: + pako "^0.2.5" + tiny-inflate "^1.0.0" + unified@^10.0.0, unified@^10.1.0: version "10.1.2" resolved "https://registry.yarnpkg.com/unified/-/unified-10.1.2.tgz#b1d64e55dafe1f0b98bb6c719881103ecf6c86df" @@ -14431,6 +15041,11 @@ uri-js@^4.2.2: dependencies: punycode "^2.1.0" +urijs@^1.19.1: + version "1.19.11" + resolved "https://registry.yarnpkg.com/urijs/-/urijs-1.19.11.tgz#204b0d6b605ae80bea54bea39280cdb7c9f923cc" + integrity sha512-HXgFDgDommxn5/bIv0cnQZsPhHDA90NPHD6+c/v21U5+Sx5hoP8+dP9IZXBU1gIfvdRfhG8cel9QNPeionfcCQ== + url-parse@^1.5.3: version "1.5.10" resolved "https://registry.yarnpkg.com/url-parse/-/url-parse-1.5.10.tgz#9d3c2f736c1d75dd3bd2be507dcc111f1e2ea9c1" @@ -14521,6 +15136,14 @@ v8-compile-cache-lib@^3.0.1: resolved "https://registry.yarnpkg.com/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz#6336e8d71965cb3d35a1bbb7868445a7c05264bf" integrity sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg== +validate-npm-package-license@^3.0.1: + version "3.0.4" + resolved "https://registry.yarnpkg.com/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz#fc91f6b9c7ba15c857f4cb2c5defeec39d4f410a" + integrity sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew== + dependencies: + spdx-correct "^3.0.0" + spdx-expression-parse "^3.0.0" + value-equal@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/value-equal/-/value-equal-1.0.1.tgz#1e0b794c734c5c0cade179c437d356d931a34d6c" @@ -15292,6 +15915,11 @@ wordwrap@^1.0.0: resolved "https://registry.yarnpkg.com/wordwrap/-/wordwrap-1.0.0.tgz#27584810891456a4171c8d0226441ade90cbcaeb" integrity sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q== +wordwrapjs@^5.1.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/wordwrapjs/-/wordwrapjs-5.1.0.tgz#4c4d20446dcc670b14fa115ef4f8fd9947af2b3a" + integrity sha512-JNjcULU2e4KJwUNv6CHgI46UvDGitb6dGryHajXTDiLgg1/RiGoPSDw4kZfYnwGtEXf2ZMeIewDQgFGzkCB2Sg== + world-calendars@^1.0.3: version "1.0.3" resolved "https://registry.yarnpkg.com/world-calendars/-/world-calendars-1.0.3.tgz#b25c5032ba24128ffc41d09faf4a5ec1b9c14335" @@ -15414,7 +16042,12 @@ yaml@^2.3.1: resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.3.2.tgz#f522db4313c671a0ca963a75670f1c12ea909144" integrity sha512-N/lyzTPaJasoDmfV7YTrYCI0G/3ivm/9wdG0aHuheKowWQwGTsK0Eoiw6utmzAnI6pkJa0DUVygvp3spqqEKXg== -yargs-parser@20.x: +yaml@^2.4.1: + version "2.6.0" + resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.6.0.tgz#14059ad9d0b1680d0f04d3a60fe00f3a857303c3" + integrity sha512-a6ae//JvKDEra2kdi1qzCyrJW/WZCgFi8ydDV+eXExl95t+5R+ijnqHJbz9tmMh8FUjx3iv2fCQ4dclAQlO2UQ== + +yargs-parser@20.x, yargs-parser@^20.2.3: version "20.2.9" resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.9.tgz#2eb7dc3b0289718fc295f362753845c41a0c94ee" integrity sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w== @@ -15449,7 +16082,7 @@ yargs@^15.3.1: y18n "^4.0.0" yargs-parser "^18.1.2" -yargs@^17.0.0, yargs@^17.3.1, yargs@^17.5.1: +yargs@^17.0.0, yargs@^17.1.1, yargs@^17.3.1, yargs@^17.5.1: version "17.7.2" resolved "https://registry.yarnpkg.com/yargs/-/yargs-17.7.2.tgz#991df39aca675a192b816e1e0363f9d75d2aa269" integrity sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w== @@ -15525,6 +16158,11 @@ zen-observable@^0.10.0: resolved "https://registry.yarnpkg.com/zen-observable/-/zen-observable-0.10.0.tgz#ee10eba75272897dbee5f152ab26bb5e0107f0c8" integrity sha512-iI3lT0iojZhKwT5DaFy2Ce42n3yFcLdFyOh01G7H0flMY60P8MJuVFEoJoNwXlmAyQ45GrjL6AcZmmlv8A5rbw== +zod@^3.23.8: + version "3.23.8" + resolved "https://registry.yarnpkg.com/zod/-/zod-3.23.8.tgz#e37b957b5d52079769fb8097099b592f0ef4067d" + integrity sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g== + zwitch@^2.0.0, zwitch@^2.0.4: version "2.0.4" resolved "https://registry.yarnpkg.com/zwitch/-/zwitch-2.0.4.tgz#c827d4b0acb76fc3e685a4c6ec2902d51070e9d7" diff --git a/weave/Makefile b/weave/Makefile index 2a9ae350dd6f..a49ad9bee72f 100644 --- a/weave/Makefile +++ b/weave/Makefile @@ -1,5 +1,8 @@ update_costs: python trace_server/costs/update_costs.py +generate_base_object_schemas: + python scripts/generate_base_object_schemas.py + update_model_providers: python trace_server/model_providers/model_providers.py diff --git a/weave/scripts/generate_base_object_schemas.py b/weave/scripts/generate_base_object_schemas.py new file mode 100644 index 000000000000..06b958a9c333 --- /dev/null +++ b/weave/scripts/generate_base_object_schemas.py @@ -0,0 +1,47 @@ +import json +from pathlib import Path + +from pydantic import create_model + +from weave.trace_server.interface.base_object_classes.base_object_registry import ( + BASE_OBJECT_REGISTRY, +) + +OUTPUT_DIR = ( + Path(__file__).parent.parent + / "trace_server" + / "interface" + / "base_object_classes" + / "generated" +) +OUTPUT_PATH = OUTPUT_DIR / "generated_base_object_class_schemas.json" + + +def generate_schemas() -> None: + """ + Generate JSON schemas for all registered base objects in BASE_OBJECT_REGISTRY. + Creates a top-level schema that includes all registered objects and writes it + to 'generated_base_object_class_schemas.json'. + """ + # Dynamically create a parent model with all registered objects as properties + CompositeModel = create_model( + "CompositeBaseObject", + **{name: (cls, ...) for name, cls in BASE_OBJECT_REGISTRY.items()}, + ) + + # Generate the schema using the composite model + top_level_schema = CompositeModel.model_json_schema(mode="validation") + + # make sure the output directory exists + OUTPUT_DIR.mkdir(parents=True, exist_ok=True) + + # Write schema to file + with OUTPUT_PATH.open("w") as f: + json.dump(top_level_schema, f, indent=2) + + print(f"Generated schema for {len(BASE_OBJECT_REGISTRY)} objects") + print(f"Wrote schema to {OUTPUT_PATH.absolute()}") + + +if __name__ == "__main__": + generate_schemas() diff --git a/weave/trace/base_objects.py b/weave/trace/base_objects.py new file mode 100644 index 000000000000..5a520e87ee97 --- /dev/null +++ b/weave/trace/base_objects.py @@ -0,0 +1 @@ +from weave.trace_server.interface.base_object_classes.base_object_registry import * diff --git a/weave/trace/serialize.py b/weave/trace/serialize.py index 2f605e24b771..09df639e91a6 100644 --- a/weave/trace/serialize.py +++ b/weave/trace/serialize.py @@ -5,6 +5,9 @@ from weave.trace import custom_objs from weave.trace.object_record import ObjectRecord from weave.trace.refs import ObjectRef, TableRef, parse_uri +from weave.trace_server.interface.base_object_classes.base_object_registry import ( + BASE_OBJECT_REGISTRY, +) from weave.trace_server.trace_server_interface import ( FileContentReadReq, FileCreateReq, @@ -225,6 +228,12 @@ def from_json(obj: Any, project_id: str, server: TraceServerInterface) -> Any: return custom_objs.decode_custom_obj( obj["weave_type"], files, obj.get("load_op") ) + elif ( + isinstance(val_type, str) + and obj.get("_class_name") == val_type + and (baseObject := BASE_OBJECT_REGISTRY.get(val_type)) + ): + return baseObject.model_validate(obj) else: return ObjectRecord( {k: from_json(v, project_id, server) for k, v in obj.items()} diff --git a/weave/trace_server/base_object_class_util.py b/weave/trace_server/base_object_class_util.py new file mode 100644 index 000000000000..ac5e572313d7 --- /dev/null +++ b/weave/trace_server/base_object_class_util.py @@ -0,0 +1,121 @@ +from typing import Any, Optional, Tuple + +from pydantic import BaseModel + +from weave.trace_server.interface.base_object_classes.base_object_registry import ( + BASE_OBJECT_REGISTRY, +) + +""" +There are two standard base object classes: BaseObject and Object + +`Object` is the base class for the more advanced object-oriented `weave.Object` use cases. +`BaseObject` is the more simple schema-based base object class. +""" +base_object_class_names = ["BaseObject", "Object"] + + +def get_base_object_class(val: Any) -> Optional[str]: + if isinstance(val, dict): + if "_bases" in val: + if isinstance(val["_bases"], list): + if len(val["_bases"]) >= 2: + if val["_bases"][-1] == "BaseModel": + if val["_bases"][-2] in base_object_class_names: + if len(val["_bases"]) > 2: + return val["_bases"][-3] + elif "_class_name" in val: + return val["_class_name"] + return None + + +def process_incoming_object( + val: Any, req_base_object_class: Optional[str] = None +) -> Tuple[dict, Optional[str]]: + """ + This method is responsible for accepting an incoming object from the user, validating it + against the base object class, and returning the object with the base object class + set. It does not mutate the original object, but returns a new object with values set if needed. + + Specifically,: + + 1. If the object is not a dict, it is returned as is, and the base object class is set to None. + 2. There are 2 ways to specify the base object class: + a. The `req_base_object_class` argument. + * used by non-pythonic writers of weave objects + b. The `_bases` & `_class_name` attributes of the object, which is a list of base class names. + * used by pythonic weave object writers (legacy) + 3. If the object has a base object class that does not match the requested base object class, + an error is thrown. + 4. if the object contains a base object class inside the payload, then we simply validate + the object against the base object class (if a match is found in BASE_OBJECT_REGISTRY) + 5. If the object does not have a base object class and a requested base object class is + provided, we require a match in BASE_OBJECT_REGISTRY and validate the object against + the requested base object class. Finally, we set the correct feilds. + """ + if not isinstance(val, dict): + if req_base_object_class is not None: + raise ValueError( + "set_base_object_class cannot be provided for non-dict objects" + ) + return val, None + + dict_val = val.copy() + val_base_object_class = get_base_object_class(dict_val) + + if ( + val_base_object_class != None + and req_base_object_class != None + and val_base_object_class != req_base_object_class + ): + raise ValueError( + f"set_base_object_class must match base_object_class: {val_base_object_class} != {req_base_object_class}" + ) + + if val_base_object_class is not None: + # In this case, we simply validate if the match is found + if base_object_class_type := BASE_OBJECT_REGISTRY.get(val_base_object_class): + base_object_class_type.model_validate(dict_val) + elif req_base_object_class is not None: + # In this case, we require that the base object class is registered + if base_object_class_type := BASE_OBJECT_REGISTRY.get(req_base_object_class): + dict_val = dump_base_object(base_object_class_type.model_validate(dict_val)) + else: + raise ValueError(f"Unknown base object class: {req_base_object_class}") + + base_object_class = val_base_object_class or req_base_object_class + + return dict_val, base_object_class + + +# Server-side version of `pydantic_object_record` +def dump_base_object(val: BaseModel) -> dict: + cls = val.__class__ + cls_name = val.__class__.__name__ + bases = [c.__name__ for c in cls.mro()[1:-1]] + + dump = {} + # Order matters here due to the way we calculate the digest! + # This matches the client + dump["_type"] = cls_name + for k in val.model_fields: + dump[k] = _general_dump(getattr(val, k)) + # yes, this is done twice, to match the client + dump["_class_name"] = cls_name + dump["_bases"] = bases + return dump + + +def _general_dump(val: Any) -> Any: + if isinstance(val, BaseModel): + return dump_base_object(val) + elif isinstance(val, dict): + return {k: _general_dump(v) for k, v in val.items()} + elif isinstance(val, list): + return [_general_dump(v) for v in val] + elif isinstance(val, tuple): + return tuple(_general_dump(v) for v in val) + elif isinstance(val, set): + return {_general_dump(v) for v in val} + else: + return val diff --git a/weave/trace_server/clickhouse_trace_server_batched.py b/weave/trace_server/clickhouse_trace_server_batched.py index 0bb383c5bc75..cb032104f6aa 100644 --- a/weave/trace_server/clickhouse_trace_server_batched.py +++ b/weave/trace_server/clickhouse_trace_server_batched.py @@ -53,6 +53,9 @@ from weave.trace_server import environment as wf_env from weave.trace_server import refs_internal as ri from weave.trace_server import trace_server_interface as tsi +from weave.trace_server.base_object_class_util import ( + process_incoming_object, +) from weave.trace_server.calls_query_builder import ( CallsQuery, HardCodedFilter, @@ -596,16 +599,19 @@ def ops_query(self, req: tsi.OpQueryReq) -> tsi.OpQueryRes: return tsi.OpQueryRes(op_objs=objs) def obj_create(self, req: tsi.ObjCreateReq) -> tsi.ObjCreateRes: - json_val = json.dumps(req.obj.val) + val, base_object_class = process_incoming_object( + req.obj.val, req.obj.set_base_object_class + ) + + json_val = json.dumps(val) digest = str_digest(json_val) - req_obj = req.obj ch_obj = ObjCHInsertable( - project_id=req_obj.project_id, - object_id=req_obj.object_id, - kind=get_kind(req.obj.val), - base_object_class=get_base_object_class(req.obj.val), - refs=extract_refs_from_values(req.obj.val), + project_id=req.obj.project_id, + object_id=req.obj.object_id, + kind=get_kind(val), + base_object_class=base_object_class, + refs=extract_refs_from_values(val), val_dump=json_val, digest=digest, ) @@ -2056,20 +2062,6 @@ def get_kind(val: Any) -> str: return "object" -def get_base_object_class(val: Any) -> Optional[str]: - if isinstance(val, dict): - if "_bases" in val: - if isinstance(val["_bases"], list): - if len(val["_bases"]) >= 2: - if val["_bases"][-1] == "BaseModel": - if val["_bases"][-2] == "Object": - if len(val["_bases"]) > 2: - return val["_bases"][-3] - elif "_class_name" in val: - return val["_class_name"] - return None - - def find_call_descendants( root_ids: list[str], all_calls: list[tsi.CallSchema], diff --git a/weave/trace_server/interface/base_object_classes/base_object_def.py b/weave/trace_server/interface/base_object_classes/base_object_def.py new file mode 100644 index 000000000000..746aae1df5ff --- /dev/null +++ b/weave/trace_server/interface/base_object_classes/base_object_def.py @@ -0,0 +1,10 @@ +from typing import Optional + +import pydantic + +RefStr = str + + +class BaseObject(pydantic.BaseModel): + name: Optional[str] = None + description: Optional[str] = None diff --git a/weave/trace_server/interface/base_object_classes/base_object_registry.py b/weave/trace_server/interface/base_object_classes/base_object_registry.py new file mode 100644 index 000000000000..8941dfa75ad3 --- /dev/null +++ b/weave/trace_server/interface/base_object_classes/base_object_registry.py @@ -0,0 +1,20 @@ +from typing import Dict, Type + +from weave.trace_server.interface.base_object_classes.base_object_def import BaseObject +from weave.trace_server.interface.base_object_classes.test_only_example import * + +BASE_OBJECT_REGISTRY: Dict[str, Type[BaseObject]] = {} + + +def register_base_object(cls: Type[BaseObject]) -> None: + """ + Register a BaseObject class in the global registry. + + Args: + cls: The BaseObject class to register + """ + BASE_OBJECT_REGISTRY[cls.__name__] = cls + + +register_base_object(TestOnlyExample) +register_base_object(TestOnlyNestedBaseObject) diff --git a/weave/trace_server/interface/base_object_classes/generated/generated_base_object_class_schemas.json b/weave/trace_server/interface/base_object_classes/generated/generated_base_object_class_schemas.json new file mode 100644 index 000000000000..4f7aee9dd543 --- /dev/null +++ b/weave/trace_server/interface/base_object_classes/generated/generated_base_object_class_schemas.json @@ -0,0 +1,114 @@ +{ + "$defs": { + "TestOnlyExample": { + "properties": { + "name": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null, + "title": "Name" + }, + "description": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null, + "title": "Description" + }, + "primitive": { + "title": "Primitive", + "type": "integer" + }, + "nested_base_model": { + "$ref": "#/$defs/TestOnlyNestedBaseModel" + }, + "nested_base_object": { + "title": "Nested Base Object", + "type": "string" + } + }, + "required": [ + "primitive", + "nested_base_model", + "nested_base_object" + ], + "title": "TestOnlyExample", + "type": "object" + }, + "TestOnlyNestedBaseModel": { + "properties": { + "a": { + "title": "A", + "type": "integer" + } + }, + "required": [ + "a" + ], + "title": "TestOnlyNestedBaseModel", + "type": "object" + }, + "TestOnlyNestedBaseObject": { + "properties": { + "name": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null, + "title": "Name" + }, + "description": { + "anyOf": [ + { + "type": "string" + }, + { + "type": "null" + } + ], + "default": null, + "title": "Description" + }, + "b": { + "title": "B", + "type": "integer" + } + }, + "required": [ + "b" + ], + "title": "TestOnlyNestedBaseObject", + "type": "object" + } + }, + "properties": { + "TestOnlyExample": { + "$ref": "#/$defs/TestOnlyExample" + }, + "TestOnlyNestedBaseObject": { + "$ref": "#/$defs/TestOnlyNestedBaseObject" + } + }, + "required": [ + "TestOnlyExample", + "TestOnlyNestedBaseObject" + ], + "title": "CompositeBaseObject", + "type": "object" +} \ No newline at end of file diff --git a/weave/trace_server/interface/base_object_classes/test_only_example.py b/weave/trace_server/interface/base_object_classes/test_only_example.py new file mode 100644 index 000000000000..49afb93bf891 --- /dev/null +++ b/weave/trace_server/interface/base_object_classes/test_only_example.py @@ -0,0 +1,26 @@ +from pydantic import BaseModel + +from weave.trace_server.interface.base_object_classes import base_object_def + + +class TestOnlyNestedBaseModel(BaseModel): + a: int + + +class TestOnlyNestedBaseObject(base_object_def.BaseObject): + b: int + + +class TestOnlyExample(base_object_def.BaseObject): + primitive: int + nested_base_model: TestOnlyNestedBaseModel + # Important: `RefStr` is just an alias for `str`. When defining `BaseObject`s, we + # should never have a property point to another `BaseObject`. This is because each + # base object is stored in the database and should be treated like a foreign key. + # + # It would be nice to have a way to ensure that no `BaseObject` has any `BaseObject` + # properties. + nested_base_object: base_object_def.RefStr + + +__all__ = ["TestOnlyExample", "TestOnlyNestedBaseObject"] diff --git a/weave/trace_server/sqlite_trace_server.py b/weave/trace_server/sqlite_trace_server.py index 93a4f510090c..1fd50cf1cb26 100644 --- a/weave/trace_server/sqlite_trace_server.py +++ b/weave/trace_server/sqlite_trace_server.py @@ -13,6 +13,9 @@ from weave.trace_server import refs_internal as ri from weave.trace_server import trace_server_interface as tsi +from weave.trace_server.base_object_class_util import ( + process_incoming_object, +) from weave.trace_server.emoji_util import detone_emojis from weave.trace_server.errors import InvalidRequest from weave.trace_server.feedback import ( @@ -608,25 +611,28 @@ def ops_query(self, req: tsi.OpQueryReq) -> tsi.OpQueryRes: def obj_create(self, req: tsi.ObjCreateReq) -> tsi.ObjCreateRes: conn, cursor = get_conn_cursor(self.db_path) - json_val = json.dumps(req.obj.val) + + val, base_object_class = process_incoming_object( + req.obj.val, req.obj.set_base_object_class + ) + json_val = json.dumps(val) digest = str_digest(json_val) # Validate object_id_validator(req.obj.object_id) - req_obj = req.obj # TODO: version index isn't right here, what if we delete stuff? with self.lock: cursor.execute("BEGIN TRANSACTION") # Mark all existing objects with such id as not latest cursor.execute( """UPDATE objects SET is_latest = 0 WHERE project_id = ? AND object_id = ?""", - (req_obj.project_id, req_obj.object_id), + (req.obj.project_id, req.obj.object_id), ) # first get version count cursor.execute( """SELECT COUNT(*) FROM objects WHERE project_id = ? AND object_id = ?""", - (req_obj.project_id, req_obj.object_id), + (req.obj.project_id, req.obj.object_id), ) version_index = cursor.fetchone()[0] @@ -644,11 +650,11 @@ def obj_create(self, req: tsi.ObjCreateReq) -> tsi.ObjCreateRes: is_latest ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)""", ( - req_obj.project_id, - req_obj.object_id, + req.obj.project_id, + req.obj.object_id, datetime.datetime.now().isoformat(), - get_kind(req_obj.val), - get_base_object_class(req_obj.val), + get_kind(val), + base_object_class, json.dumps([]), json_val, digest, @@ -1205,20 +1211,6 @@ def get_kind(val: Any) -> str: return "object" -def get_base_object_class(val: Any) -> Optional[str]: - if isinstance(val, dict): - if "_bases" in val: - if isinstance(val["_bases"], list): - if len(val["_bases"]) >= 2: - if val["_bases"][-1] == "BaseModel": - if val["_bases"][-2] == "Object": - if len(val["_bases"]) > 2: - return val["_bases"][-3] - elif "_class_name" in val: - return val["_class_name"] - return None - - def _transform_external_calls_field_to_internal_calls_field( field: str, cast: Optional[str] = None, diff --git a/weave/trace_server/trace_server_interface.py b/weave/trace_server/trace_server_interface.py index abdfeae38aca..af8778dc1d30 100644 --- a/weave/trace_server/trace_server_interface.py +++ b/weave/trace_server/trace_server_interface.py @@ -190,6 +190,7 @@ class ObjSchemaForInsert(BaseModel): project_id: str object_id: str val: Any + set_base_object_class: Optional[str] = None class TableSchemaForInsert(BaseModel):