diff --git a/docs/kcl/offsetPlane.md b/docs/kcl/offsetPlane.md index 4e45792b71..c5e87f6559 100644 --- a/docs/kcl/offsetPlane.md +++ b/docs/kcl/offsetPlane.md @@ -99,4 +99,20 @@ loft([squareSketch, circleSketch]) ![Rendered example of offsetPlane 3]() +```js +// A circle on the XY plane +startSketchOn("XY") + |> startProfileAt([0, 0], %) + |> circle({ radius: 10, center: [0, 0] }, %) + +// Triangle on the plane 4 units above +startSketchOn(offsetPlane("XY", 4)) + |> startProfileAt([0, 0], %) + |> line([10, 0], %) + |> line([0, 10], %) + |> close(%) +``` + +![Rendered example of offsetPlane 4]() + diff --git a/docs/kcl/polygon.md b/docs/kcl/polygon.md index de75c3538a..9e84c5e8a3 100644 --- a/docs/kcl/polygon.md +++ b/docs/kcl/polygon.md @@ -55,6 +55,6 @@ square = startSketchOn('XY') example = extrude(5, square) ``` -![Rendered example of polygon 1]() +![Rendered example of polygon 1]() diff --git a/docs/kcl/std.json b/docs/kcl/std.json index 29d2118ce2..bf8310168f 100644 --- a/docs/kcl/std.json +++ b/docs/kcl/std.json @@ -89825,10 +89825,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" ] }, - "value": {}, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" + ] + }, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" + ] + }, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -90046,6 +90207,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -90202,6 +90402,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ] }, @@ -93540,6 +93765,10 @@ "nullable": true } } + }, + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" } } }, @@ -93567,10 +93796,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" + ] + }, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" ] }, - "value": {}, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" + ] + }, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -93788,6 +94178,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -93944,6 +94373,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ] }, @@ -97282,6 +97736,10 @@ "nullable": true } } + }, + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" } } }, @@ -97313,10 +97771,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" + ] + }, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" ] }, - "value": {}, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" + ] + }, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -97534,6 +98153,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -97690,6 +98348,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ] }, @@ -101028,6 +101711,10 @@ "nullable": true } } + }, + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" } } }, @@ -105194,7 +105881,8 @@ "// Loft a square and a circle on the `XY` plane using offset.\nsquareSketch = startSketchOn('XY')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\ncircleSketch = startSketchOn(offsetPlane('XY', 150))\n |> circle({ center: [0, 100], radius: 50 }, %)\n\nloft([squareSketch, circleSketch])", "// Loft a square and a circle on the `XZ` plane using offset.\nsquareSketch = startSketchOn('XZ')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\ncircleSketch = startSketchOn(offsetPlane('XZ', 150))\n |> circle({ center: [0, 100], radius: 50 }, %)\n\nloft([squareSketch, circleSketch])", "// Loft a square and a circle on the `YZ` plane using offset.\nsquareSketch = startSketchOn('YZ')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\ncircleSketch = startSketchOn(offsetPlane('YZ', 150))\n |> circle({ center: [0, 100], radius: 50 }, %)\n\nloft([squareSketch, circleSketch])", - "// Loft a square and a circle on the `-XZ` plane using offset.\nsquareSketch = startSketchOn('-XZ')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\ncircleSketch = startSketchOn(offsetPlane('-XZ', -150))\n |> circle({ center: [0, 100], radius: 50 }, %)\n\nloft([squareSketch, circleSketch])" + "// Loft a square and a circle on the `-XZ` plane using offset.\nsquareSketch = startSketchOn('-XZ')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\ncircleSketch = startSketchOn(offsetPlane('-XZ', -150))\n |> circle({ center: [0, 100], radius: 50 }, %)\n\nloft([squareSketch, circleSketch])", + "// A circle on the XY plane\nstartSketchOn(\"XY\")\n |> startProfileAt([0, 0], %)\n |> circle({ radius: 10, center: [0, 0] }, %)\n\n// Triangle on the plane 4 units above\nstartSketchOn(offsetPlane(\"XY\", 4))\n |> startProfileAt([0, 0], %)\n |> line([10, 0], %)\n |> line([0, 10], %)\n |> close(%)" ] }, { @@ -127507,10 +128195,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" ] }, - "value": {}, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" + ] + }, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" + ] + }, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -127728,6 +128577,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -127884,6 +128772,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ] }, @@ -131222,6 +132135,10 @@ "nullable": true } } + }, + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" } } }, @@ -131246,10 +132163,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" ] }, - "value": {}, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" + ] + }, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" + ] + }, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -131467,6 +132545,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -131623,6 +132740,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ], "definitions": { @@ -131640,10 +132782,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" + ] + }, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" + ] + }, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" ] }, - "value": {}, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -131861,6 +133164,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -132017,6 +133359,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ] }, @@ -135355,6 +136722,10 @@ "nullable": true } } + }, + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" } } }, @@ -135380,10 +136751,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" + ] + }, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" + ] + }, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" ] }, - "value": {}, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -135601,6 +137133,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -135757,6 +137328,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ], "definitions": { @@ -135787,1041 +137383,589 @@ "maxItems": 3, "minItems": 3 }, - "TagEngineInfo": { - "description": "Engine information for a tag.", - "type": "object", - "required": [ - "id", - "sketch" - ], - "properties": { - "id": { - "description": "The id of the tagged object.", - "type": "string", - "format": "uuid" - }, - "sketch": { - "description": "The sketch the tag is on.", - "type": "string", - "format": "uuid" - }, - "path": { - "description": "The path the tag is on.", - "allOf": [ - { - "$ref": "#/components/schemas/Path" - } - ], - "nullable": true - }, - "surface": { - "description": "The surface information for the tag.", - "allOf": [ - { - "$ref": "#/components/schemas/ExtrudeSurface" - } - ], - "nullable": true - } - } - }, - "Path": { - "description": "A path.", + "KclValue": { + "description": "Any KCL value.", "oneOf": [ { - "description": "A path that goes to a point.", "type": "object", "required": [ - "__geoMeta", - "from", - "to", - "type" + "__meta", + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "ToPoint" + "Uuid" ] }, - "from": { - "description": "The from point.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "value": { + "type": "string", + "format": "uuid" }, - "to": { - "description": "The to point.", + "__meta": { "type": "array", "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "tag": { - "description": "The tag of the path.", - "allOf": [ - { - "$ref": "#/components/schemas/TagDeclarator" - } - ], - "nullable": true - }, - "__geoMeta": { - "description": "Metadata.", - "allOf": [ - { - "$ref": "#/components/schemas/GeoMeta" - } - ] + "$ref": "#/components/schemas/Metadata" + } } } }, { - "description": "A arc that is tangential to the last path segment that goes to a point", "type": "object", "required": [ - "__geoMeta", - "ccw", - "center", - "from", - "to", - "type" + "__meta", + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "TangentialArcTo" + "Bool" ] }, - "center": { - "description": "the arc's center", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "ccw": { - "description": "arc's direction", + "value": { "type": "boolean" }, - "from": { - "description": "The from point.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "to": { - "description": "The to point.", + "__meta": { "type": "array", "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "tag": { - "description": "The tag of the path.", - "allOf": [ - { - "$ref": "#/components/schemas/TagDeclarator" - } - ], - "nullable": true - }, - "__geoMeta": { - "description": "Metadata.", - "allOf": [ - { - "$ref": "#/components/schemas/GeoMeta" - } - ] + "$ref": "#/components/schemas/Metadata" + } } } }, { - "description": "A arc that is tangential to the last path segment", "type": "object", "required": [ - "__geoMeta", - "ccw", - "center", - "from", - "to", - "type" + "__meta", + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "TangentialArc" + "Number" ] }, - "center": { - "description": "the arc's center", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "ccw": { - "description": "arc's direction", - "type": "boolean" - }, - "from": { - "description": "The from point.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "value": { + "type": "number", + "format": "double" }, - "to": { - "description": "The to point.", + "__meta": { "type": "array", "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "tag": { - "description": "The tag of the path.", - "allOf": [ - { - "$ref": "#/components/schemas/TagDeclarator" - } - ], - "nullable": true - }, - "__geoMeta": { - "description": "Metadata.", - "allOf": [ - { - "$ref": "#/components/schemas/GeoMeta" - } - ] + "$ref": "#/components/schemas/Metadata" + } } } }, { - "description": "a complete arc", "type": "object", "required": [ - "__geoMeta", - "ccw", - "center", - "from", - "radius", - "to", - "type" + "__meta", + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "Circle" + "Int" ] }, - "center": { - "description": "the arc's center", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "radius": { - "description": "the arc's radius", - "type": "number", - "format": "double" - }, - "ccw": { - "description": "arc's direction", - "type": "boolean" - }, - "from": { - "description": "The from point.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "value": { + "type": "integer", + "format": "int64" }, - "to": { - "description": "The to point.", + "__meta": { "type": "array", "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "tag": { - "description": "The tag of the path.", - "allOf": [ - { - "$ref": "#/components/schemas/TagDeclarator" - } - ], - "nullable": true - }, - "__geoMeta": { - "description": "Metadata.", - "allOf": [ - { - "$ref": "#/components/schemas/GeoMeta" - } - ] + "$ref": "#/components/schemas/Metadata" + } } } }, { - "description": "A path that is horizontal.", "type": "object", "required": [ - "__geoMeta", - "from", - "to", + "__meta", "type", - "x" + "value" ], "properties": { "type": { "type": "string", "enum": [ - "Horizontal" + "String" ] }, - "x": { - "description": "The x coordinate.", - "type": "number", - "format": "double" - }, - "from": { - "description": "The from point.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "value": { + "type": "string" }, - "to": { - "description": "The to point.", + "__meta": { "type": "array", "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "tag": { - "description": "The tag of the path.", - "allOf": [ - { - "$ref": "#/components/schemas/TagDeclarator" - } - ], - "nullable": true - }, - "__geoMeta": { - "description": "Metadata.", - "allOf": [ - { - "$ref": "#/components/schemas/GeoMeta" - } - ] + "$ref": "#/components/schemas/Metadata" + } } } }, { - "description": "An angled line to.", "type": "object", "required": [ - "__geoMeta", - "from", - "to", - "type" + "__meta", + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "AngledLineTo" + "Array" ] }, - "x": { - "description": "The x coordinate.", - "type": "number", - "format": "double", - "nullable": true - }, - "y": { - "description": "The y coordinate.", - "type": "number", - "format": "double", - "nullable": true - }, - "from": { - "description": "The from point.", + "value": { "type": "array", "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "$ref": "#/components/schemas/KclValue" + } }, - "to": { - "description": "The to point.", + "__meta": { "type": "array", "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "tag": { - "description": "The tag of the path.", - "allOf": [ - { - "$ref": "#/components/schemas/TagDeclarator" - } - ], - "nullable": true - }, - "__geoMeta": { - "description": "Metadata.", - "allOf": [ - { - "$ref": "#/components/schemas/GeoMeta" - } - ] + "$ref": "#/components/schemas/Metadata" + } } } }, { - "description": "A base path.", "type": "object", "required": [ - "__geoMeta", - "from", - "to", - "type" + "__meta", + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "Base" + "Object" ] }, - "from": { - "description": "The from point.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } }, - "to": { - "description": "The to point.", + "__meta": { "type": "array", "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TagIdentifier" + ] }, - "tag": { - "description": "The tag of the path.", + "value": { + "type": "string" + }, + "info": { "allOf": [ { - "$ref": "#/components/schemas/TagDeclarator" + "$ref": "#/components/schemas/TagEngineInfo" } ], "nullable": true }, - "__geoMeta": { - "description": "Metadata.", - "allOf": [ - { - "$ref": "#/components/schemas/GeoMeta" - } - ] + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } } } }, { - "description": "A circular arc, not necessarily tangential to the current point.", "type": "object", "required": [ - "__geoMeta", - "center", - "from", - "radius", - "to", - "type" + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "Arc" + "TagDeclarator" ] }, - "center": { - "description": "Center of the circle that this arc is drawn on.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 - }, - "radius": { - "description": "Radius of the circle that this arc is drawn on.", - "type": "number", - "format": "double" - }, - "from": { - "description": "The from point.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "value": { + "type": "string" }, - "to": { - "description": "The to point.", + "digest": { "type": "array", "items": { - "type": "number", - "format": "double" + "type": "integer", + "format": "uint8", + "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 - }, - "tag": { - "description": "The tag of the path.", - "allOf": [ - { - "$ref": "#/components/schemas/TagDeclarator" - } - ], + "maxItems": 32, + "minItems": 32, "nullable": true }, - "__geoMeta": { - "description": "Metadata.", - "allOf": [ - { - "$ref": "#/components/schemas/GeoMeta" - } - ] + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } - } - ] - }, - "TagDeclarator": { - "type": "object", - "required": [ - "value" - ], - "properties": { - "value": { - "type": "string" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - "GeoMeta": { - "description": "Geometry metadata.", - "type": "object", - "required": [ - "id", - "sourceRange" - ], - "properties": { - "id": { - "description": "The id of the geometry.", - "type": "string", - "format": "uuid" }, - "sourceRange": { - "description": "The source range.", - "allOf": [ - { - "$ref": "#/components/schemas/SourceRange" - } - ] - } - } - }, - "ExtrudeSurface": { - "description": "An extrude surface.", - "oneOf": [ { - "description": "An extrude plane.", + "description": "A plane.", "type": "object", "required": [ - "faceId", + "__meta", "id", - "sourceRange", - "type" + "origin", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" ], "properties": { "type": { "type": "string", "enum": [ - "extrudePlane" + "Plane" ] }, - "faceId": { - "description": "The face id for the extrude plane.", + "id": { + "description": "The id of the plane.", "type": "string", "format": "uuid" }, - "tag": { - "description": "The tag.", + "value": { + "$ref": "#/components/schemas/PlaneType" + }, + "origin": { + "description": "Origin of the plane.", "allOf": [ { - "$ref": "#/components/schemas/TagDeclarator" + "$ref": "#/components/schemas/Point3d" } - ], - "nullable": true + ] }, - "id": { - "description": "The id of the geometry.", - "type": "string", - "format": "uuid" + "xAxis": { + "description": "What should the plane’s X axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] }, - "sourceRange": { - "description": "The source range.", + "yAxis": { + "description": "What should the plane’s Y axis be?", "allOf": [ { - "$ref": "#/components/schemas/SourceRange" + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "zAxis": { + "description": "The z-axis (normal).", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" } ] + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } } } }, { - "description": "An extruded arc.", + "description": "A face.", "type": "object", "required": [ - "faceId", + "__meta", "id", - "sourceRange", - "type" + "solid", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" ], "properties": { "type": { "type": "string", "enum": [ - "extrudeArc" + "Face" ] }, - "faceId": { - "description": "The face id for the extrude plane.", + "id": { + "description": "The id of the face.", "type": "string", "format": "uuid" }, - "tag": { - "description": "The tag.", + "value": { + "description": "The tag of the face.", + "type": "string" + }, + "xAxis": { + "description": "What should the face’s X axis be?", "allOf": [ { - "$ref": "#/components/schemas/TagDeclarator" + "$ref": "#/components/schemas/Point3d" } - ], - "nullable": true + ] }, - "id": { - "description": "The id of the geometry.", - "type": "string", - "format": "uuid" + "yAxis": { + "description": "What should the face’s Y axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] }, - "sourceRange": { - "description": "The source range.", + "zAxis": { + "description": "The z-axis (normal).", "allOf": [ { - "$ref": "#/components/schemas/SourceRange" + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "solid": { + "description": "The solid the face is on.", + "allOf": [ + { + "$ref": "#/components/schemas/Solid" } ] + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } } } }, { - "description": "Geometry metadata.", "type": "object", "required": [ - "faceId", - "id", - "sourceRange", - "type" + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "chamfer" + "Sketch" ] }, - "faceId": { - "description": "The id for the chamfer surface.", - "type": "string", - "format": "uuid" - }, - "tag": { - "description": "The tag.", - "allOf": [ - { - "$ref": "#/components/schemas/TagDeclarator" - } - ], - "nullable": true - }, - "id": { - "description": "The id of the geometry.", + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { "type": "string", - "format": "uuid" - }, - "sourceRange": { - "description": "The source range.", - "allOf": [ - { - "$ref": "#/components/schemas/SourceRange" - } + "enum": [ + "Sketches" ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } } } }, { - "description": "Geometry metadata.", + "description": "An solid is a collection of extrude surfaces.", "type": "object", "required": [ - "faceId", + "__meta", + "height", "id", - "sourceRange", - "type" + "sketch", + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "fillet" + "Solid" ] }, - "faceId": { - "description": "The id for the fillet surface.", + "id": { + "description": "The id of the solid.", "type": "string", "format": "uuid" }, - "tag": { - "description": "The tag.", + "value": { + "description": "The extrude surfaces.", + "type": "array", + "items": { + "$ref": "#/components/schemas/ExtrudeSurface" + } + }, + "sketch": { + "description": "The sketch.", "allOf": [ { - "$ref": "#/components/schemas/TagDeclarator" + "$ref": "#/components/schemas/Sketch" } - ], - "nullable": true - }, - "id": { - "description": "The id of the geometry.", - "type": "string", - "format": "uuid" + ] }, - "sourceRange": { - "description": "The source range.", - "allOf": [ - { - "$ref": "#/components/schemas/SourceRange" - } - ] + "height": { + "description": "The height of the solid.", + "type": "number", + "format": "double" + }, + "startCapId": { + "description": "The id of the extrusion start cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "endCapId": { + "description": "The id of the extrusion end cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "edgeCuts": { + "description": "Chamfers or fillets on this solid.", + "type": "array", + "items": { + "$ref": "#/components/schemas/EdgeCut" + } + }, + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } } } - } - ] - }, - "PlaneType": { - "description": "Type for a plane.", - "oneOf": [ - { - "type": "string", - "enum": [ - "XY", - "XZ", - "YZ" - ] }, { - "description": "A custom plane.", - "type": "string", - "enum": [ - "Custom" - ] - } - ] - }, - "Point3d": { - "type": "object", - "required": [ - "x", - "y", - "z" - ], - "properties": { - "x": { - "type": "number", - "format": "double" - }, - "y": { - "type": "number", - "format": "double" - }, - "z": { - "type": "number", - "format": "double" - } - } - }, - "Solid": { - "description": "An solid is a collection of extrude surfaces.", - "type": "object", - "required": [ - "__meta", - "height", - "id", - "sketch", - "value" - ], - "properties": { - "id": { - "description": "The id of the solid.", - "type": "string", - "format": "uuid" - }, - "value": { - "description": "The extrude surfaces.", - "type": "array", - "items": { - "$ref": "#/components/schemas/ExtrudeSurface" - } - }, - "sketch": { - "description": "The sketch.", - "allOf": [ - { - "$ref": "#/components/schemas/Sketch" - } - ] - }, - "height": { - "description": "The height of the solid.", - "type": "number", - "format": "double" - }, - "startCapId": { - "description": "The id of the extrusion start cap", - "type": "string", - "format": "uuid", - "nullable": true - }, - "endCapId": { - "description": "The id of the extrusion end cap", - "type": "string", - "format": "uuid", - "nullable": true - }, - "edgeCuts": { - "description": "Chamfers or fillets on this solid.", - "type": "array", - "items": { - "$ref": "#/components/schemas/EdgeCut" - } - }, - "__meta": { - "description": "Metadata.", - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - "Sketch": { - "description": "A sketch is a collection of paths.", - "type": "object", - "required": [ - "__meta", - "id", - "on", - "paths", - "start" - ], - "properties": { - "id": { - "description": "The id of the sketch (this will change when the engine's reference to it changes).", - "type": "string", - "format": "uuid" - }, - "paths": { - "description": "The paths in the sketch.", - "type": "array", - "items": { - "$ref": "#/components/schemas/Path" - } - }, - "on": { - "description": "What the sketch is on (can be a plane or a face).", - "allOf": [ - { - "$ref": "#/components/schemas/SketchSurface" - } - ] - }, - "start": { - "description": "The starting path.", - "allOf": [ - { - "$ref": "#/components/schemas/BasePath" - } - ] - }, - "tags": { - "description": "Tag identifiers that have been declared in this sketch.", "type": "object", - "additionalProperties": { - "$ref": "#/components/schemas/TagIdentifier" + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Solids" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Solid" + } + } } }, - "__meta": { - "description": "Metadata.", - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - "SketchSurface": { - "description": "A sketch type.", - "oneOf": [ { - "description": "A plane.", + "description": "Data for an imported geometry.", "type": "object", "required": [ "__meta", "id", - "origin", "type", - "value", - "xAxis", - "yAxis", - "zAxis" + "value" ], "properties": { "type": { "type": "string", "enum": [ - "plane" + "ImportedGeometry" ] }, "id": { - "description": "The id of the plane.", + "description": "The ID of the imported geometry.", "type": "string", "format": "uuid" }, "value": { - "$ref": "#/components/schemas/PlaneType" - }, - "origin": { - "description": "Origin of the plane.", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] + "description": "The original file paths.", + "type": "array", + "items": { + "type": "string" + } }, - "xAxis": { - "description": "What should the plane’s X axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "expression", + "memory", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Function" ] }, - "yAxis": { - "description": "What should the plane’s Y axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] + "expression": { + "$ref": "#/components/schemas/FunctionExpression" }, - "zAxis": { - "description": "The z-axis (normal).", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] + "memory": { + "$ref": "#/components/schemas/ProgramMemory" }, "__meta": { "type": "array", @@ -136832,65 +137976,21 @@ } }, { - "description": "A face.", "type": "object", "required": [ "__meta", - "id", - "solid", "type", - "value", - "xAxis", - "yAxis", - "zAxis" + "value" ], "properties": { "type": { "type": "string", "enum": [ - "face" + "KclNone" ] }, - "id": { - "description": "The id of the face.", - "type": "string", - "format": "uuid" - }, "value": { - "description": "The tag of the face.", - "type": "string" - }, - "xAxis": { - "description": "What should the face’s X axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "yAxis": { - "description": "What should the face’s Y axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "zAxis": { - "description": "The z-axis (normal).", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "solid": { - "description": "The solid the face is on.", - "allOf": [ - { - "$ref": "#/components/schemas/Solid" - } - ] + "$ref": "#/components/schemas/KclNone" }, "__meta": { "type": "array", @@ -136902,1317 +138002,1450 @@ } ] }, - "BasePath": { - "description": "A base path.", + "TagEngineInfo": { + "description": "Engine information for a tag.", "type": "object", "required": [ - "__geoMeta", - "from", - "to" + "id", + "sketch" ], "properties": { - "from": { - "description": "The from point.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "id": { + "description": "The id of the tagged object.", + "type": "string", + "format": "uuid" }, - "to": { - "description": "The to point.", - "type": "array", - "items": { - "type": "number", - "format": "double" - }, - "maxItems": 2, - "minItems": 2 + "sketch": { + "description": "The sketch the tag is on.", + "type": "string", + "format": "uuid" }, - "tag": { - "description": "The tag of the path.", + "path": { + "description": "The path the tag is on.", "allOf": [ { - "$ref": "#/components/schemas/TagDeclarator" + "$ref": "#/components/schemas/Path" } ], "nullable": true }, - "__geoMeta": { - "description": "Metadata.", - "allOf": [ - { - "$ref": "#/components/schemas/GeoMeta" - } - ] - } - } - }, - "TagIdentifier": { - "type": "object", - "required": [ - "__meta", - "value" - ], - "properties": { - "value": { - "type": "string" - }, - "info": { + "surface": { + "description": "The surface information for the tag.", "allOf": [ { - "$ref": "#/components/schemas/TagEngineInfo" + "$ref": "#/components/schemas/ExtrudeSurface" } ], "nullable": true - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } } } }, - "EdgeCut": { - "description": "A fillet or a chamfer.", + "Path": { + "description": "A path.", "oneOf": [ { - "description": "A fillet.", + "description": "A path that goes to a point.", "type": "object", "required": [ - "edgeId", - "id", - "radius", + "__geoMeta", + "from", + "to", "type" ], "properties": { "type": { "type": "string", "enum": [ - "fillet" + "ToPoint" ] }, - "id": { - "description": "The id of the engine command that called this fillet.", - "type": "string", - "format": "uuid" - }, - "radius": { - "type": "number", - "format": "double" + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, - "edgeId": { - "description": "The engine id of the edge to fillet.", - "type": "string", - "format": "uuid" + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, "tag": { + "description": "The tag of the path.", "allOf": [ { "$ref": "#/components/schemas/TagDeclarator" } ], "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] } } }, { - "description": "A chamfer.", + "description": "A arc that is tangential to the last path segment that goes to a point", "type": "object", "required": [ - "edgeId", - "id", - "length", + "__geoMeta", + "ccw", + "center", + "from", + "to", "type" ], "properties": { "type": { "type": "string", "enum": [ - "chamfer" + "TangentialArcTo" ] }, - "id": { - "description": "The id of the engine command that called this chamfer.", - "type": "string", - "format": "uuid" + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, - "length": { - "type": "number", - "format": "double" + "ccw": { + "description": "arc's direction", + "type": "boolean" }, - "edgeId": { - "description": "The engine id of the edge to chamfer.", - "type": "string", - "format": "uuid" + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, "tag": { + "description": "The tag of the path.", "allOf": [ { "$ref": "#/components/schemas/TagDeclarator" } ], "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] } } - } - ] - }, - "FunctionExpression": { - "type": "object", - "required": [ - "body", - "params" - ], - "properties": { - "params": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Parameter" - } - }, - "body": { - "$ref": "#/components/schemas/Program" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - "Parameter": { - "description": "Parameter of a KCL function.", - "type": "object", - "required": [ - "identifier", - "optional" - ], - "properties": { - "identifier": { - "description": "The parameter's label or name.", - "allOf": [ - { - "$ref": "#/components/schemas/Identifier" - } - ] - }, - "optional": { - "description": "Is the parameter optional?", - "type": "boolean" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - } - } - }, - "Identifier": { - "type": "object", - "required": [ - "name" - ], - "properties": { - "name": { - "type": "string" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - "Program": { - "description": "A KCL program top level, or function body.", - "type": "object", - "required": [ - "body" - ], - "properties": { - "body": { - "type": "array", - "items": { - "$ref": "#/components/schemas/BodyItem" - } }, - "nonCodeMeta": { - "$ref": "#/components/schemas/NonCodeMeta" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - "BodyItem": { - "oneOf": [ { + "description": "A arc that is tangential to the last path segment", "type": "object", "required": [ - "items", - "path", - "raw_path", + "__geoMeta", + "ccw", + "center", + "from", + "to", "type" ], "properties": { "type": { "type": "string", "enum": [ - "ImportStatement" + "TangentialArc" ] }, - "items": { + "center": { + "description": "the arc's center", "type": "array", "items": { - "$ref": "#/components/schemas/ImportItem" - } + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, - "path": { - "type": "string" + "ccw": { + "description": "arc's direction", + "type": "boolean" }, - "raw_path": { - "type": "string" + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, - "digest": { + "to": { + "description": "The to point.", "type": "array", "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 + "type": "number", + "format": "double" }, - "maxItems": 32, - "minItems": 32, - "nullable": true + "maxItems": 2, + "minItems": 2 }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] } } }, { + "description": "a complete arc", "type": "object", "required": [ - "expression", + "__geoMeta", + "ccw", + "center", + "from", + "radius", + "to", "type" ], "properties": { "type": { "type": "string", "enum": [ - "ExpressionStatement" + "Circle" ] }, - "expression": { - "$ref": "#/components/schemas/Expr" + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, - "digest": { + "radius": { + "description": "the arc's radius", + "type": "number", + "format": "double" + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", "type": "array", "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 + "type": "number", + "format": "double" }, - "maxItems": 32, - "minItems": 32, - "nullable": true + "maxItems": 2, + "minItems": 2 }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] } } }, { + "description": "A path that is horizontal.", "type": "object", "required": [ - "declarations", - "kind", - "type" + "__geoMeta", + "from", + "to", + "type", + "x" ], "properties": { "type": { "type": "string", "enum": [ - "VariableDeclaration" + "Horizontal" ] }, - "declarations": { + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", "type": "array", "items": { - "$ref": "#/components/schemas/VariableDeclarator" - } - }, - "visibility": { - "$ref": "#/components/schemas/ItemVisibility" - }, - "kind": { - "$ref": "#/components/schemas/VariableKind" + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, - "digest": { + "to": { + "description": "The to point.", "type": "array", "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 + "type": "number", + "format": "double" }, - "maxItems": 32, - "minItems": 32, - "nullable": true + "maxItems": 2, + "minItems": 2 }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] } } }, { + "description": "An angled line to.", "type": "object", "required": [ - "argument", + "__geoMeta", + "from", + "to", "type" ], "properties": { "type": { "type": "string", "enum": [ - "ReturnStatement" + "AngledLineTo" ] }, - "argument": { - "$ref": "#/components/schemas/Expr" + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double", + "nullable": true }, - "digest": { + "y": { + "description": "The y coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "from": { + "description": "The from point.", "type": "array", "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 + "type": "number", + "format": "double" }, - "maxItems": 32, - "minItems": 32, - "nullable": true + "maxItems": 2, + "minItems": 2 }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] } } - } - ] - }, - "ImportItem": { - "type": "object", - "required": [ - "name" - ], - "properties": { - "name": { - "description": "Name of the item to import.", - "allOf": [ - { - "$ref": "#/components/schemas/Identifier" - } - ] - }, - "alias": { - "description": "Rename the item using an identifier after \"as\".", - "allOf": [ - { - "$ref": "#/components/schemas/Identifier" - } - ], - "nullable": true - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - "Expr": { - "description": "An expression can be evaluated to yield a single KCL value.", - "oneOf": [ { + "description": "A base path.", "type": "object", "required": [ - "raw", - "type", - "value" + "__geoMeta", + "from", + "to", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "Literal" + "Base" ] }, - "value": { - "$ref": "#/components/schemas/LiteralValue" - }, - "raw": { - "type": "string" - }, - "digest": { + "from": { + "description": "The from point.", "type": "array", "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 + "type": "number", + "format": "double" }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - { - "type": "object", - "required": [ - "name", - "type" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "Identifier" - ] - }, - "name": { - "type": "string" + "maxItems": 2, + "minItems": 2 }, - "digest": { + "to": { + "description": "The to point.", "type": "array", "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 + "type": "number", + "format": "double" }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - { - "type": "object", - "required": [ - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "TagDeclarator" - ] - }, - "value": { - "type": "string" + "maxItems": 2, + "minItems": 2 }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], "nullable": true }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] } } }, { + "description": "A circular arc, not necessarily tangential to the current point.", "type": "object", "required": [ - "left", - "operator", - "right", + "__geoMeta", + "center", + "from", + "radius", + "to", "type" ], "properties": { "type": { "type": "string", "enum": [ - "BinaryExpression" + "Arc" ] }, - "operator": { - "$ref": "#/components/schemas/BinaryOperator" - }, - "left": { - "$ref": "#/components/schemas/BinaryPart" - }, - "right": { - "$ref": "#/components/schemas/BinaryPart" - }, - "digest": { + "center": { + "description": "Center of the circle that this arc is drawn on.", "type": "array", "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 + "type": "number", + "format": "double" }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "maxItems": 2, + "minItems": 2 }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - { - "type": "object", - "required": [ - "body", - "params", - "type" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "FunctionExpression" - ] + "radius": { + "description": "Radius of the circle that this arc is drawn on.", + "type": "number", + "format": "double" }, - "params": { + "from": { + "description": "The from point.", "type": "array", "items": { - "$ref": "#/components/schemas/Parameter" - } - }, - "body": { - "$ref": "#/components/schemas/Program" + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, - "digest": { + "to": { + "description": "The to point.", "type": "array", "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 + "type": "number", + "format": "double" }, - "maxItems": 32, - "minItems": 32, - "nullable": true + "maxItems": 2, + "minItems": 2 }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] } } + } + ] + }, + "TagDeclarator": { + "type": "object", + "required": [ + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "GeoMeta": { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + "ExtrudeSurface": { + "description": "An extrude surface.", + "oneOf": [ { + "description": "An extrude plane.", "type": "object", "required": [ - "arguments", - "callee", - "optional", + "faceId", + "id", + "sourceRange", "type" ], "properties": { "type": { "type": "string", "enum": [ - "CallExpression" + "extrudePlane" ] }, - "callee": { - "$ref": "#/components/schemas/Identifier" - }, - "arguments": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Expr" - } - }, - "optional": { - "type": "boolean" + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], "nullable": true }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] } } }, { + "description": "An extruded arc.", "type": "object", "required": [ - "body", + "faceId", + "id", + "sourceRange", "type" ], "properties": { "type": { "type": "string", "enum": [ - "PipeExpression" + "extrudeArc" ] }, - "body": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Expr" - } - }, - "nonCodeMeta": { - "$ref": "#/components/schemas/NonCodeMeta" + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], "nullable": true }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] } } }, { + "description": "Geometry metadata.", "type": "object", "required": [ + "faceId", + "id", + "sourceRange", "type" ], "properties": { "type": { "type": "string", "enum": [ - "PipeSubstitution" + "chamfer" ] }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], "nullable": true }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] } } }, { + "description": "Geometry metadata.", "type": "object", "required": [ - "elements", + "faceId", + "id", + "sourceRange", "type" ], "properties": { "type": { "type": "string", "enum": [ - "ArrayExpression" + "fillet" ] }, - "elements": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Expr" - } - }, - "nonCodeMeta": { - "$ref": "#/components/schemas/NonCodeMeta" + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], "nullable": true }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] } } + } + ] + }, + "PlaneType": { + "description": "Type for a plane.", + "oneOf": [ + { + "type": "string", + "enum": [ + "XY", + "XZ", + "YZ" + ] }, { - "type": "object", - "required": [ - "endElement", - "endInclusive", - "startElement", - "type" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "ArrayRangeExpression" - ] - }, - "startElement": { - "$ref": "#/components/schemas/Expr" - }, - "endElement": { - "$ref": "#/components/schemas/Expr" - }, - "endInclusive": { - "description": "Is the `end_element` included in the range?", - "type": "boolean" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "description": "A custom plane.", + "type": "string", + "enum": [ + "Custom" + ] + } + ] + }, + "Point3d": { + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "Solid": { + "description": "An solid is a collection of extrude surfaces.", + "type": "object", + "required": [ + "__meta", + "height", + "id", + "sketch", + "value" + ], + "properties": { + "id": { + "description": "The id of the solid.", + "type": "string", + "format": "uuid" + }, + "value": { + "description": "The extrude surfaces.", + "type": "array", + "items": { + "$ref": "#/components/schemas/ExtrudeSurface" + } + }, + "sketch": { + "description": "The sketch.", + "allOf": [ + { + "$ref": "#/components/schemas/Sketch" + } + ] + }, + "height": { + "description": "The height of the solid.", + "type": "number", + "format": "double" + }, + "startCapId": { + "description": "The id of the extrusion start cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "endCapId": { + "description": "The id of the extrusion end cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "edgeCuts": { + "description": "Chamfers or fillets on this solid.", + "type": "array", + "items": { + "$ref": "#/components/schemas/EdgeCut" + } + }, + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + "Sketch": { + "description": "A sketch is a collection of paths.", + "type": "object", + "required": [ + "__meta", + "id", + "on", + "paths", + "start" + ], + "properties": { + "id": { + "description": "The id of the sketch (this will change when the engine's reference to it changes).", + "type": "string", + "format": "uuid" + }, + "paths": { + "description": "The paths in the sketch.", + "type": "array", + "items": { + "$ref": "#/components/schemas/Path" + } + }, + "on": { + "description": "What the sketch is on (can be a plane or a face).", + "allOf": [ + { + "$ref": "#/components/schemas/SketchSurface" + } + ] + }, + "start": { + "description": "The starting path.", + "allOf": [ + { + "$ref": "#/components/schemas/BasePath" } + ] + }, + "tags": { + "description": "Tag identifiers that have been declared in this sketch.", + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/TagIdentifier" } }, + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + "SketchSurface": { + "description": "A sketch type.", + "oneOf": [ { + "description": "A plane.", "type": "object", "required": [ - "properties", - "type" + "__meta", + "id", + "origin", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" ], "properties": { "type": { "type": "string", "enum": [ - "ObjectExpression" + "plane" ] }, - "properties": { - "type": "array", - "items": { - "$ref": "#/components/schemas/ObjectProperty" - } + "id": { + "description": "The id of the plane.", + "type": "string", + "format": "uuid" }, - "nonCodeMeta": { - "$ref": "#/components/schemas/NonCodeMeta" + "value": { + "$ref": "#/components/schemas/PlaneType" }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true + "origin": { + "description": "Origin of the plane.", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "xAxis": { + "description": "What should the plane’s X axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "yAxis": { + "description": "What should the plane’s Y axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "zAxis": { + "description": "The z-axis (normal).", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } } } }, { + "description": "A face.", "type": "object", "required": [ - "computed", - "object", - "property", - "type" + "__meta", + "id", + "solid", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" ], "properties": { "type": { "type": "string", "enum": [ - "MemberExpression" + "face" ] }, - "object": { - "$ref": "#/components/schemas/MemberObject" + "id": { + "description": "The id of the face.", + "type": "string", + "format": "uuid" }, - "property": { - "$ref": "#/components/schemas/LiteralIdentifier" + "value": { + "description": "The tag of the face.", + "type": "string" }, - "computed": { - "type": "boolean" + "xAxis": { + "description": "What should the face’s X axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true + "yAxis": { + "description": "What should the face’s Y axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "zAxis": { + "description": "The z-axis (normal).", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "solid": { + "description": "The solid the face is on.", + "allOf": [ + { + "$ref": "#/components/schemas/Solid" + } + ] + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } } } + } + ] + }, + "BasePath": { + "description": "A base path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to" + ], + "properties": { + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + "TagIdentifier": { + "type": "object", + "required": [ + "__meta", + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "info": { + "allOf": [ + { + "$ref": "#/components/schemas/TagEngineInfo" + } + ], + "nullable": true }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + "EdgeCut": { + "description": "A fillet or a chamfer.", + "oneOf": [ { + "description": "A fillet.", "type": "object", "required": [ - "argument", - "operator", + "edgeId", + "id", + "radius", "type" ], "properties": { "type": { "type": "string", "enum": [ - "UnaryExpression" + "fillet" ] }, - "operator": { - "$ref": "#/components/schemas/UnaryOperator" - }, - "argument": { - "$ref": "#/components/schemas/BinaryPart" + "id": { + "description": "The id of the engine command that called this fillet.", + "type": "string", + "format": "uuid" }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true + "radius": { + "type": "number", + "format": "double" }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "edgeId": { + "description": "The engine id of the edge to fillet.", + "type": "string", + "format": "uuid" }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + "tag": { + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true } } }, { + "description": "A chamfer.", "type": "object", "required": [ - "cond", - "else_ifs", - "final_else", - "then_val", + "edgeId", + "id", + "length", "type" ], "properties": { "type": { "type": "string", "enum": [ - "IfExpression" + "chamfer" ] }, - "cond": { - "$ref": "#/components/schemas/Expr" - }, - "then_val": { - "$ref": "#/components/schemas/Program" + "id": { + "description": "The id of the engine command that called this chamfer.", + "type": "string", + "format": "uuid" }, - "else_ifs": { - "type": "array", - "items": { - "$ref": "#/components/schemas/ElseIf" - } + "length": { + "type": "number", + "format": "double" }, - "final_else": { - "$ref": "#/components/schemas/Program" + "edgeId": { + "description": "The engine id of the edge to chamfer.", + "type": "string", + "format": "uuid" }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, + "tag": { + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - { - "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", - "type": "object", - "required": [ - "type" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "None" - ] - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 } } } ] }, - "LiteralValue": { - "anyOf": [ - { - "type": "integer", - "format": "int64" + "FunctionExpression": { + "type": "object", + "required": [ + "body", + "params" + ], + "properties": { + "params": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Parameter" + } }, - { - "type": "number", - "format": "double" + "body": { + "$ref": "#/components/schemas/Program" }, - { - "type": "string" + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true }, - { - "type": "boolean" + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } - ] + } }, - "BinaryOperator": { - "oneOf": [ - { - "description": "Add two numbers.", - "type": "string", - "enum": [ - "+" - ] - }, - { - "description": "Subtract two numbers.", - "type": "string", - "enum": [ - "-" - ] - }, - { - "description": "Multiply two numbers.", - "type": "string", - "enum": [ - "*" + "Parameter": { + "description": "Parameter of a KCL function.", + "type": "object", + "required": [ + "identifier", + "optional" + ], + "properties": { + "identifier": { + "description": "The parameter's label or name.", + "allOf": [ + { + "$ref": "#/components/schemas/Identifier" + } ] }, - { - "description": "Divide two numbers.", - "type": "string", - "enum": [ - "/" - ] + "optional": { + "description": "Is the parameter optional?", + "type": "boolean" }, - { - "description": "Modulo two numbers.", - "type": "string", - "enum": [ - "%" - ] + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + } + } + }, + "Identifier": { + "type": "object", + "required": [ + "name" + ], + "properties": { + "name": { + "type": "string" }, - { - "description": "Raise a number to a power.", - "type": "string", - "enum": [ - "^" - ] + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true }, - { - "description": "Are two numbers equal?", - "type": "string", - "enum": [ - "==" - ] + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 }, - { - "description": "Are two numbers not equal?", - "type": "string", - "enum": [ - "!=" - ] + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "Program": { + "description": "A KCL program top level, or function body.", + "type": "object", + "required": [ + "body" + ], + "properties": { + "body": { + "type": "array", + "items": { + "$ref": "#/components/schemas/BodyItem" + } }, - { - "description": "Is left greater than right", - "type": "string", - "enum": [ - ">" - ] + "nonCodeMeta": { + "$ref": "#/components/schemas/NonCodeMeta" }, - { - "description": "Is left greater than or equal to right", - "type": "string", - "enum": [ - ">=" - ] + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true }, - { - "description": "Is left less than right", - "type": "string", - "enum": [ - "<" - ] + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 }, - { - "description": "Is left less than or equal to right", - "type": "string", - "enum": [ - "<=" - ] + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } - ] + } }, - "BinaryPart": { + "BodyItem": { "oneOf": [ { "type": "object", "required": [ - "raw", - "type", - "value" + "items", + "path", + "raw_path", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "Literal" + "ImportStatement" ] }, - "value": { - "$ref": "#/components/schemas/LiteralValue" + "items": { + "type": "array", + "items": { + "$ref": "#/components/schemas/ImportItem" + } }, - "raw": { + "path": { + "type": "string" + }, + "raw_path": { "type": "string" }, "digest": { @@ -138241,18 +139474,18 @@ { "type": "object", "required": [ - "name", + "expression", "type" ], "properties": { "type": { "type": "string", "enum": [ - "Identifier" + "ExpressionStatement" ] }, - "name": { - "type": "string" + "expression": { + "$ref": "#/components/schemas/Expr" }, "digest": { "type": "array", @@ -138280,26 +139513,28 @@ { "type": "object", "required": [ - "left", - "operator", - "right", + "declarations", + "kind", "type" ], "properties": { "type": { "type": "string", "enum": [ - "BinaryExpression" + "VariableDeclaration" ] }, - "operator": { - "$ref": "#/components/schemas/BinaryOperator" + "declarations": { + "type": "array", + "items": { + "$ref": "#/components/schemas/VariableDeclarator" + } }, - "left": { - "$ref": "#/components/schemas/BinaryPart" + "visibility": { + "$ref": "#/components/schemas/ItemVisibility" }, - "right": { - "$ref": "#/components/schemas/BinaryPart" + "kind": { + "$ref": "#/components/schemas/VariableKind" }, "digest": { "type": "array", @@ -138327,29 +139562,18 @@ { "type": "object", "required": [ - "arguments", - "callee", - "optional", + "argument", "type" ], "properties": { "type": { "type": "string", "enum": [ - "CallExpression" + "ReturnStatement" ] }, - "callee": { - "$ref": "#/components/schemas/Identifier" - }, - "arguments": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Expr" - } - }, - "optional": { - "type": "boolean" + "argument": { + "$ref": "#/components/schemas/Expr" }, "digest": { "type": "array", @@ -138373,26 +139597,77 @@ "minimum": 0.0 } } + } + ] + }, + "ImportItem": { + "type": "object", + "required": [ + "name" + ], + "properties": { + "name": { + "description": "Name of the item to import.", + "allOf": [ + { + "$ref": "#/components/schemas/Identifier" + } + ] + }, + "alias": { + "description": "Rename the item using an identifier after \"as\".", + "allOf": [ + { + "$ref": "#/components/schemas/Identifier" + } + ], + "nullable": true + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "Expr": { + "description": "An expression can be evaluated to yield a single KCL value.", + "oneOf": [ { "type": "object", "required": [ - "argument", - "operator", - "type" + "raw", + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "UnaryExpression" + "Literal" ] }, - "operator": { - "$ref": "#/components/schemas/UnaryOperator" + "value": { + "$ref": "#/components/schemas/LiteralValue" }, - "argument": { - "$ref": "#/components/schemas/BinaryPart" + "raw": { + "type": "string" }, "digest": { "type": "array", @@ -138420,26 +139695,18 @@ { "type": "object", "required": [ - "computed", - "object", - "property", + "name", "type" ], "properties": { "type": { "type": "string", "enum": [ - "MemberExpression" + "Identifier" ] }, - "object": { - "$ref": "#/components/schemas/MemberObject" - }, - "property": { - "$ref": "#/components/schemas/LiteralIdentifier" - }, - "computed": { - "type": "boolean" + "name": { + "type": "string" }, "digest": { "type": "array", @@ -138467,33 +139734,18 @@ { "type": "object", "required": [ - "cond", - "else_ifs", - "final_else", - "then_val", - "type" + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "IfExpression" + "TagDeclarator" ] }, - "cond": { - "$ref": "#/components/schemas/Expr" - }, - "then_val": { - "$ref": "#/components/schemas/Program" - }, - "else_ifs": { - "type": "array", - "items": { - "$ref": "#/components/schemas/ElseIf" - } - }, - "final_else": { - "$ref": "#/components/schemas/Program" + "value": { + "type": "string" }, "digest": { "type": "array", @@ -138517,52 +139769,30 @@ "minimum": 0.0 } } - } - ] - }, - "UnaryOperator": { - "oneOf": [ - { - "description": "Negate a number.", - "type": "string", - "enum": [ - "-" - ] }, - { - "description": "Negate a boolean.", - "type": "string", - "enum": [ - "!" - ] - } - ] - }, - "MemberObject": { - "oneOf": [ { "type": "object", "required": [ - "computed", - "object", - "property", + "left", + "operator", + "right", "type" ], "properties": { "type": { "type": "string", "enum": [ - "MemberExpression" + "BinaryExpression" ] }, - "object": { - "$ref": "#/components/schemas/MemberObject" + "operator": { + "$ref": "#/components/schemas/BinaryOperator" }, - "property": { - "$ref": "#/components/schemas/LiteralIdentifier" + "left": { + "$ref": "#/components/schemas/BinaryPart" }, - "computed": { - "type": "boolean" + "right": { + "$ref": "#/components/schemas/BinaryPart" }, "digest": { "type": "array", @@ -138590,18 +139820,25 @@ { "type": "object", "required": [ - "name", + "body", + "params", "type" ], "properties": { "type": { "type": "string", "enum": [ - "Identifier" + "FunctionExpression" ] }, - "name": { - "type": "string" + "params": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Parameter" + } + }, + "body": { + "$ref": "#/components/schemas/Program" }, "digest": { "type": "array", @@ -138625,26 +139862,33 @@ "minimum": 0.0 } } - } - ] - }, - "LiteralIdentifier": { - "oneOf": [ + }, { "type": "object", "required": [ - "name", + "arguments", + "callee", + "optional", "type" ], "properties": { "type": { "type": "string", "enum": [ - "Identifier" + "CallExpression" ] }, - "name": { - "type": "string" + "callee": { + "$ref": "#/components/schemas/Identifier" + }, + "arguments": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Expr" + } + }, + "optional": { + "type": "boolean" }, "digest": { "type": "array", @@ -138672,22 +139916,24 @@ { "type": "object", "required": [ - "raw", - "type", - "value" + "body", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "Literal" + "PipeExpression" ] }, - "value": { - "$ref": "#/components/schemas/LiteralValue" + "body": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Expr" + } }, - "raw": { - "type": "string" + "nonCodeMeta": { + "$ref": "#/components/schemas/NonCodeMeta" }, "digest": { "type": "array", @@ -138711,203 +139957,326 @@ "minimum": 0.0 } } - } - ] - }, - "ElseIf": { - "type": "object", - "required": [ - "cond", - "then_val" - ], - "properties": { - "cond": { - "$ref": "#/components/schemas/Expr" - }, - "then_val": { - "$ref": "#/components/schemas/Program" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - "NonCodeMeta": { - "type": "object", - "required": [ - "nonCodeNodes", - "startNodes" - ], - "properties": { - "nonCodeNodes": { + { "type": "object", - "additionalProperties": { - "type": "array", - "items": { - "$ref": "#/components/schemas/NonCodeNode" + "required": [ + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "PipeSubstitution" + ] + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, - "startNodes": { - "type": "array", - "items": { - "$ref": "#/components/schemas/NonCodeNode" + { + "type": "object", + "required": [ + "elements", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "ArrayExpression" + ] + }, + "elements": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Expr" + } + }, + "nonCodeMeta": { + "$ref": "#/components/schemas/NonCodeMeta" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } } }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - } - } - }, - "NonCodeNode": { - "type": "object", - "required": [ - "value" - ], - "properties": { - "value": { - "$ref": "#/components/schemas/NonCodeValue" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + { + "type": "object", + "required": [ + "endElement", + "endInclusive", + "startElement", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "ArrayRangeExpression" + ] + }, + "startElement": { + "$ref": "#/components/schemas/Expr" + }, + "endElement": { + "$ref": "#/components/schemas/Expr" + }, + "endInclusive": { + "description": "Is the `end_element` included in the range?", + "type": "boolean" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - "NonCodeValue": { - "oneOf": [ { - "description": "A shebang. This is a special type of comment that is at the top of the file. It looks like this: ```python,no_run #!/usr/bin/env python ```", "type": "object", "required": [ - "type", - "value" + "properties", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "shebang" + "ObjectExpression" ] }, - "value": { - "type": "string" + "properties": { + "type": "array", + "items": { + "$ref": "#/components/schemas/ObjectProperty" + } + }, + "nonCodeMeta": { + "$ref": "#/components/schemas/NonCodeMeta" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, { - "description": "An inline comment. Here are examples: `1 + 1 // This is an inline comment`. `1 + 1 /* Here's another */`.", "type": "object", "required": [ - "style", - "type", - "value" + "computed", + "object", + "property", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "inlineComment" + "MemberExpression" ] }, - "value": { - "type": "string" + "object": { + "$ref": "#/components/schemas/MemberObject" }, - "style": { - "$ref": "#/components/schemas/CommentStyle" + "property": { + "$ref": "#/components/schemas/LiteralIdentifier" + }, + "computed": { + "type": "boolean" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, { - "description": "A block comment. An example of this is the following: ```python,no_run /* This is a block comment */ 1 + 1 ``` Now this is important. The block comment is attached to the next line. This is always the case. Also the block comment doesn't have a new line above it. If it did it would be a `NewLineBlockComment`.", "type": "object", "required": [ - "style", - "type", - "value" + "argument", + "operator", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "blockComment" + "UnaryExpression" ] }, - "value": { - "type": "string" + "operator": { + "$ref": "#/components/schemas/UnaryOperator" }, - "style": { - "$ref": "#/components/schemas/CommentStyle" + "argument": { + "$ref": "#/components/schemas/BinaryPart" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, { - "description": "A block comment that has a new line above it. The user explicitly added a new line above the block comment.", "type": "object", "required": [ - "style", - "type", - "value" + "cond", + "else_ifs", + "final_else", + "then_val", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "newLineBlockComment" + "IfExpression" ] }, - "value": { - "type": "string" + "cond": { + "$ref": "#/components/schemas/Expr" }, - "style": { - "$ref": "#/components/schemas/CommentStyle" + "then_val": { + "$ref": "#/components/schemas/Program" + }, + "else_ifs": { + "type": "array", + "items": { + "$ref": "#/components/schemas/ElseIf" + } + }, + "final_else": { + "$ref": "#/components/schemas/Program" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ "type" @@ -138916,216 +140285,135 @@ "type": { "type": "string", "enum": [ - "newLine" + "None" ] + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } } ] }, - "CommentStyle": { + "LiteralValue": { + "anyOf": [ + { + "type": "integer", + "format": "int64" + }, + { + "type": "number", + "format": "double" + }, + { + "type": "string" + }, + { + "type": "boolean" + } + ] + }, + "BinaryOperator": { "oneOf": [ { - "description": "Like // foo", + "description": "Add two numbers.", "type": "string", "enum": [ - "line" + "+" ] }, { - "description": "Like /* foo */", + "description": "Subtract two numbers.", "type": "string", "enum": [ - "block" + "-" ] - } - ] - }, - "ObjectProperty": { - "type": "object", - "required": [ - "key", - "value" - ], - "properties": { - "key": { - "$ref": "#/components/schemas/Identifier" }, - "value": { - "$ref": "#/components/schemas/Expr" + { + "description": "Multiply two numbers.", + "type": "string", + "enum": [ + "*" + ] }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true + { + "description": "Divide two numbers.", + "type": "string", + "enum": [ + "/" + ] }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + { + "description": "Modulo two numbers.", + "type": "string", + "enum": [ + "%" + ] }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - "VariableDeclarator": { - "type": "object", - "required": [ - "id", - "init" - ], - "properties": { - "id": { - "description": "The identifier of the variable.", - "allOf": [ - { - "$ref": "#/components/schemas/Identifier" - } + { + "description": "Raise a number to a power.", + "type": "string", + "enum": [ + "^" ] }, - "init": { - "description": "The value of the variable.", - "allOf": [ - { - "$ref": "#/components/schemas/Expr" - } + { + "description": "Are two numbers equal?", + "type": "string", + "enum": [ + "==" ] }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - "ItemVisibility": { - "type": "string", - "enum": [ - "default", - "export" - ] - }, - "VariableKind": { - "oneOf": [ { - "description": "Declare a named constant.", + "description": "Are two numbers not equal?", "type": "string", "enum": [ - "const" + "!=" ] }, { - "description": "Declare a function.", + "description": "Is left greater than right", "type": "string", "enum": [ - "fn" + ">" ] - } - ] - }, - "ProgramMemory": { - "type": "object", - "required": [ - "currentEnv", - "environments" - ], - "properties": { - "environments": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Environment" - } }, - "currentEnv": { - "type": "integer", - "format": "uint", - "minimum": 0.0 + { + "description": "Is left greater than or equal to right", + "type": "string", + "enum": [ + ">=" + ] }, - "return": { - "allOf": [ - { - "$ref": "#/components/schemas/KclValue" - } - ], - "nullable": true - } - } - }, - "Environment": { - "type": "object", - "required": [ - "bindings" - ], - "properties": { - "bindings": { - "type": "object", - "additionalProperties": { - "$ref": "#/components/schemas/KclValue" - } + { + "description": "Is left less than right", + "type": "string", + "enum": [ + "<" + ] }, - "parent": { - "type": "integer", - "format": "uint", - "minimum": 0.0, - "nullable": true + { + "description": "Is left less than or equal to right", + "type": "string", + "enum": [ + "<=" + ] } - } + ] }, - "KclValue": { - "description": "Any KCL value.", + "BinaryPart": { "oneOf": [ { "type": "object", "required": [ - "__meta", - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "UserVal" - ] - }, - "value": {}, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "type": "object", - "required": [ - "__meta", + "raw", "type", "value" ], @@ -139133,42 +140421,52 @@ "type": { "type": "string", "enum": [ - "TagIdentifier" + "Literal" ] }, "value": { - "type": "string" + "$ref": "#/components/schemas/LiteralValue" }, - "info": { - "allOf": [ - { - "$ref": "#/components/schemas/TagEngineInfo" - } - ], - "nullable": true + "raw": { + "type": "string" }, - "__meta": { + "digest": { "type": "array", "items": { - "$ref": "#/components/schemas/Metadata" - } + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, { "type": "object", "required": [ - "type", - "value" + "name", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "TagDeclarator" + "Identifier" ] }, - "value": { + "name": { "type": "string" }, "digest": { @@ -139195,627 +140493,954 @@ } }, { - "description": "A plane.", "type": "object", "required": [ - "__meta", - "id", - "origin", - "type", - "value", - "xAxis", - "yAxis", - "zAxis" + "left", + "operator", + "right", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "Plane" - ] - }, - "id": { - "description": "The id of the plane.", - "type": "string", - "format": "uuid" - }, - "value": { - "$ref": "#/components/schemas/PlaneType" - }, - "origin": { - "description": "Origin of the plane.", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } + "BinaryExpression" ] }, - "xAxis": { - "description": "What should the plane’s X axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] + "operator": { + "$ref": "#/components/schemas/BinaryOperator" }, - "yAxis": { - "description": "What should the plane’s Y axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] + "left": { + "$ref": "#/components/schemas/BinaryPart" }, - "zAxis": { - "description": "The z-axis (normal).", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] + "right": { + "$ref": "#/components/schemas/BinaryPart" }, - "__meta": { + "digest": { "type": "array", "items": { - "$ref": "#/components/schemas/Metadata" - } + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, { - "description": "A face.", "type": "object", "required": [ - "__meta", - "id", - "solid", - "type", - "value", - "xAxis", - "yAxis", - "zAxis" + "arguments", + "callee", + "optional", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "Face" - ] - }, - "id": { - "description": "The id of the face.", - "type": "string", - "format": "uuid" - }, - "value": { - "description": "The tag of the face.", - "type": "string" - }, - "xAxis": { - "description": "What should the face’s X axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } + "CallExpression" ] }, - "yAxis": { - "description": "What should the face’s Y axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] + "callee": { + "$ref": "#/components/schemas/Identifier" }, - "zAxis": { - "description": "The z-axis (normal).", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] + "arguments": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Expr" + } }, - "solid": { - "description": "The solid the face is on.", - "allOf": [ - { - "$ref": "#/components/schemas/Solid" - } - ] + "optional": { + "type": "boolean" }, - "__meta": { + "digest": { "type": "array", "items": { - "$ref": "#/components/schemas/Metadata" - } + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, { - "description": "An solid is a collection of extrude surfaces.", "type": "object", "required": [ - "__meta", - "height", - "id", - "sketch", - "type", - "value" + "argument", + "operator", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "Solid" + "UnaryExpression" ] }, - "id": { - "description": "The id of the solid.", - "type": "string", - "format": "uuid" + "operator": { + "$ref": "#/components/schemas/UnaryOperator" }, - "value": { - "description": "The extrude surfaces.", + "argument": { + "$ref": "#/components/schemas/BinaryPart" + }, + "digest": { "type": "array", "items": { - "$ref": "#/components/schemas/ExtrudeSurface" - } - }, - "sketch": { - "description": "The sketch.", - "allOf": [ - { - "$ref": "#/components/schemas/Sketch" - } - ] - }, - "height": { - "description": "The height of the solid.", - "type": "number", - "format": "double" - }, - "startCapId": { - "description": "The id of the extrusion start cap", - "type": "string", - "format": "uuid", - "nullable": true - }, - "endCapId": { - "description": "The id of the extrusion end cap", - "type": "string", - "format": "uuid", + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, "nullable": true }, - "edgeCuts": { - "description": "Chamfers or fillets on this solid.", - "type": "array", - "items": { - "$ref": "#/components/schemas/EdgeCut" - } + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 }, - "__meta": { - "description": "Metadata.", - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, { "type": "object", "required": [ - "type", - "value" + "computed", + "object", + "property", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "Solids" + "MemberExpression" ] }, - "value": { + "object": { + "$ref": "#/components/schemas/MemberObject" + }, + "property": { + "$ref": "#/components/schemas/LiteralIdentifier" + }, + "computed": { + "type": "boolean" + }, + "digest": { "type": "array", "items": { - "$ref": "#/components/schemas/Solid" - } + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } }, { - "description": "Data for an imported geometry.", "type": "object", "required": [ - "__meta", - "id", - "type", - "value" + "cond", + "else_ifs", + "final_else", + "then_val", + "type" ], "properties": { "type": { "type": "string", "enum": [ - "ImportedGeometry" + "IfExpression" ] }, - "id": { - "description": "The ID of the imported geometry.", - "type": "string", - "format": "uuid" + "cond": { + "$ref": "#/components/schemas/Expr" }, - "value": { - "description": "The original file paths.", + "then_val": { + "$ref": "#/components/schemas/Program" + }, + "else_ifs": { "type": "array", "items": { - "type": "string" + "$ref": "#/components/schemas/ElseIf" } }, - "__meta": { + "final_else": { + "$ref": "#/components/schemas/Program" + }, + "digest": { "type": "array", "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + } + ] + }, + "UnaryOperator": { + "oneOf": [ + { + "description": "Negate a number.", + "type": "string", + "enum": [ + "-" + ] + }, + { + "description": "Negate a boolean.", + "type": "string", + "enum": [ + "!" + ] + } + ] + }, + "MemberObject": { + "oneOf": [ + { "type": "object", "required": [ - "__meta", - "expression", - "memory", + "computed", + "object", + "property", "type" ], "properties": { "type": { "type": "string", "enum": [ - "Function" + "MemberExpression" ] }, - "expression": { - "$ref": "#/components/schemas/FunctionExpression" + "object": { + "$ref": "#/components/schemas/MemberObject" }, - "memory": { - "$ref": "#/components/schemas/ProgramMemory" + "property": { + "$ref": "#/components/schemas/LiteralIdentifier" }, - "__meta": { + "computed": { + "type": "boolean" + }, + "digest": { "type": "array", "items": { - "$ref": "#/components/schemas/Metadata" - } + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + { + "type": "object", + "required": [ + "name", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Identifier" + ] + }, + "name": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } } } ] - } - } - }, - "required": true - }, - "unpublished": false, - "deprecated": false, - "examples": [ - "arr = [1, 2, 3]\nnew_arr = push(arr, 4)\nassertEqual(new_arr[3], 4, 0.00001, \"4 was added to the end of the array\")" - ] - }, - { - "name": "reduce", - "summary": "Take a starting value. Then, for each element of an array, calculate the next value,", - "description": "using the previous value and the element.", - "tags": [], - "args": [ - { - "name": "array", - "type": "[KclValue]", - "schema": { - "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", - "title": "Array_of_KclValue", - "type": "array", - "items": { - "$ref": "#/components/schemas/KclValue" }, - "definitions": { - "KclValue": { - "description": "Any KCL value.", - "oneOf": [ - { - "type": "object", - "required": [ - "__meta", - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "UserVal" - ] - }, - "value": {}, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "type": "object", - "required": [ - "__meta", - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "TagIdentifier" - ] - }, - "value": { - "type": "string" - }, - "info": { - "allOf": [ - { - "$ref": "#/components/schemas/TagEngineInfo" - } - ], - "nullable": true - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "type": "object", - "required": [ - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "TagDeclarator" - ] - }, - "value": { - "type": "string" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { + "LiteralIdentifier": { + "oneOf": [ + { + "type": "object", + "required": [ + "name", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Identifier" + ] + }, + "name": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { "type": "integer", - "format": "uint", + "format": "uint8", "minimum": 0.0 }, - "end": { + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + { + "type": "object", + "required": [ + "raw", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Literal" + ] + }, + "value": { + "$ref": "#/components/schemas/LiteralValue" + }, + "raw": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { "type": "integer", - "format": "uint", + "format": "uint8", "minimum": 0.0 - } - } - }, - { - "description": "A plane.", - "type": "object", - "required": [ - "__meta", - "id", - "origin", - "type", - "value", - "xAxis", - "yAxis", - "zAxis" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "Plane" - ] - }, - "id": { - "description": "The id of the plane.", - "type": "string", - "format": "uuid" - }, - "value": { - "$ref": "#/components/schemas/PlaneType" - }, - "origin": { - "description": "Origin of the plane.", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "xAxis": { - "description": "What should the plane’s X axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "yAxis": { - "description": "What should the plane’s Y axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "zAxis": { - "description": "The z-axis (normal).", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 } + } + } + ] + }, + "ElseIf": { + "type": "object", + "required": [ + "cond", + "then_val" + ], + "properties": { + "cond": { + "$ref": "#/components/schemas/Expr" + }, + "then_val": { + "$ref": "#/components/schemas/Program" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 }, - { - "description": "A face.", - "type": "object", - "required": [ - "__meta", - "id", - "solid", - "type", - "value", - "xAxis", - "yAxis", - "zAxis" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "Face" - ] - }, - "id": { - "description": "The id of the face.", - "type": "string", - "format": "uuid" - }, - "value": { - "description": "The tag of the face.", - "type": "string" - }, - "xAxis": { - "description": "What should the face’s X axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "yAxis": { - "description": "What should the face’s Y axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "zAxis": { - "description": "The z-axis (normal).", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "solid": { - "description": "The solid the face is on.", - "allOf": [ - { - "$ref": "#/components/schemas/Solid" - } - ] - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "NonCodeMeta": { + "type": "object", + "required": [ + "nonCodeNodes", + "startNodes" + ], + "properties": { + "nonCodeNodes": { + "type": "object", + "additionalProperties": { + "type": "array", + "items": { + "$ref": "#/components/schemas/NonCodeNode" } - }, - { - "description": "An solid is a collection of extrude surfaces.", - "type": "object", - "required": [ - "__meta", - "height", - "id", - "sketch", - "type", - "value" - ], - "properties": { + } + }, + "startNodes": { + "type": "array", + "items": { + "$ref": "#/components/schemas/NonCodeNode" + } + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + } + } + }, + "NonCodeNode": { + "type": "object", + "required": [ + "value" + ], + "properties": { + "value": { + "$ref": "#/components/schemas/NonCodeValue" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "NonCodeValue": { + "oneOf": [ + { + "description": "A shebang. This is a special type of comment that is at the top of the file. It looks like this: ```python,no_run #!/usr/bin/env python ```", + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "shebang" + ] + }, + "value": { + "type": "string" + } + } + }, + { + "description": "An inline comment. Here are examples: `1 + 1 // This is an inline comment`. `1 + 1 /* Here's another */`.", + "type": "object", + "required": [ + "style", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "inlineComment" + ] + }, + "value": { + "type": "string" + }, + "style": { + "$ref": "#/components/schemas/CommentStyle" + } + } + }, + { + "description": "A block comment. An example of this is the following: ```python,no_run /* This is a block comment */ 1 + 1 ``` Now this is important. The block comment is attached to the next line. This is always the case. Also the block comment doesn't have a new line above it. If it did it would be a `NewLineBlockComment`.", + "type": "object", + "required": [ + "style", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "blockComment" + ] + }, + "value": { + "type": "string" + }, + "style": { + "$ref": "#/components/schemas/CommentStyle" + } + } + }, + { + "description": "A block comment that has a new line above it. The user explicitly added a new line above the block comment.", + "type": "object", + "required": [ + "style", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "newLineBlockComment" + ] + }, + "value": { + "type": "string" + }, + "style": { + "$ref": "#/components/schemas/CommentStyle" + } + } + }, + { + "type": "object", + "required": [ + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "newLine" + ] + } + } + } + ] + }, + "CommentStyle": { + "oneOf": [ + { + "description": "Like // foo", + "type": "string", + "enum": [ + "line" + ] + }, + { + "description": "Like /* foo */", + "type": "string", + "enum": [ + "block" + ] + } + ] + }, + "ObjectProperty": { + "type": "object", + "required": [ + "key", + "value" + ], + "properties": { + "key": { + "$ref": "#/components/schemas/Identifier" + }, + "value": { + "$ref": "#/components/schemas/Expr" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "VariableDeclarator": { + "type": "object", + "required": [ + "id", + "init" + ], + "properties": { + "id": { + "description": "The identifier of the variable.", + "allOf": [ + { + "$ref": "#/components/schemas/Identifier" + } + ] + }, + "init": { + "description": "The value of the variable.", + "allOf": [ + { + "$ref": "#/components/schemas/Expr" + } + ] + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "ItemVisibility": { + "type": "string", + "enum": [ + "default", + "export" + ] + }, + "VariableKind": { + "oneOf": [ + { + "description": "Declare a named constant.", + "type": "string", + "enum": [ + "const" + ] + }, + { + "description": "Declare a function.", + "type": "string", + "enum": [ + "fn" + ] + } + ] + }, + "ProgramMemory": { + "type": "object", + "required": [ + "currentEnv", + "environments" + ], + "properties": { + "environments": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Environment" + } + }, + "currentEnv": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "return": { + "allOf": [ + { + "$ref": "#/components/schemas/KclValue" + } + ], + "nullable": true + } + } + }, + "Environment": { + "type": "object", + "required": [ + "bindings" + ], + "properties": { + "bindings": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, + "parent": { + "type": "integer", + "format": "uint", + "minimum": 0.0, + "nullable": true + } + } + }, + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" + } + } + }, + "required": true + }, + "unpublished": false, + "deprecated": false, + "examples": [ + "arr = [1, 2, 3]\nnew_arr = push(arr, 4)\nassertEqual(new_arr[3], 4, 0.00001, \"4 was added to the end of the array\")" + ] + }, + { + "name": "reduce", + "summary": "Take a starting value. Then, for each element of an array, calculate the next value,", + "description": "using the previous value and the element.", + "tags": [], + "args": [ + { + "name": "array", + "type": "[KclValue]", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "Array_of_KclValue", + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + }, + "definitions": { + "KclValue": { + "description": "Any KCL value.", + "oneOf": [ + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { "type": { "type": "string", "enum": [ - "Solid" + "Uuid" ] }, - "id": { - "description": "The id of the solid.", + "value": { "type": "string", "format": "uuid" }, - "value": { - "description": "The extrude surfaces.", + "__meta": { "type": "array", "items": { - "$ref": "#/components/schemas/ExtrudeSurface" + "$ref": "#/components/schemas/Metadata" } - }, - "sketch": { - "description": "The sketch.", - "allOf": [ - { - "$ref": "#/components/schemas/Sketch" - } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" ] }, - "height": { - "description": "The height of the solid.", - "type": "number", - "format": "double" + "value": { + "type": "boolean" }, - "startCapId": { - "description": "The id of the extrusion start cap", + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { "type": "string", - "format": "uuid", - "nullable": true + "enum": [ + "Number" + ] }, - "endCapId": { - "description": "The id of the extrusion end cap", - "type": "string", - "format": "uuid", - "nullable": true + "value": { + "type": "number", + "format": "double" }, - "edgeCuts": { - "description": "Chamfers or fillets on this solid.", + "__meta": { "type": "array", "items": { - "$ref": "#/components/schemas/EdgeCut" + "$ref": "#/components/schemas/Metadata" } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" }, "__meta": { - "description": "Metadata.", "type": "array", "items": { "$ref": "#/components/schemas/Metadata" @@ -139826,6 +141451,7 @@ { "type": "object", "required": [ + "__meta", "type", "value" ], @@ -139833,23 +141459,24 @@ "type": { "type": "string", "enum": [ - "Solids" + "String" ] }, "value": { + "type": "string" + }, + "__meta": { "type": "array", "items": { - "$ref": "#/components/schemas/Solid" + "$ref": "#/components/schemas/Metadata" } } } }, { - "description": "Data for an imported geometry.", "type": "object", "required": [ "__meta", - "id", "type", "value" ], @@ -139857,19 +141484,13 @@ "type": { "type": "string", "enum": [ - "ImportedGeometry" + "Array" ] }, - "id": { - "description": "The ID of the imported geometry.", - "type": "string", - "format": "uuid" - }, "value": { - "description": "The original file paths.", "type": "array", "items": { - "type": "string" + "$ref": "#/components/schemas/KclValue" } }, "__meta": { @@ -139884,22 +141505,21 @@ "type": "object", "required": [ "__meta", - "expression", - "memory", - "type" + "type", + "value" ], "properties": { "type": { "type": "string", "enum": [ - "Function" + "Object" ] }, - "expression": { - "$ref": "#/components/schemas/FunctionExpression" - }, - "memory": { - "$ref": "#/components/schemas/ProgramMemory" + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } }, "__meta": { "type": "array", @@ -139908,25 +141528,455 @@ } } } - } - ] - }, - "Metadata": { - "description": "Metadata.", - "type": "object", - "required": [ - "sourceRange" - ], - "properties": { - "sourceRange": { - "description": "The source range.", - "allOf": [ - { - "$ref": "#/components/schemas/SourceRange" - } - ] - } - } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TagIdentifier" + ] + }, + "value": { + "type": "string" + }, + "info": { + "allOf": [ + { + "$ref": "#/components/schemas/TagEngineInfo" + } + ], + "nullable": true + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TagDeclarator" + ] + }, + "value": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + { + "description": "A plane.", + "type": "object", + "required": [ + "__meta", + "id", + "origin", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Plane" + ] + }, + "id": { + "description": "The id of the plane.", + "type": "string", + "format": "uuid" + }, + "value": { + "$ref": "#/components/schemas/PlaneType" + }, + "origin": { + "description": "Origin of the plane.", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "xAxis": { + "description": "What should the plane’s X axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "yAxis": { + "description": "What should the plane’s Y axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "zAxis": { + "description": "The z-axis (normal).", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "description": "A face.", + "type": "object", + "required": [ + "__meta", + "id", + "solid", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Face" + ] + }, + "id": { + "description": "The id of the face.", + "type": "string", + "format": "uuid" + }, + "value": { + "description": "The tag of the face.", + "type": "string" + }, + "xAxis": { + "description": "What should the face’s X axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "yAxis": { + "description": "What should the face’s Y axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "zAxis": { + "description": "The z-axis (normal).", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "solid": { + "description": "The solid the face is on.", + "allOf": [ + { + "$ref": "#/components/schemas/Solid" + } + ] + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, + { + "description": "An solid is a collection of extrude surfaces.", + "type": "object", + "required": [ + "__meta", + "height", + "id", + "sketch", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Solid" + ] + }, + "id": { + "description": "The id of the solid.", + "type": "string", + "format": "uuid" + }, + "value": { + "description": "The extrude surfaces.", + "type": "array", + "items": { + "$ref": "#/components/schemas/ExtrudeSurface" + } + }, + "sketch": { + "description": "The sketch.", + "allOf": [ + { + "$ref": "#/components/schemas/Sketch" + } + ] + }, + "height": { + "description": "The height of the solid.", + "type": "number", + "format": "double" + }, + "startCapId": { + "description": "The id of the extrusion start cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "endCapId": { + "description": "The id of the extrusion end cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "edgeCuts": { + "description": "Chamfers or fillets on this solid.", + "type": "array", + "items": { + "$ref": "#/components/schemas/EdgeCut" + } + }, + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Solids" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Solid" + } + } + } + }, + { + "description": "Data for an imported geometry.", + "type": "object", + "required": [ + "__meta", + "id", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "ImportedGeometry" + ] + }, + "id": { + "description": "The ID of the imported geometry.", + "type": "string", + "format": "uuid" + }, + "value": { + "description": "The original file paths.", + "type": "array", + "items": { + "type": "string" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "expression", + "memory", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Function" + ] + }, + "expression": { + "$ref": "#/components/schemas/FunctionExpression" + }, + "memory": { + "$ref": "#/components/schemas/ProgramMemory" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + } + ] + }, + "Metadata": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } }, "SourceRange": { "type": "array", @@ -143246,6 +145296,10 @@ "nullable": true } } + }, + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" } } }, @@ -143270,10 +145324,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" + ] + }, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" ] }, - "value": {}, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" + ] + }, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -143491,6 +145706,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -143647,6 +145901,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ], "definitions": { @@ -143664,10 +145943,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" + ] + }, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" + ] + }, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" ] }, - "value": {}, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -143885,6 +146325,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -144041,6 +146520,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ] }, @@ -147379,6 +149883,10 @@ "nullable": true } } + }, + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" } } }, @@ -147406,10 +149914,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" ] }, - "value": {}, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" + ] + }, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" + ] + }, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -147627,6 +150296,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -147783,6 +150491,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ] }, @@ -151121,6 +153854,10 @@ "nullable": true } } + }, + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" } } }, @@ -151146,10 +153883,171 @@ "type": { "type": "string", "enum": [ - "UserVal" + "Uuid" ] }, - "value": {}, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" + ] + }, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" + ] + }, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, "__meta": { "type": "array", "items": { @@ -151367,6 +154265,45 @@ } } }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, { "description": "An solid is a collection of extrude surfaces.", "type": "object", @@ -151523,6 +154460,31 @@ } } } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } } ], "definitions": { @@ -151553,6 +154515,625 @@ "maxItems": 3, "minItems": 3 }, + "KclValue": { + "description": "Any KCL value.", + "oneOf": [ + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Uuid" + ] + }, + "value": { + "type": "string", + "format": "uuid" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Bool" + ] + }, + "value": { + "type": "boolean" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Number" + ] + }, + "value": { + "type": "number", + "format": "double" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Int" + ] + }, + "value": { + "type": "integer", + "format": "int64" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "String" + ] + }, + "value": { + "type": "string" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Array" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Object" + ] + }, + "value": { + "type": "object", + "additionalProperties": { + "$ref": "#/components/schemas/KclValue" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TagIdentifier" + ] + }, + "value": { + "type": "string" + }, + "info": { + "allOf": [ + { + "$ref": "#/components/schemas/TagEngineInfo" + } + ], + "nullable": true + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TagDeclarator" + ] + }, + "value": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + { + "description": "A plane.", + "type": "object", + "required": [ + "__meta", + "id", + "origin", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Plane" + ] + }, + "id": { + "description": "The id of the plane.", + "type": "string", + "format": "uuid" + }, + "value": { + "$ref": "#/components/schemas/PlaneType" + }, + "origin": { + "description": "Origin of the plane.", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "xAxis": { + "description": "What should the plane’s X axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "yAxis": { + "description": "What should the plane’s Y axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "zAxis": { + "description": "The z-axis (normal).", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "description": "A face.", + "type": "object", + "required": [ + "__meta", + "id", + "solid", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Face" + ] + }, + "id": { + "description": "The id of the face.", + "type": "string", + "format": "uuid" + }, + "value": { + "description": "The tag of the face.", + "type": "string" + }, + "xAxis": { + "description": "What should the face’s X axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "yAxis": { + "description": "What should the face’s Y axis be?", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "zAxis": { + "description": "The z-axis (normal).", + "allOf": [ + { + "$ref": "#/components/schemas/Point3d" + } + ] + }, + "solid": { + "description": "The solid the face is on.", + "allOf": [ + { + "$ref": "#/components/schemas/Solid" + } + ] + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketch" + ] + }, + "value": { + "$ref": "#/components/schemas/Sketch" + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Sketches" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Sketch" + } + } + } + }, + { + "description": "An solid is a collection of extrude surfaces.", + "type": "object", + "required": [ + "__meta", + "height", + "id", + "sketch", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Solid" + ] + }, + "id": { + "description": "The id of the solid.", + "type": "string", + "format": "uuid" + }, + "value": { + "description": "The extrude surfaces.", + "type": "array", + "items": { + "$ref": "#/components/schemas/ExtrudeSurface" + } + }, + "sketch": { + "description": "The sketch.", + "allOf": [ + { + "$ref": "#/components/schemas/Sketch" + } + ] + }, + "height": { + "description": "The height of the solid.", + "type": "number", + "format": "double" + }, + "startCapId": { + "description": "The id of the extrusion start cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "endCapId": { + "description": "The id of the extrusion end cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "edgeCuts": { + "description": "Chamfers or fillets on this solid.", + "type": "array", + "items": { + "$ref": "#/components/schemas/EdgeCut" + } + }, + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Solids" + ] + }, + "value": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Solid" + } + } + } + }, + { + "description": "Data for an imported geometry.", + "type": "object", + "required": [ + "__meta", + "id", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "ImportedGeometry" + ] + }, + "id": { + "description": "The ID of the imported geometry.", + "type": "string", + "format": "uuid" + }, + "value": { + "description": "The original file paths.", + "type": "array", + "items": { + "type": "string" + } + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "expression", + "memory", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Function" + ] + }, + "expression": { + "$ref": "#/components/schemas/FunctionExpression" + }, + "memory": { + "$ref": "#/components/schemas/ProgramMemory" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": "object", + "required": [ + "__meta", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "KclNone" + ] + }, + "value": { + "$ref": "#/components/schemas/KclNone" + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + } + ] + }, "TagEngineInfo": { "description": "Engine information for a tag.", "type": "object", @@ -154862,399 +158443,9 @@ } } }, - "KclValue": { - "description": "Any KCL value.", - "oneOf": [ - { - "type": "object", - "required": [ - "__meta", - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "UserVal" - ] - }, - "value": {}, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "type": "object", - "required": [ - "__meta", - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "TagIdentifier" - ] - }, - "value": { - "type": "string" - }, - "info": { - "allOf": [ - { - "$ref": "#/components/schemas/TagEngineInfo" - } - ], - "nullable": true - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "type": "object", - "required": [ - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "TagDeclarator" - ] - }, - "value": { - "type": "string" - }, - "digest": { - "type": "array", - "items": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "maxItems": 32, - "minItems": 32, - "nullable": true - }, - "start": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - }, - "end": { - "type": "integer", - "format": "uint", - "minimum": 0.0 - } - } - }, - { - "description": "A plane.", - "type": "object", - "required": [ - "__meta", - "id", - "origin", - "type", - "value", - "xAxis", - "yAxis", - "zAxis" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "Plane" - ] - }, - "id": { - "description": "The id of the plane.", - "type": "string", - "format": "uuid" - }, - "value": { - "$ref": "#/components/schemas/PlaneType" - }, - "origin": { - "description": "Origin of the plane.", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "xAxis": { - "description": "What should the plane’s X axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "yAxis": { - "description": "What should the plane’s Y axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "zAxis": { - "description": "The z-axis (normal).", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "description": "A face.", - "type": "object", - "required": [ - "__meta", - "id", - "solid", - "type", - "value", - "xAxis", - "yAxis", - "zAxis" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "Face" - ] - }, - "id": { - "description": "The id of the face.", - "type": "string", - "format": "uuid" - }, - "value": { - "description": "The tag of the face.", - "type": "string" - }, - "xAxis": { - "description": "What should the face’s X axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "yAxis": { - "description": "What should the face’s Y axis be?", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "zAxis": { - "description": "The z-axis (normal).", - "allOf": [ - { - "$ref": "#/components/schemas/Point3d" - } - ] - }, - "solid": { - "description": "The solid the face is on.", - "allOf": [ - { - "$ref": "#/components/schemas/Solid" - } - ] - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "description": "An solid is a collection of extrude surfaces.", - "type": "object", - "required": [ - "__meta", - "height", - "id", - "sketch", - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "Solid" - ] - }, - "id": { - "description": "The id of the solid.", - "type": "string", - "format": "uuid" - }, - "value": { - "description": "The extrude surfaces.", - "type": "array", - "items": { - "$ref": "#/components/schemas/ExtrudeSurface" - } - }, - "sketch": { - "description": "The sketch.", - "allOf": [ - { - "$ref": "#/components/schemas/Sketch" - } - ] - }, - "height": { - "description": "The height of the solid.", - "type": "number", - "format": "double" - }, - "startCapId": { - "description": "The id of the extrusion start cap", - "type": "string", - "format": "uuid", - "nullable": true - }, - "endCapId": { - "description": "The id of the extrusion end cap", - "type": "string", - "format": "uuid", - "nullable": true - }, - "edgeCuts": { - "description": "Chamfers or fillets on this solid.", - "type": "array", - "items": { - "$ref": "#/components/schemas/EdgeCut" - } - }, - "__meta": { - "description": "Metadata.", - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "type": "object", - "required": [ - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "Solids" - ] - }, - "value": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Solid" - } - } - } - }, - { - "description": "Data for an imported geometry.", - "type": "object", - "required": [ - "__meta", - "id", - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "ImportedGeometry" - ] - }, - "id": { - "description": "The ID of the imported geometry.", - "type": "string", - "format": "uuid" - }, - "value": { - "description": "The original file paths.", - "type": "array", - "items": { - "type": "string" - } - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "type": "object", - "required": [ - "__meta", - "expression", - "memory", - "type" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "Function" - ] - }, - "expression": { - "$ref": "#/components/schemas/FunctionExpression" - }, - "memory": { - "$ref": "#/components/schemas/ProgramMemory" - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - } - ] + "KclNone": { + "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", + "type": "object" } } }, diff --git a/docs/kcl/types/KclNone.md b/docs/kcl/types/KclNone.md new file mode 100644 index 0000000000..96a7c11cae --- /dev/null +++ b/docs/kcl/types/KclNone.md @@ -0,0 +1,16 @@ +--- +title: "KclNone" +excerpt: "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application)." +layout: manual +--- + +KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application). + +**Type:** `object` + + + + + + + diff --git a/docs/kcl/types/KclValue.md b/docs/kcl/types/KclValue.md index 91914ffd40..6cfeac72eb 100644 --- a/docs/kcl/types/KclValue.md +++ b/docs/kcl/types/KclValue.md @@ -23,8 +23,110 @@ Any KCL value. | Property | Type | Description | Required | |----------|------|-------------|----------| -| `type` |enum: `UserVal`| | No | -| `value` |``| | No | +| `type` |enum: `Uuid`| | No | +| `value` |`string`| | No | +| `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| | No | + + +---- + +**Type:** `object` + + + + + +## Properties + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `type` |enum: `Bool`| | No | +| `value` |`boolean`| | No | +| `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| | No | + + +---- + +**Type:** `object` + + + + + +## Properties + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `type` |enum: `Number`| | No | +| `value` |`number`| | No | +| `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| | No | + + +---- + +**Type:** `object` + + + + + +## Properties + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `type` |enum: `Int`| | No | +| `value` |`integer`| | No | +| `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| | No | + + +---- + +**Type:** `object` + + + + + +## Properties + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `type` |enum: `String`| | No | +| `value` |`string`| | No | +| `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| | No | + + +---- + +**Type:** `object` + + + + + +## Properties + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `type` |enum: `Array`| | No | +| `value` |`[` [`KclValue`](/docs/kcl/types/KclValue) `]`| | No | +| `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| | No | + + +---- + +**Type:** `object` + + + + + +## Properties + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `type` |enum: `Object`| | No | +| `value` |`object`| | No | | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| | No | @@ -111,6 +213,38 @@ A face. | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| | No | +---- + +**Type:** `object` + + + + + +## Properties + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `type` |enum: [`Sketch`](/docs/kcl/types/Sketch)| | No | +| `value` |[`Sketch`](/docs/kcl/types/Sketch)| Any KCL value. | No | + + +---- + +**Type:** `object` + + + + + +## Properties + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `type` |enum: `Sketches`| | No | +| `value` |`[` [`Sketch`](/docs/kcl/types/Sketch) `]`| | No | + + ---- An solid is a collection of extrude surfaces. @@ -190,6 +324,23 @@ Data for an imported geometry. ---- +**Type:** `object` + + + + + +## Properties + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `type` |enum: [`KclNone`](/docs/kcl/types/KclNone)| | No | +| `value` |[`KclNone`](/docs/kcl/types/KclNone)| Any KCL value. | No | +| `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`| | No | + + +---- + diff --git a/src/components/AvailableVarsHelpers.tsx b/src/components/AvailableVarsHelpers.tsx index 48a5a50c89..6dbfb492c4 100644 --- a/src/components/AvailableVarsHelpers.tsx +++ b/src/components/AvailableVarsHelpers.tsx @@ -146,7 +146,7 @@ export function useCalc({ const _programMem: ProgramMemory = ProgramMemory.empty() for (const { key, value } of availableVarInfo.variables) { const error = _programMem.set(key, { - type: 'UserVal', + type: 'String', value, __meta: [], }) diff --git a/src/components/ModelingSidebar/ModelingPanes/MemoryPane.tsx b/src/components/ModelingSidebar/ModelingPanes/MemoryPane.tsx index af73f28391..6d737595a7 100644 --- a/src/components/ModelingSidebar/ModelingPanes/MemoryPane.tsx +++ b/src/components/ModelingSidebar/ModelingPanes/MemoryPane.tsx @@ -89,9 +89,9 @@ export const processMemory = (programMemory: ProgramMemory) => { const processedMemory: any = {} for (const [key, val] of programMemory?.visibleEntries()) { if ( - (val.type === 'UserVal' && val.value.type === 'Sketch') || + val.type === 'Sketch' || // @ts-ignore - (val.type !== 'Function' && val.type !== 'UserVal') + val.type !== 'Function' ) { const sg = sketchFromKclValue(val, key) if (val.type === 'Solid') { @@ -110,8 +110,6 @@ export const processMemory = (programMemory: ProgramMemory) => { processedMemory[key] = `__function(${(val as any)?.expression?.params ?.map?.(({ identifier }: any) => identifier?.name || '') .join(', ')})__` - } else { - processedMemory[key] = val.value } } return processedMemory diff --git a/src/lang/artifact.test.ts b/src/lang/artifact.test.ts index 52a8746de8..d961dd6ed8 100644 --- a/src/lang/artifact.test.ts +++ b/src/lang/artifact.test.ts @@ -18,8 +18,7 @@ const mySketch001 = startSketchOn('XY') // @ts-ignore const sketch001 = execState.memory.get('mySketch001') expect(sketch001).toEqual({ - type: 'UserVal', - __meta: [{ sourceRange: [46, 71, 0] }], + type: 'Sketch', value: { type: 'Sketch', on: expect.any(Object), diff --git a/src/lang/executor.test.ts b/src/lang/executor.test.ts index 39773481b7..e10bbbd766 100644 --- a/src/lang/executor.test.ts +++ b/src/lang/executor.test.ts @@ -58,7 +58,13 @@ const newVar = myVar + 1` ` const mem = await exe(code) // geo is three js buffer geometry and is very bloated to have in tests - const minusGeo = mem.get('mySketch')?.value?.paths + const sk = mem.get('mySketch') + expect(sk?.type).toEqual('Sketch') + if (sk?.type !== 'Sketch') { + return + } + + const minusGeo = sk?.value?.paths expect(minusGeo).toEqual([ { type: 'ToPoint', @@ -150,7 +156,7 @@ const newVar = myVar + 1` ].join('\n') const mem = await exe(code) expect(mem.get('mySk1')).toEqual({ - type: 'UserVal', + type: 'Sketch', value: { type: 'Sketch', on: expect.any(Object), @@ -215,7 +221,6 @@ const newVar = myVar + 1` id: expect.any(String), __meta: [{ sourceRange: [39, 63, 0] }], }, - __meta: [{ sourceRange: [39, 63, 0] }], }) }) it('execute array expression', async () => { @@ -225,7 +230,7 @@ const newVar = myVar + 1` const mem = await exe(code) // TODO path to node is probably wrong here, zero indexes are not correct expect(mem.get('three')).toEqual({ - type: 'UserVal', + type: 'Int', value: 3, __meta: [ { @@ -234,8 +239,17 @@ const newVar = myVar + 1` ], }) expect(mem.get('yo')).toEqual({ - type: 'UserVal', - value: [1, '2', 3, 9], + type: 'Array', + value: [ + { type: 'Int', value: 1, __meta: [{ sourceRange: [28, 29, 0] }] }, + { type: 'String', value: '2', __meta: [{ sourceRange: [31, 34, 0] }] }, + { type: 'Int', value: 3, __meta: [{ sourceRange: [14, 15, 0] }] }, + { + type: 'Number', + value: 9, + __meta: [{ sourceRange: [43, 44, 0] }, { sourceRange: [47, 48, 0] }], + }, + ], __meta: [ { sourceRange: [27, 49, 0], @@ -253,8 +267,25 @@ const newVar = myVar + 1` ].join('\n') const mem = await exe(code) expect(mem.get('yo')).toEqual({ - type: 'UserVal', - value: { aStr: 'str', anum: 2, identifier: 3, binExp: 9 }, + type: 'Object', + value: { + aStr: { + type: 'String', + value: 'str', + __meta: [{ sourceRange: [34, 39, 0] }], + }, + anum: { type: 'Int', value: 2, __meta: [{ sourceRange: [47, 48, 0] }] }, + identifier: { + type: 'Int', + value: 3, + __meta: [{ sourceRange: [14, 15, 0] }], + }, + binExp: { + type: 'Number', + value: 9, + __meta: [{ sourceRange: [77, 78, 0] }, { sourceRange: [81, 82, 0] }], + }, + }, __meta: [ { sourceRange: [27, 83, 0], @@ -268,11 +299,11 @@ const newVar = myVar + 1` ) const mem = await exe(code) expect(mem.get('myVar')).toEqual({ - type: 'UserVal', + type: 'String', value: '123', __meta: [ { - sourceRange: [41, 50, 0], + sourceRange: [19, 24, 0], }, ], }) @@ -356,7 +387,26 @@ describe('testing math operators', () => { it('with unaryExpression in ArrayExpression', async () => { const code = 'const myVar = [1,-legLen(5, 4)]' const mem = await exe(code) - expect(mem.get('myVar')?.value).toEqual([1, -3]) + expect(mem.get('myVar')?.value).toEqual([ + { + __meta: [ + { + sourceRange: [15, 16, 0], + }, + ], + type: 'Int', + value: 1, + }, + { + __meta: [ + { + sourceRange: [17, 30, 0], + }, + ], + type: 'Number', + value: -3, + }, + ]) }) it('with unaryExpression in ArrayExpression in CallExpression, checking nothing funny happens when used in a sketch', async () => { const code = [ diff --git a/src/lang/kclSamples.test.ts b/src/lang/kclSamples.test.ts index 1e1a7d0218..0981abbfaa 100644 --- a/src/lang/kclSamples.test.ts +++ b/src/lang/kclSamples.test.ts @@ -55,18 +55,13 @@ describe('Test KCL Samples from public Github repository', () => { }) // Run through all of the files in the manifest json. This will allow us to be automatically updated // with the latest changes in github. We won't be hard coding the filenames - it( - 'should run through all the files', - async () => { - for (let i = 0; i < files.length; i++) { - const file: KclSampleFile = files[i] - const code = await getKclSampleCodeFromGithub(file.filename) - const parsed = parse(code) - assert(!(parsed instanceof Error)) - } - }, - files.length * 1000 - ) + files.forEach((file: KclSampleFile) => { + it(`should parse ${file.filename} without errors`, async () => { + const code = await getKclSampleCodeFromGithub(file.filename) + const parsed = parse(code) + assert(!(parsed instanceof Error)) + }, 1000) + }) }) describe('when performing enginelessExecutor', () => { diff --git a/src/lang/wasm.ts b/src/lang/wasm.ts index 89e4b9ff80..80a49b4be9 100644 --- a/src/lang/wasm.ts +++ b/src/lang/wasm.ts @@ -336,7 +336,7 @@ export class ProgramMemory { */ hasSketchOrSolid(): boolean { for (const node of this.visibleEntries().values()) { - if (node.type === 'Solid' || node.value?.type === 'Sketch') { + if (node.type === 'Solid' || node.type === 'Sketch') { return true } } diff --git a/src/lib/useCalculateKclExpression.ts b/src/lib/useCalculateKclExpression.ts index c581e8c6ff..e244c61f55 100644 --- a/src/lib/useCalculateKclExpression.ts +++ b/src/lib/useCalculateKclExpression.ts @@ -91,7 +91,7 @@ export function useCalculateKclExpression({ const _programMem: ProgramMemory = ProgramMemory.empty() for (const { key, value } of availableVarInfo.variables) { const error = _programMem.set(key, { - type: 'UserVal', + type: 'String', value, __meta: [], }) diff --git a/src/wasm-lib/kcl/src/ast/types.rs b/src/wasm-lib/kcl/src/ast/types.rs index 01a2d99377..59a03554c8 100644 --- a/src/wasm-lib/kcl/src/ast/types.rs +++ b/src/wasm-lib/kcl/src/ast/types.rs @@ -27,7 +27,7 @@ pub use crate::ast::types::{ use crate::{ docs::StdLibFn, errors::KclError, - executor::{ExecState, ExecutorContext, KclValue, Metadata, SourceRange, TagIdentifier, UserVal}, + executor::{ExecState, ExecutorContext, KclValue, Metadata, SourceRange, TagIdentifier}, parser::PIPE_OPERATOR, std::kcl_stdlib::KclStdLibFn, }; @@ -59,6 +59,14 @@ pub struct Node { pub module_id: ModuleId, } +impl Node { + pub fn metadata(&self) -> Metadata { + Metadata { + source_range: SourceRange([self.start, self.end, self.module_id.0 as usize]), + } + } +} + impl schemars::JsonSchema for Node { fn schema_name() -> String { T::schema_name() @@ -1708,34 +1716,26 @@ impl Literal { impl From> for KclValue { fn from(literal: Node) -> Self { - KclValue::UserVal(UserVal { - value: JValue::from(literal.value.clone()), - meta: vec![Metadata { - source_range: literal.into(), - }], - }) + let meta = vec![literal.metadata()]; + match literal.inner.value { + LiteralValue::IInteger(value) => KclValue::Int { value, meta }, + LiteralValue::Fractional(value) => KclValue::Number { value, meta }, + LiteralValue::String(value) => KclValue::String { value, meta }, + LiteralValue::Bool(value) => KclValue::Bool { value, meta }, + } } } impl From<&Node> for KclValue { fn from(literal: &Node) -> Self { - KclValue::UserVal(UserVal { - value: JValue::from(literal.value.clone()), - meta: vec![Metadata { - source_range: literal.into(), - }], - }) + Self::from(literal.to_owned()) } } impl From<&BoxNode> for KclValue { fn from(literal: &BoxNode) -> Self { - KclValue::UserVal(UserVal { - value: JValue::from(literal.value.clone()), - meta: vec![Metadata { - source_range: literal.into(), - }], - }) + let b: &Node = literal; + Self::from(b) } } @@ -3010,17 +3010,6 @@ impl ConstraintLevels { } } -pub(crate) fn human_friendly_type(j: &JValue) -> &'static str { - match j { - JValue::Null => "null", - JValue::Bool(_) => "boolean (true/false value)", - JValue::Number(_) => "number", - JValue::String(_) => "string (text)", - JValue::Array(_) => "array (list)", - JValue::Object(_) => "object", - } -} - #[cfg(test)] mod tests { use pretty_assertions::assert_eq; diff --git a/src/wasm-lib/kcl/src/ast/types/execute.rs b/src/wasm-lib/kcl/src/ast/types/execute.rs index b90ee9da75..a9eb0eef81 100644 --- a/src/wasm-lib/kcl/src/ast/types/execute.rs +++ b/src/wasm-lib/kcl/src/ast/types/execute.rs @@ -1,18 +1,21 @@ +use std::collections::HashMap; + use super::{ - human_friendly_type, ArrayExpression, ArrayRangeExpression, BinaryExpression, BinaryOperator, BinaryPart, - CallExpression, Expr, IfExpression, LiteralIdentifier, LiteralValue, MemberExpression, MemberObject, Node, - ObjectExpression, TagDeclarator, UnaryExpression, UnaryOperator, + ArrayExpression, ArrayRangeExpression, BinaryExpression, BinaryOperator, BinaryPart, CallExpression, Expr, + IfExpression, KclNone, LiteralIdentifier, LiteralValue, MemberExpression, MemberObject, Node, ObjectExpression, + TagDeclarator, UnaryExpression, UnaryOperator, }; use crate::{ errors::{KclError, KclErrorDetails}, executor::{ - BodyType, ExecState, ExecutorContext, KclValue, Metadata, Sketch, SourceRange, StatementKind, TagEngineInfo, - TagIdentifier, UserVal, + BodyType, ExecState, ExecutorContext, KclValue, Metadata, SourceRange, StatementKind, TagEngineInfo, + TagIdentifier, }, std::FunctionKind, }; use async_recursion::async_recursion; -use serde_json::Value as JValue; + +const FLOAT_TO_INT_MAX_DELTA: f64 = 0.01; impl BinaryPart { #[async_recursion] @@ -42,25 +45,18 @@ impl Node { } }; - let array_json = array.get_json_value()?; + let KclValue::Array { value: array, meta: _ } = array else { + return Err(KclError::Semantic(KclErrorDetails { + message: format!("MemberExpression array is not an array: {:?}", array), + source_ranges: vec![self.clone().into()], + })); + }; - if let serde_json::Value::Array(array) = array_json { - if let Some(value) = array.get(index) { - Ok(KclValue::UserVal(UserVal { - value: value.clone(), - meta: vec![Metadata { - source_range: self.into(), - }], - })) - } else { - Err(KclError::UndefinedValue(KclErrorDetails { - message: format!("index {} not found in array", index), - source_ranges: vec![self.clone().into()], - })) - } + if let Some(value) = array.get(index) { + Ok(value.to_owned()) } else { - Err(KclError::Semantic(KclErrorDetails { - message: format!("MemberExpression array is not an array: {:?}", array), + Err(KclError::UndefinedValue(KclErrorDetails { + message: format!("index {} not found in array", index), source_ranges: vec![self.clone().into()], })) } @@ -77,18 +73,11 @@ impl Node { } }; - let object_json = object.get_json_value()?; - // Check the property and object match -- e.g. ints for arrays, strs for objects. - match (object_json, property) { - (JValue::Object(map), Property::String(property)) => { + match (object, property) { + (KclValue::Object { value: map, meta: _ }, Property::String(property)) => { if let Some(value) = map.get(&property) { - Ok(KclValue::UserVal(UserVal { - value: value.clone(), - meta: vec![Metadata { - source_range: self.into(), - }], - })) + Ok(value.to_owned()) } else { Err(KclError::UndefinedValue(KclErrorDetails { message: format!("Property '{property}' not found in object"), @@ -96,22 +85,20 @@ impl Node { })) } } - (JValue::Object(_), p) => Err(KclError::Semantic(KclErrorDetails { - message: format!( - "Only strings can be used as the property of an object, but you're using a {}", - p.type_name() - ), - source_ranges: vec![self.clone().into()], - })), - (JValue::Array(arr), Property::Number(index)) => { - let value_of_arr: Option<&JValue> = arr.get(index); + (KclValue::Object { .. }, p) => { + let t = p.type_name(); + let article = article_for(t); + Err(KclError::Semantic(KclErrorDetails { + message: format!( + "Only strings can be used as the property of an object, but you're using {article} {t}", + ), + source_ranges: vec![self.clone().into()], + })) + } + (KclValue::Array { value: arr, meta: _ }, Property::Number(index)) => { + let value_of_arr = arr.get(index); if let Some(value) = value_of_arr { - Ok(KclValue::UserVal(UserVal { - value: value.clone(), - meta: vec![Metadata { - source_range: self.into(), - }], - })) + Ok(value.to_owned()) } else { Err(KclError::UndefinedValue(KclErrorDetails { message: format!("The array doesn't have any item at index {index}"), @@ -119,17 +106,36 @@ impl Node { })) } } - (JValue::Array(_), p) => Err(KclError::Semantic(KclErrorDetails { - message: format!( - "Only integers >= 0 can be used as the index of an array, but you're using a {}", - p.type_name() - ), - source_ranges: vec![self.clone().into()], - })), + (KclValue::Array { .. }, p) => { + let t = p.type_name(); + let article = article_for(t); + Err(KclError::Semantic(KclErrorDetails { + message: format!( + "Only integers >= 0 can be used as the index of an array, but you're using {article} {t}", + ), + source_ranges: vec![self.clone().into()], + })) + } + (KclValue::Solid(solid), Property::String(prop)) if prop == "sketch" => Ok(KclValue::Sketch { + value: Box::new(solid.sketch), + }), + (KclValue::Sketch { value: sk }, Property::String(prop)) if prop == "tags" => Ok(KclValue::Object { + meta: vec![Metadata { + source_range: SourceRange::from(self.clone()), + }], + value: sk + .tags + .iter() + .map(|(k, tag)| (k.to_owned(), KclValue::TagIdentifier(Box::new(tag.to_owned())))) + .collect(), + }), (being_indexed, _) => { - let t = human_friendly_type(&being_indexed); + let t = being_indexed.human_friendly_type(); + let article = article_for(t); Err(KclError::Semantic(KclErrorDetails { - message: format!("Only arrays and objects can be indexed, but you're trying to index a {t}"), + message: format!( + "Only arrays and objects can be indexed, but you're trying to index {article} {t}" + ), source_ranges: vec![self.clone().into()], })) } @@ -140,81 +146,134 @@ impl Node { impl Node { #[async_recursion] pub async fn get_result(&self, exec_state: &mut ExecState, ctx: &ExecutorContext) -> Result { - let left_json_value = self.left.get_result(exec_state, ctx).await?.get_json_value()?; - let right_json_value = self.right.get_result(exec_state, ctx).await?.get_json_value()?; + let left_value = self.left.get_result(exec_state, ctx).await?; + let right_value = self.right.get_result(exec_state, ctx).await?; + let mut meta = left_value.metadata(); + meta.extend(right_value.metadata()); // First check if we are doing string concatenation. if self.operator == BinaryOperator::Add { - if let (Some(left), Some(right)) = ( - parse_json_value_as_string(&left_json_value), - parse_json_value_as_string(&right_json_value), - ) { - let value = serde_json::Value::String(format!("{}{}", left, right)); - return Ok(KclValue::UserVal(UserVal { - value, - meta: vec![Metadata { - source_range: self.into(), - }], - })); + if let (KclValue::String { value: left, meta: _ }, KclValue::String { value: right, meta: _ }) = + (&left_value, &right_value) + { + return Ok(KclValue::String { + value: format!("{}{}", left, right), + meta, + }); } } - let left = parse_json_number_as_f64(&left_json_value, self.left.clone().into())?; - let right = parse_json_number_as_f64(&right_json_value, self.right.clone().into())?; - - let value: serde_json::Value = match self.operator { - BinaryOperator::Add => (left + right).into(), - BinaryOperator::Sub => (left - right).into(), - BinaryOperator::Mul => (left * right).into(), - BinaryOperator::Div => (left / right).into(), - BinaryOperator::Mod => (left % right).into(), - BinaryOperator::Pow => (left.powf(right)).into(), - BinaryOperator::Eq => (left == right).into(), - BinaryOperator::Neq => (left != right).into(), - BinaryOperator::Gt => (left > right).into(), - BinaryOperator::Gte => (left >= right).into(), - BinaryOperator::Lt => (left < right).into(), - BinaryOperator::Lte => (left <= right).into(), + let left = parse_number_as_f64(&left_value, self.left.clone().into())?; + let right = parse_number_as_f64(&right_value, self.right.clone().into())?; + + let value = match self.operator { + BinaryOperator::Add => KclValue::Number { + value: left + right, + meta, + }, + BinaryOperator::Sub => KclValue::Number { + value: left - right, + meta, + }, + BinaryOperator::Mul => KclValue::Number { + value: left * right, + meta, + }, + BinaryOperator::Div => KclValue::Number { + value: left / right, + meta, + }, + BinaryOperator::Mod => KclValue::Number { + value: left % right, + meta, + }, + BinaryOperator::Pow => KclValue::Number { + value: left.powf(right), + meta, + }, + BinaryOperator::Neq => KclValue::Bool { + value: left != right, + meta, + }, + BinaryOperator::Gt => KclValue::Bool { + value: left > right, + meta, + }, + BinaryOperator::Gte => KclValue::Bool { + value: left >= right, + meta, + }, + BinaryOperator::Lt => KclValue::Bool { + value: left < right, + meta, + }, + BinaryOperator::Lte => KclValue::Bool { + value: left <= right, + meta, + }, + BinaryOperator::Eq => KclValue::Bool { + value: left == right, + meta, + }, }; - Ok(KclValue::UserVal(UserVal { - value, - meta: vec![Metadata { - source_range: self.into(), - }], - })) + Ok(value) } } impl Node { pub async fn get_result(&self, exec_state: &mut ExecState, ctx: &ExecutorContext) -> Result { if self.operator == UnaryOperator::Not { - let value = self.argument.get_result(exec_state, ctx).await?.get_json_value()?; - let Some(bool_value) = json_as_bool(&value) else { + let value = self.argument.get_result(exec_state, ctx).await?; + let KclValue::Bool { + value: bool_value, + meta: _, + } = value + else { return Err(KclError::Semantic(KclErrorDetails { - message: format!("Cannot apply unary operator ! to non-boolean value: {}", value), + message: format!( + "Cannot apply unary operator ! to non-boolean value: {}", + value.human_friendly_type() + ), source_ranges: vec![self.into()], })); }; - let negated = !bool_value; - return Ok(KclValue::UserVal(UserVal { - value: serde_json::Value::Bool(negated), - meta: vec![Metadata { - source_range: self.into(), - }], - })); + let meta = vec![Metadata { + source_range: self.into(), + }]; + let negated = KclValue::Bool { + value: !bool_value, + meta, + }; + + return Ok(negated); } - let num = parse_json_number_as_f64( - &self.argument.get_result(exec_state, ctx).await?.get_json_value()?, - self.into(), - )?; - Ok(KclValue::UserVal(UserVal { - value: (-(num)).into(), - meta: vec![Metadata { - source_range: self.into(), - }], - })) + let value = &self.argument.get_result(exec_state, ctx).await?; + match value { + KclValue::Number { value, meta: _ } => { + let meta = vec![Metadata { + source_range: self.into(), + }]; + Ok(KclValue::Number { value: -value, meta }) + } + KclValue::Int { value, meta: _ } => { + let meta = vec![Metadata { + source_range: self.into(), + }]; + Ok(KclValue::Number { + value: (-value) as f64, + meta, + }) + } + _ => Err(KclError::Semantic(KclErrorDetails { + message: format!( + "You can only negate numbers, but this is a {}", + value.human_friendly_type() + ), + source_ranges: vec![self.into()], + })), + } } } @@ -325,13 +384,10 @@ impl Node { // TODO: This could probably be done in a better way, but as of now this was my only idea // and it works. match result { - KclValue::UserVal(ref mut uval) => { - uval.mutate(|sketch: &mut Sketch| { - for (_, tag) in sketch.tags.iter() { - exec_state.memory.update_tag(&tag.value, tag.clone())?; - } - Ok::<_, KclError>(()) - })?; + KclValue::Sketch { value: ref mut sketch } => { + for (_, tag) in sketch.tags.iter() { + exec_state.memory.update_tag(&tag.value, tag.clone())?; + } } KclValue::Solid(ref mut solid) => { for value in &solid.value { @@ -425,10 +481,10 @@ impl Node { } else { fn_memory.add( ¶m.identifier.name, - KclValue::UserVal(UserVal { - value: serde_json::value::Value::Null, - meta: Default::default(), - }), + KclValue::KclNone { + value: KclNone::new(), + meta: vec![self.into()], + }, param.identifier.clone().into(), )?; } @@ -531,15 +587,13 @@ impl Node { .execute_expr(element, exec_state, &metadata, StatementKind::Expression) .await?; - results.push(value.get_json_value()?); + results.push(value); } - Ok(KclValue::UserVal(UserVal { - value: results.into(), - meta: vec![Metadata { - source_range: self.into(), - }], - })) + Ok(KclValue::Array { + value: results, + meta: vec![self.into()], + }) } } @@ -549,15 +603,19 @@ impl Node { let metadata = Metadata::from(&self.start_element); let start = ctx .execute_expr(&self.start_element, exec_state, &metadata, StatementKind::Expression) - .await? - .get_json_value()?; - let start = parse_json_number_as_i64(&start, (&self.start_element).into())?; + .await?; + let start = start.as_int().ok_or(KclError::Semantic(KclErrorDetails { + source_ranges: vec![self.into()], + message: format!("Expected int but found {}", start.human_friendly_type()), + }))?; let metadata = Metadata::from(&self.end_element); let end = ctx .execute_expr(&self.end_element, exec_state, &metadata, StatementKind::Expression) - .await? - .get_json_value()?; - let end = parse_json_number_as_i64(&end, (&self.end_element).into())?; + .await?; + let end = end.as_int().ok_or(KclError::Semantic(KclErrorDetails { + source_ranges: vec![self.into()], + message: format!("Expected int but found {}", end.human_friendly_type()), + }))?; if end < start { return Err(KclError::Semantic(KclErrorDetails { @@ -567,94 +625,76 @@ impl Node { } let range: Vec<_> = if self.end_inclusive { - (start..=end).map(JValue::from).collect() + (start..=end).collect() } else { - (start..end).map(JValue::from).collect() + (start..end).collect() }; - Ok(KclValue::UserVal(UserVal { - value: range.into(), - meta: vec![Metadata { - source_range: self.into(), - }], - })) + let meta = vec![Metadata { + source_range: self.into(), + }]; + Ok(KclValue::Array { + value: range + .into_iter() + .map(|num| KclValue::Int { + value: num, + meta: meta.clone(), + }) + .collect(), + meta, + }) } } impl Node { #[async_recursion] pub async fn execute(&self, exec_state: &mut ExecState, ctx: &ExecutorContext) -> Result { - let mut object = serde_json::Map::new(); + let mut object = HashMap::with_capacity(self.properties.len()); for property in &self.properties { let metadata = Metadata::from(&property.value); let result = ctx .execute_expr(&property.value, exec_state, &metadata, StatementKind::Expression) .await?; - object.insert(property.key.name.clone(), result.get_json_value()?); + object.insert(property.key.name.clone(), result); } - Ok(KclValue::UserVal(UserVal { - value: object.into(), + Ok(KclValue::Object { + value: object, meta: vec![Metadata { source_range: self.into(), }], - })) + }) } } -fn parse_json_number_as_i64(j: &serde_json::Value, source_range: SourceRange) -> Result { - if let serde_json::Value::Number(n) = &j { - n.as_i64().ok_or_else(|| { - KclError::Syntax(KclErrorDetails { - source_ranges: vec![source_range], - message: format!("Invalid integer: {}", j), - }) - }) +fn article_for(s: &str) -> &'static str { + if s.starts_with(['a', 'e', 'i', 'o', 'u']) { + "an" } else { - Err(KclError::Syntax(KclErrorDetails { - source_ranges: vec![source_range], - message: format!("Invalid integer: {}", j), - })) + "a" } } -pub fn parse_json_number_as_f64(j: &serde_json::Value, source_range: SourceRange) -> Result { - if let serde_json::Value::Number(n) = &j { - n.as_f64().ok_or_else(|| { - KclError::Syntax(KclErrorDetails { - source_ranges: vec![source_range], - message: format!("Invalid number: {}", j), - }) - }) +pub fn parse_number_as_f64(v: &KclValue, source_range: SourceRange) -> Result { + if let KclValue::Number { value: n, .. } = &v { + Ok(*n) + } else if let KclValue::Int { value: n, .. } = &v { + Ok(*n as f64) } else { - Err(KclError::Syntax(KclErrorDetails { + let actual_type = v.human_friendly_type(); + let article = if actual_type.starts_with(['a', 'e', 'i', 'o', 'u']) { + "an" + } else { + "a" + }; + Err(KclError::Semantic(KclErrorDetails { source_ranges: vec![source_range], - message: format!("Invalid number: {}", j), + message: format!("Expected a number, but found {article} {actual_type}",), })) } } -pub fn parse_json_value_as_string(j: &serde_json::Value) -> Option { - if let serde_json::Value::String(n) = &j { - Some(n.clone()) - } else { - None - } -} - -/// JSON value as bool. If it isn't a bool, returns None. -pub fn json_as_bool(j: &serde_json::Value) -> Option { - match j { - JValue::Null => None, - JValue::Bool(b) => Some(*b), - JValue::Number(_) => None, - JValue::String(_) => None, - JValue::Array(_) => None, - JValue::Object(_) => None, - } -} - impl Node { #[async_recursion] pub async fn get_result(&self, exec_state: &mut ExecState, ctx: &ExecutorContext) -> Result { @@ -724,15 +764,7 @@ impl Property { } else { // Actually evaluate memory to compute the property. let prop = exec_state.memory.get(name, property_src)?; - let KclValue::UserVal(prop) = prop else { - return Err(KclError::Semantic(KclErrorDetails { - source_ranges: property_sr, - message: format!( - "{name} is not a valid property/index, you can only use a string or int (>= 0) here", - ), - })); - }; - jvalue_to_prop(&prop.value, property_sr, name) + jvalue_to_prop(prop, property_sr, name) } } LiteralIdentifier::Literal(literal) => { @@ -759,35 +791,37 @@ impl Property { } } -fn jvalue_to_prop(value: &JValue, property_sr: Vec, name: &str) -> Result { +fn jvalue_to_prop(value: &KclValue, property_sr: Vec, name: &str) -> Result { let make_err = |message: String| { Err::(KclError::Semantic(KclErrorDetails { source_ranges: property_sr, message, })) }; - const MUST_BE_POSINT: &str = "indices must be whole positive numbers"; - const TRY_INT: &str = "try using the int() function to make this a whole number"; match value { - JValue::Number(ref num) => { - let maybe_uint = num.as_u64().and_then(|x| usize::try_from(x).ok()); - if let Some(uint) = maybe_uint { + KclValue::Int { value:num, meta: _ } => { + let maybe_int: Result = (*num).try_into(); + if let Ok(uint) = maybe_int { Ok(Property::Number(uint)) - } else if let Some(iint) = num.as_i64() { - make_err(format!("'{iint}' is not a valid index, {MUST_BE_POSINT}")) - } else if let Some(fnum) = num.as_f64() { - if fnum < 0.0 { - make_err(format!("'{fnum}' is not a valid index, {MUST_BE_POSINT}")) - } else if fnum.fract() == 0.0 { - make_err(format!("'{fnum:.1}' is stored as a fractional number but indices must be whole numbers, {TRY_INT}")) - } else { - make_err(format!("'{fnum}' is not a valid index, {MUST_BE_POSINT}, {TRY_INT}")) - } + } + else { + make_err(format!("'{num}' is negative, so you can't index an array with it")) + } + } + KclValue::Number{value: num, meta:_} => { + let num = *num; + if num < 0.0 { + return make_err(format!("'{num}' is negative, so you can't index an array with it")) + } + let nearest_int = num.round(); + let delta = num-nearest_int; + if delta < FLOAT_TO_INT_MAX_DELTA { + Ok(Property::Number(nearest_int as usize)) } else { - make_err(format!("'{num}' is not a valid index, {MUST_BE_POSINT}")) + make_err(format!("'{num}' is not an integer, so you can't index an array with it")) } } - JValue::String(ref x) => Ok(Property::String(x.to_owned())), + KclValue::String{value: x, meta:_} => Ok(Property::String(x.to_owned())), _ => { make_err(format!("{name} is not a valid property/index, you can only use a string to get the property of an object, or an int (>= 0) to get an item in an array")) } diff --git a/src/wasm-lib/kcl/src/ast/types/none.rs b/src/wasm-lib/kcl/src/ast/types/none.rs index 6e60243311..618ee7be5c 100644 --- a/src/wasm-lib/kcl/src/ast/types/none.rs +++ b/src/wasm-lib/kcl/src/ast/types/none.rs @@ -4,10 +4,7 @@ use databake::*; use schemars::JsonSchema; use serde::{Deserialize, Serialize}; -use crate::{ - ast::types::ConstraintLevel, - executor::{KclValue, UserVal}, -}; +use crate::{ast::types::ConstraintLevel, executor::KclValue}; use super::Node; @@ -16,7 +13,7 @@ const KCL_NONE_ID: &str = "KCL_NONE_ID"; /// KCL value for an optional parameter which was not given an argument. /// (remember, parameters are in the function declaration, /// arguments are in the function call/application). -#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema, Bake, Default)] +#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema, Bake, Default, Copy)] #[databake(path = kcl_lib::ast::types)] #[ts(export)] #[serde(tag = "type")] @@ -58,22 +55,15 @@ where } } -impl From<&KclNone> for UserVal { +impl From<&KclNone> for KclValue { fn from(none: &KclNone) -> Self { - UserVal { - value: serde_json::to_value(none).expect("can always serialize a None"), + KclValue::KclNone { + value: *none, meta: Default::default(), } } } -impl From<&KclNone> for KclValue { - fn from(none: &KclNone) -> Self { - let val = UserVal::from(none); - KclValue::UserVal(val) - } -} - impl From<&Node> for KclValue { fn from(none: &Node) -> Self { Self::from(&none.inner) diff --git a/src/wasm-lib/kcl/src/executor.rs b/src/wasm-lib/kcl/src/executor.rs index ecbd4f4b97..cfc7a46f32 100644 --- a/src/wasm-lib/kcl/src/executor.rs +++ b/src/wasm-lib/kcl/src/executor.rs @@ -19,7 +19,6 @@ use kittycad_modeling_cmds::length_unit::LengthUnit; use parse_display::{Display, FromStr}; use schemars::JsonSchema; use serde::{Deserialize, Serialize}; -use serde_json::Value as JValue; use tower_lsp::lsp_types::{Position as LspPosition, Range as LspRange}; type Point2D = kcmc::shared::Point2d; @@ -27,8 +26,8 @@ type Point3D = kcmc::shared::Point3d; use crate::{ ast::types::{ - human_friendly_type, BodyItem, Expr, FunctionExpression, ItemVisibility, KclNone, ModuleId, Node, NodeRef, - Program, TagDeclarator, TagNode, + BodyItem, Expr, FunctionExpression, ItemVisibility, KclNone, ModuleId, Node, NodeRef, Program, TagDeclarator, + TagNode, }, engine::{EngineManager, ExecutionKind}, errors::{KclError, KclErrorDetails}, @@ -201,33 +200,18 @@ impl Environment { Self { // Prelude bindings: HashMap::from([ - ( - "ZERO".to_string(), - KclValue::UserVal(UserVal { - value: serde_json::Value::Number(serde_json::value::Number::from(0)), - meta: Default::default(), - }), - ), + ("ZERO".to_string(), KclValue::from_number(0.0, Default::default())), ( "QUARTER_TURN".to_string(), - KclValue::UserVal(UserVal { - value: serde_json::Value::Number(serde_json::value::Number::from(90)), - meta: Default::default(), - }), + KclValue::from_number(90.0, Default::default()), ), ( "HALF_TURN".to_string(), - KclValue::UserVal(UserVal { - value: serde_json::Value::Number(serde_json::value::Number::from(180)), - meta: Default::default(), - }), + KclValue::from_number(180.0, Default::default()), ), ( "THREE_QUARTER_TURN".to_string(), - KclValue::UserVal(UserVal { - value: serde_json::Value::Number(serde_json::value::Number::from(270)), - meta: Default::default(), - }), + KclValue::from_number(270.0, Default::default()), ), ]), parent: None, @@ -264,22 +248,15 @@ impl Environment { } for (_, val) in self.bindings.iter_mut() { - let KclValue::UserVal(v) = val else { continue }; - let meta = v.meta.clone(); - let maybe_sg: Result = serde_json::from_value(v.value.clone()); - let Ok(mut sketch) = maybe_sg else { - continue; - }; + let KclValue::Sketch { value } = val else { continue }; + let mut sketch = value.to_owned(); if sketch.original_id == sg.original_id { for tag in sg.tags.iter() { sketch.tags.insert(tag.0.clone(), tag.1.clone()); } } - *val = KclValue::UserVal(UserVal { - meta, - value: serde_json::to_value(sketch).expect("can always turn Sketch into JSON"), - }); + *val = KclValue::Sketch { value: sketch }; } } } @@ -360,12 +337,52 @@ impl IdGenerator { #[ts(export)] #[serde(tag = "type")] pub enum KclValue { - UserVal(UserVal), + Uuid { + value: ::uuid::Uuid, + #[serde(rename = "__meta")] + meta: Vec, + }, + Bool { + value: bool, + #[serde(rename = "__meta")] + meta: Vec, + }, + Number { + value: f64, + #[serde(rename = "__meta")] + meta: Vec, + }, + Int { + value: i64, + #[serde(rename = "__meta")] + meta: Vec, + }, + String { + value: String, + #[serde(rename = "__meta")] + meta: Vec, + }, + Array { + value: Vec, + #[serde(rename = "__meta")] + meta: Vec, + }, + Object { + value: HashMap, + #[serde(rename = "__meta")] + meta: Vec, + }, TagIdentifier(Box), TagDeclarator(crate::ast::types::BoxNode), Plane(Box), Face(Box), + Sketch { + value: Box, + }, + Sketches { + value: Vec>, + }, Solid(Box), Solids { value: Vec>, @@ -380,31 +397,55 @@ pub enum KclValue { #[serde(rename = "__meta")] meta: Vec, }, + KclNone { + value: KclNone, + #[serde(rename = "__meta")] + meta: Vec, + }, } impl KclValue { - pub(crate) fn new_user_val(meta: Vec, val: T) -> Self { - Self::UserVal(UserVal::new(meta, val)) + pub(crate) fn metadata(&self) -> Vec { + match self { + KclValue::Uuid { value: _, meta } => meta.clone(), + KclValue::Bool { value: _, meta } => meta.clone(), + KclValue::Number { value: _, meta } => meta.clone(), + KclValue::Int { value: _, meta } => meta.clone(), + KclValue::String { value: _, meta } => meta.clone(), + KclValue::Array { value: _, meta } => meta.clone(), + KclValue::Object { value: _, meta } => meta.clone(), + KclValue::TagIdentifier(x) => x.meta.clone(), + KclValue::TagDeclarator(x) => vec![x.metadata()], + KclValue::Plane(x) => x.meta.clone(), + KclValue::Face(x) => x.meta.clone(), + KclValue::Sketch { value } => value.meta.clone(), + KclValue::Sketches { value } => value.iter().flat_map(|sketch| &sketch.meta).copied().collect(), + KclValue::Solid(x) => x.meta.clone(), + KclValue::Solids { value } => value.iter().flat_map(|sketch| &sketch.meta).copied().collect(), + KclValue::ImportedGeometry(x) => x.meta.clone(), + KclValue::Function { meta, .. } => meta.clone(), + KclValue::KclNone { meta, .. } => meta.clone(), + } } pub(crate) fn get_solid_set(&self) -> Result { match self { KclValue::Solid(e) => Ok(SolidSet::Solid(e.clone())), KclValue::Solids { value } => Ok(SolidSet::Solids(value.clone())), - KclValue::UserVal(value) => { - let value = value.value.clone(); - match value { - JValue::Null | JValue::Bool(_) | JValue::Number(_) | JValue::String(_) => Err(anyhow::anyhow!( - "Failed to deserialize solid set from JSON {}", - human_friendly_type(&value) - )), - JValue::Array(_) => serde_json::from_value::>>(value) - .map(SolidSet::from) - .map_err(|e| anyhow::anyhow!("Failed to deserialize array of solids from JSON: {}", e)), - JValue::Object(_) => serde_json::from_value::>(value) - .map(SolidSet::from) - .map_err(|e| anyhow::anyhow!("Failed to deserialize solid from JSON: {}", e)), - } + KclValue::Array { value, .. } => { + let solids: Vec<_> = value + .iter() + .enumerate() + .map(|(i, v)| { + v.as_solid().map(|v| v.to_owned()).map(Box::new).ok_or_else(|| { + anyhow::anyhow!( + "expected this array to only contain solids, but element {i} was actually {}", + v.human_friendly_type() + ) + }) + }) + .collect::>()?; + Ok(SolidSet::Solids(solids)) } _ => anyhow::bail!("Not a solid or solids: {:?}", self), } @@ -414,43 +455,44 @@ impl KclValue { /// on for program logic. pub(crate) fn human_friendly_type(&self) -> &'static str { match self { - KclValue::UserVal(u) => human_friendly_type(&u.value), + KclValue::Uuid { .. } => "Unique ID (uuid)", KclValue::TagDeclarator(_) => "TagDeclarator", KclValue::TagIdentifier(_) => "TagIdentifier", KclValue::Solid(_) => "Solid", KclValue::Solids { .. } => "Solids", + KclValue::Sketch { .. } => "Sketch", + KclValue::Sketches { .. } => "Sketches", KclValue::ImportedGeometry(_) => "ImportedGeometry", KclValue::Function { .. } => "Function", KclValue::Plane(_) => "Plane", KclValue::Face(_) => "Face", + KclValue::Bool { .. } => "boolean (true/false value)", + KclValue::Number { .. } => "number", + KclValue::Int { .. } => "integer", + KclValue::String { .. } => "string (text)", + KclValue::Array { .. } => "array (list)", + KclValue::Object { .. } => "object", + KclValue::KclNone { .. } => "None", } } pub(crate) fn is_function(&self) -> bool { - match self { - KclValue::UserVal(..) - | KclValue::TagIdentifier(..) - | KclValue::TagDeclarator(..) - | KclValue::Plane(..) - | KclValue::Face(..) - | KclValue::Solid(..) - | KclValue::Solids { .. } - | KclValue::ImportedGeometry(..) => false, - KclValue::Function { .. } => true, - } + matches!(self, KclValue::Function { .. }) } } impl From for KclValue { fn from(sg: SketchSet) -> Self { - KclValue::UserVal(UserVal::new(sg.meta(), sg)) + match sg { + SketchSet::Sketch(value) => KclValue::Sketch { value }, + SketchSet::Sketches(value) => KclValue::Sketches { value }, + } } } impl From>> for KclValue { fn from(sg: Vec>) -> Self { - let meta = sg.iter().flat_map(|sg| sg.meta.clone()).collect(); - KclValue::UserVal(UserVal::new(meta, sg)) + KclValue::Sketches { value: sg } } } @@ -815,52 +857,6 @@ pub enum PlaneType { Custom, } -#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema)] -#[ts(export)] -#[serde(tag = "type", rename_all = "camelCase")] -pub struct UserVal { - #[ts(type = "any")] - pub value: serde_json::Value, - #[serde(rename = "__meta")] - pub meta: Vec, -} - -impl UserVal { - pub fn new(meta: Vec, val: T) -> Self { - Self { - meta, - value: serde_json::to_value(val).expect("all KCL values should be compatible with JSON"), - } - } - - /// If the UserVal matches the type `T`, return it. - pub fn get(&self) -> Option<(T, Vec)> { - let meta = self.meta.clone(); - // TODO: This clone might cause performance problems, it'll happen a lot. - let res: Result = serde_json::from_value(self.value.clone()); - if let Ok(t) = res { - Some((t, meta)) - } else { - None - } - } - - /// If the UserVal matches the type `T`, then mutate it via the given closure. - /// If the closure returns Err, the mutation won't be applied. - pub fn mutate(&mut self, mutate: F) -> Result<(), E> - where - T: serde::de::DeserializeOwned + Serialize, - F: FnOnce(&mut T) -> Result<(), E>, - { - let Some((mut val, meta)) = self.get::() else { - return Ok(()); - }; - mutate(&mut val)?; - *self = Self::new(meta, val); - Ok(()) - } -} - #[derive(Debug, Clone, Serialize, Deserialize, PartialEq, ts_rs::TS, JsonSchema)] #[ts(export)] #[serde(tag = "type", rename_all = "camelCase")] @@ -922,108 +918,177 @@ pub type MemoryFunction = impl From for Vec { fn from(item: KclValue) -> Self { match item { - KclValue::UserVal(u) => u.meta.iter().map(|m| m.source_range).collect(), - KclValue::TagDeclarator(t) => vec![(&t).into()], - KclValue::TagIdentifier(t) => t.meta.iter().map(|m| m.source_range).collect(), - KclValue::Solid(e) => e.meta.iter().map(|m| m.source_range).collect(), - KclValue::Solids { value } => value - .iter() - .flat_map(|eg| eg.meta.iter().map(|m| m.source_range)) - .collect(), - KclValue::ImportedGeometry(i) => i.meta.iter().map(|m| m.source_range).collect(), - KclValue::Function { meta, .. } => meta.iter().map(|m| m.source_range).collect(), - KclValue::Plane(p) => p.meta.iter().map(|m| m.source_range).collect(), - KclValue::Face(f) => f.meta.iter().map(|m| m.source_range).collect(), + KclValue::TagDeclarator(t) => vec![SourceRange([t.start, t.end, t.module_id.0 as usize])], + KclValue::TagIdentifier(t) => to_vec_sr(&t.meta), + KclValue::Solid(e) => to_vec_sr(&e.meta), + KclValue::Solids { value } => value.iter().flat_map(|eg| to_vec_sr(&eg.meta)).collect(), + KclValue::Sketch { value } => to_vec_sr(&value.meta), + KclValue::Sketches { value } => value.iter().flat_map(|eg| to_vec_sr(&eg.meta)).collect(), + KclValue::ImportedGeometry(i) => to_vec_sr(&i.meta), + KclValue::Function { meta, .. } => to_vec_sr(&meta), + KclValue::Plane(p) => to_vec_sr(&p.meta), + KclValue::Face(f) => to_vec_sr(&f.meta), + KclValue::Bool { meta, .. } => to_vec_sr(&meta), + KclValue::Number { meta, .. } => to_vec_sr(&meta), + KclValue::Int { meta, .. } => to_vec_sr(&meta), + KclValue::String { meta, .. } => to_vec_sr(&meta), + KclValue::Array { meta, .. } => to_vec_sr(&meta), + KclValue::Object { meta, .. } => to_vec_sr(&meta), + KclValue::Uuid { meta, .. } => to_vec_sr(&meta), + KclValue::KclNone { meta, .. } => to_vec_sr(&meta), } } } +fn to_vec_sr(meta: &[Metadata]) -> Vec { + meta.iter().map(|m| m.source_range).collect() +} + impl From<&KclValue> for Vec { fn from(item: &KclValue) -> Self { match item { - KclValue::UserVal(u) => u.meta.iter().map(|m| m.source_range).collect(), - KclValue::TagDeclarator(ref t) => vec![t.into()], - KclValue::TagIdentifier(t) => t.meta.iter().map(|m| m.source_range).collect(), - KclValue::Solid(e) => e.meta.iter().map(|m| m.source_range).collect(), - KclValue::Solids { value } => value - .iter() - .flat_map(|eg| eg.meta.iter().map(|m| m.source_range)) - .collect(), - KclValue::ImportedGeometry(i) => i.meta.iter().map(|m| m.source_range).collect(), - KclValue::Function { meta, .. } => meta.iter().map(|m| m.source_range).collect(), - KclValue::Plane(p) => p.meta.iter().map(|m| m.source_range).collect(), - KclValue::Face(f) => f.meta.iter().map(|m| m.source_range).collect(), + KclValue::TagDeclarator(t) => vec![SourceRange([t.start, t.end, t.module_id.0 as usize])], + KclValue::TagIdentifier(t) => to_vec_sr(&t.meta), + KclValue::Solid(e) => to_vec_sr(&e.meta), + KclValue::Solids { value } => value.iter().flat_map(|eg| to_vec_sr(&eg.meta)).collect(), + KclValue::Sketch { value } => to_vec_sr(&value.meta), + KclValue::Sketches { value } => value.iter().flat_map(|eg| to_vec_sr(&eg.meta)).collect(), + KclValue::ImportedGeometry(i) => to_vec_sr(&i.meta), + KclValue::Function { meta, .. } => to_vec_sr(meta), + KclValue::Plane(p) => to_vec_sr(&p.meta), + KclValue::Face(f) => to_vec_sr(&f.meta), + KclValue::Bool { meta, .. } => to_vec_sr(meta), + KclValue::Number { meta, .. } => to_vec_sr(meta), + KclValue::Int { meta, .. } => to_vec_sr(meta), + KclValue::String { meta, .. } => to_vec_sr(meta), + KclValue::Uuid { meta, .. } => to_vec_sr(meta), + KclValue::Array { meta, .. } => to_vec_sr(meta), + KclValue::Object { meta, .. } => to_vec_sr(meta), + KclValue::KclNone { meta, .. } => to_vec_sr(meta), } } } impl KclValue { - pub fn get_json_value(&self) -> Result { - if let KclValue::UserVal(user_val) = self { - Ok(user_val.value.clone()) + /// Put the number into a KCL value. + pub fn from_number(f: f64, meta: Vec) -> Self { + Self::Number { value: f, meta } + } + + /// Put the point into a KCL value. + pub fn from_point2d(p: [f64; 2], meta: Vec) -> Self { + Self::Array { + value: vec![ + Self::Number { + value: p[0], + meta: meta.clone(), + }, + Self::Number { + value: p[1], + meta: meta.clone(), + }, + ], + meta, + } + } + + pub(crate) fn as_usize(&self) -> Option { + match self { + KclValue::Int { value, .. } => Some(*value as usize), + _ => None, + } + } + + pub fn as_int(&self) -> Option { + if let KclValue::Int { value, meta: _ } = &self { + Some(*value) } else { - serde_json::to_value(self).map_err(|err| { - KclError::Semantic(KclErrorDetails { - message: format!("Cannot convert memory item to json value: {:?}", err), - source_ranges: self.clone().into(), - }) - }) + None } } - /// Get a JSON value and deserialize it into some concrete type. - pub fn get_json(&self) -> Result { - let json = self.get_json_value()?; + pub fn as_object(&self) -> Option<&HashMap> { + if let KclValue::Object { value, meta: _ } = &self { + Some(value) + } else { + None + } + } - serde_json::from_value(json).map_err(|e| { - KclError::Type(KclErrorDetails { - message: format!("Failed to deserialize struct from JSON: {}", e), - source_ranges: self.clone().into(), - }) - }) + pub fn as_str(&self) -> Option<&str> { + if let KclValue::String { value, meta: _ } = &self { + Some(value) + } else { + None + } } - /// Get a JSON value and deserialize it into some concrete type. - /// If it's a KCL None, return None. Otherwise return Some. - pub fn get_json_opt(&self) -> Result, KclError> { - let json = self.get_json_value()?; - if let JValue::Object(ref o) = json { - if let Some(JValue::String(s)) = o.get("type") { - if s == "KclNone" { - return Ok(None); - } - } + pub fn as_array(&self) -> Option<&[KclValue]> { + if let KclValue::Array { value, meta: _ } = &self { + Some(value) + } else { + None } + } - serde_json::from_value(json) - .map_err(|e| { - KclError::Type(KclErrorDetails { - message: format!("Failed to deserialize struct from JSON: {}", e), - source_ranges: self.clone().into(), - }) - }) - .map(Some) + pub fn as_point2d(&self) -> Option<[f64; 2]> { + let arr = self.as_array()?; + if arr.len() != 2 { + return None; + } + let x = arr[0].as_f64()?; + let y = arr[1].as_f64()?; + Some([x, y]) } - pub fn as_user_val(&self) -> Option<&UserVal> { - if let KclValue::UserVal(x) = self { - Some(x) + pub fn as_uuid(&self) -> Option { + if let KclValue::Uuid { value, meta: _ } = &self { + Some(*value) } else { None } } - /// If this value is of type u32, return it. + pub fn as_solid(&self) -> Option<&Solid> { + if let KclValue::Solid(value) = &self { + Some(value) + } else { + None + } + } + + pub fn as_f64(&self) -> Option { + if let KclValue::Number { value, meta: _ } = &self { + Some(*value) + } else if let KclValue::Int { value, meta: _ } = &self { + Some(*value as f64) + } else { + None + } + } + + pub fn as_bool(&self) -> Option { + if let KclValue::Bool { value, meta: _ } = &self { + Some(*value) + } else { + None + } + } + + /// If this value fits in a u32, return it. pub fn get_u32(&self, source_ranges: Vec) -> Result { - let err = KclError::Semantic(KclErrorDetails { - message: "Expected an integer >= 0".to_owned(), - source_ranges, - }); - self.as_user_val() - .and_then(|uv| uv.value.as_number()) - .and_then(|n| n.as_u64()) - .and_then(|n| u32::try_from(n).ok()) - .ok_or(err) + let u = self.as_int().and_then(|n| u64::try_from(n).ok()).ok_or_else(|| { + KclError::Semantic(KclErrorDetails { + message: "Expected an integer >= 0".to_owned(), + source_ranges: source_ranges.clone(), + }) + })?; + u32::try_from(u).map_err(|_| { + KclError::Semantic(KclErrorDetails { + message: "Number was too big".to_owned(), + source_ranges, + }) + }) } /// If this value is of type function, return it. @@ -1048,16 +1113,6 @@ impl KclValue { pub fn get_tag_identifier(&self) -> Result { match self { KclValue::TagIdentifier(t) => Ok(*t.clone()), - KclValue::UserVal(_) => { - if let Some(identifier) = self.get_json_opt::()? { - Ok(identifier) - } else { - Err(KclError::Semantic(KclErrorDetails { - message: format!("Not a tag identifier: {:?}", self), - source_ranges: self.clone().into(), - })) - } - } _ => Err(KclError::Semantic(KclErrorDetails { message: format!("Not a tag identifier: {:?}", self), source_ranges: self.clone().into(), @@ -1089,19 +1144,13 @@ impl KclValue { /// If this KCL value is a bool, retrieve it. pub fn get_bool(&self) -> Result { - let Self::UserVal(uv) = self else { + let Self::Bool { value: b, .. } = self else { return Err(KclError::Type(KclErrorDetails { source_ranges: self.into(), message: format!("Expected bool, found {}", self.human_friendly_type()), })); }; - let JValue::Bool(b) = uv.value else { - return Err(KclError::Type(KclErrorDetails { - source_ranges: self.into(), - message: format!("Expected bool, found {}", human_friendly_type(&uv.value)), - })); - }; - Ok(b) + Ok(*b) } /// If this memory item is a function, call it with the given arguments, return its val as Ok. @@ -1555,7 +1604,7 @@ impl From for kittycad_modeling_cmds::shared::Point3d { } /// Metadata. -#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema, Eq)] +#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema, Eq, Copy)] #[ts(export)] #[serde(rename_all = "camelCase")] pub struct Metadata { @@ -1563,6 +1612,12 @@ pub struct Metadata { pub source_range: SourceRange, } +impl From for Vec { + fn from(meta: Metadata) -> Self { + vec![meta.source_range] + } +} + impl From for Metadata { fn from(source_range: SourceRange) -> Self { Self { source_range } @@ -2655,74 +2710,8 @@ mod tests { } /// Convenience function to get a JSON value from memory and unwrap. - fn mem_get_json(memory: &ProgramMemory, name: &str) -> serde_json::Value { - memory - .get(name, SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - } - - #[tokio::test(flavor = "multi_thread")] - async fn test_execute_assign_two_variables() { - let ast = r#"const myVar = 5 -const newVar = myVar + 1"#; - let memory = parse_execute(ast).await.unwrap(); - assert_eq!( - serde_json::json!(5), - memory - .get("myVar", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - assert_eq!( - serde_json::json!(6.0), - memory - .get("newVar", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - } - - #[tokio::test(flavor = "multi_thread")] - async fn test_execute_angled_line_that_intersects() { - let ast_fn = |offset: &str| -> String { - format!( - r#"const part001 = startSketchOn('XY') - |> startProfileAt([0, 0], %) - |> lineTo([2, 2], %, $yo) - |> lineTo([3, 1], %) - |> angledLineThatIntersects({{ - angle: 180, - intersectTag: yo, - offset: {}, -}}, %, $yo2) -const intersect = segEndX(yo2)"#, - offset - ) - }; - - let memory = parse_execute(&ast_fn("-1")).await.unwrap(); - assert_eq!( - serde_json::json!(1.0 + 2.0f64.sqrt()), - memory - .get("intersect", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - - let memory = parse_execute(&ast_fn("0")).await.unwrap(); - assert_eq!( - serde_json::json!(1.0000000000000002), - memory - .get("intersect", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); + fn mem_get_json(memory: &ProgramMemory, name: &str) -> KclValue { + memory.get(name, SourceRange::default()).unwrap().to_owned() } #[tokio::test(flavor = "multi_thread")] @@ -3120,200 +3109,41 @@ let shape = layer() |> patternTransform(10, transform, %) ); } - #[tokio::test(flavor = "multi_thread")] - async fn test_execute_function_with_parameter_redefined_outside() { - let ast = r#" -fn myIdentity = (x) => { - return x -} - -const x = 33 - -const two = myIdentity(2)"#; - - let memory = parse_execute(ast).await.unwrap(); - assert_eq!( - serde_json::json!(2), - memory - .get("two", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - assert_eq!( - serde_json::json!(33), - memory - .get("x", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - } - - #[tokio::test(flavor = "multi_thread")] - async fn test_execute_function_referencing_variable_in_parent_scope() { - let ast = r#" -const x = 22 -const y = 3 - -fn add = (x) => { - return x + y -} - -const answer = add(2)"#; - - let memory = parse_execute(ast).await.unwrap(); - assert_eq!( - serde_json::json!(5.0), - memory - .get("answer", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - assert_eq!( - serde_json::json!(22), - memory - .get("x", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - } - - #[tokio::test(flavor = "multi_thread")] - async fn test_execute_function_redefining_variable_in_parent_scope() { - let ast = r#" -const x = 1 - -fn foo = () => { - const x = 2 - return x -} - -const answer = foo()"#; - - let memory = parse_execute(ast).await.unwrap(); - assert_eq!( - serde_json::json!(2), - memory - .get("answer", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - assert_eq!( - serde_json::json!(1), - memory - .get("x", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - } - - #[tokio::test(flavor = "multi_thread")] - async fn test_execute_pattern_transform_function_redefining_variable_in_parent_scope() { - let ast = r#" -const scale = 100 -fn transform = (replicaId) => { - // Redefine same variable as in parent scope. - const scale = 2 - return { - translate: [0, 0, replicaId * 10], - scale: [scale, 1, 0], - } -} - -fn layer = () => { - return startSketchOn("XY") - |> circle({ center: [0, 0], radius: 1 }, %, $tag1) - |> extrude(10, %) -} - -// The 10 layers are replicas of each other, with a transform applied to each. -let shape = layer() |> patternTransform(10, transform, %)"#; - - let memory = parse_execute(ast).await.unwrap(); - // TODO: Assert that scale 2 was used. - assert_eq!( - serde_json::json!(100), - memory - .get("scale", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); - } + // ADAM: Move some of these into simulation tests. #[tokio::test(flavor = "multi_thread")] async fn test_math_execute_with_functions() { let ast = r#"const myVar = 2 + min(100, -1 + legLen(5, 3))"#; let memory = parse_execute(ast).await.unwrap(); - assert_eq!( - serde_json::json!(5.0), - memory - .get("myVar", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); + assert_eq!(5.0, mem_get_json(&memory, "myVar").as_f64().unwrap()); } #[tokio::test(flavor = "multi_thread")] async fn test_math_execute() { let ast = r#"const myVar = 1 + 2 * (3 - 4) / -5 + 6"#; let memory = parse_execute(ast).await.unwrap(); - assert_eq!( - serde_json::json!(7.4), - memory - .get("myVar", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); + assert_eq!(7.4, mem_get_json(&memory, "myVar").as_f64().unwrap()); } #[tokio::test(flavor = "multi_thread")] async fn test_math_execute_start_negative() { let ast = r#"const myVar = -5 + 6"#; let memory = parse_execute(ast).await.unwrap(); - assert_eq!( - serde_json::json!(1.0), - memory - .get("myVar", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); + assert_eq!(1.0, mem_get_json(&memory, "myVar").as_f64().unwrap()); } #[tokio::test(flavor = "multi_thread")] async fn test_math_execute_with_pi() { let ast = r#"const myVar = pi() * 2"#; let memory = parse_execute(ast).await.unwrap(); - assert_eq!( - serde_json::json!(std::f64::consts::TAU), - memory - .get("myVar", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); + assert_eq!(std::f64::consts::TAU, mem_get_json(&memory, "myVar").as_f64().unwrap()); } #[tokio::test(flavor = "multi_thread")] async fn test_math_define_decimal_without_leading_zero() { let ast = r#"let thing = .4 + 7"#; let memory = parse_execute(ast).await.unwrap(); - assert_eq!( - serde_json::json!(7.4), - memory - .get("thing", SourceRange::default()) - .unwrap() - .get_json_value() - .unwrap() - ); + assert_eq!(7.4, mem_get_json(&memory, "thing").as_f64().unwrap()); } #[tokio::test(flavor = "multi_thread")] @@ -3353,10 +3183,10 @@ fn check = (x) => { check(false) "#; let mem = parse_execute(ast).await.unwrap(); - assert_eq!(serde_json::json!(false), mem_get_json(&mem, "notTrue")); - assert_eq!(serde_json::json!(true), mem_get_json(&mem, "notFalse")); - assert_eq!(serde_json::json!(true), mem_get_json(&mem, "c")); - assert_eq!(serde_json::json!(false), mem_get_json(&mem, "d")); + assert_eq!(false, mem_get_json(&mem, "notTrue").as_bool().unwrap()); + assert_eq!(true, mem_get_json(&mem, "notFalse").as_bool().unwrap()); + assert_eq!(true, mem_get_json(&mem, "c").as_bool().unwrap()); + assert_eq!(false, mem_get_json(&mem, "d").as_bool().unwrap()); } #[tokio::test(flavor = "multi_thread")] @@ -3369,7 +3199,7 @@ let notNull = !myNull assert_eq!( parse_execute(code1).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { - message: "Cannot apply unary operator ! to non-boolean value: null".to_owned(), + message: "Cannot apply unary operator ! to non-boolean value: number".to_owned(), source_ranges: vec![SourceRange([56, 63, 0])], }) ); @@ -3378,7 +3208,7 @@ let notNull = !myNull assert_eq!( parse_execute(code2).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { - message: "Cannot apply unary operator ! to non-boolean value: 0".to_owned(), + message: "Cannot apply unary operator ! to non-boolean value: integer".to_owned(), source_ranges: vec![SourceRange([14, 16, 0])], }) ); @@ -3389,7 +3219,7 @@ let notEmptyString = !"" assert_eq!( parse_execute(code3).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { - message: "Cannot apply unary operator ! to non-boolean value: \"\"".to_owned(), + message: "Cannot apply unary operator ! to non-boolean value: string (text)".to_owned(), source_ranges: vec![SourceRange([22, 25, 0])], }) ); @@ -3401,7 +3231,7 @@ let notMember = !obj.a assert_eq!( parse_execute(code4).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { - message: "Cannot apply unary operator ! to non-boolean value: 1".to_owned(), + message: "Cannot apply unary operator ! to non-boolean value: integer".to_owned(), source_ranges: vec![SourceRange([36, 42, 0])], }) ); @@ -3412,7 +3242,7 @@ let notArray = !a"; assert_eq!( parse_execute(code5).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { - message: "Cannot apply unary operator ! to non-boolean value: []".to_owned(), + message: "Cannot apply unary operator ! to non-boolean value: array (list)".to_owned(), source_ranges: vec![SourceRange([27, 29, 0])], }) ); @@ -3423,7 +3253,7 @@ let notObject = !x"; assert_eq!( parse_execute(code6).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { - message: "Cannot apply unary operator ! to non-boolean value: {}".to_owned(), + message: "Cannot apply unary operator ! to non-boolean value: object".to_owned(), source_ranges: vec![SourceRange([28, 30, 0])], }) ); @@ -3451,7 +3281,7 @@ let notTagDeclarator = !myTagDeclarator"; assert!( tag_declarator_err .message() - .starts_with("Cannot apply unary operator ! to non-boolean value: {\"type\":\"TagDeclarator\","), + .starts_with("Cannot apply unary operator ! to non-boolean value: TagDeclarator"), "Actual error: {:?}", tag_declarator_err ); @@ -3465,7 +3295,7 @@ let notTagIdentifier = !myTag"; assert!( tag_identifier_err .message() - .starts_with("Cannot apply unary operator ! to non-boolean value: {\"type\":\"TagIdentifier\","), + .starts_with("Cannot apply unary operator ! to non-boolean value: TagIdentifier"), "Actual error: {:?}", tag_identifier_err ); @@ -3603,10 +3433,10 @@ let w = f() + f() fn test_assign_args_to_params() { // Set up a little framework for this test. fn mem(number: usize) -> KclValue { - KclValue::UserVal(UserVal { - value: number.into(), + KclValue::Int { + value: number as i64, meta: Default::default(), - }) + } } fn ident(s: &'static str) -> Node { Node::no_src(Identifier { diff --git a/src/wasm-lib/kcl/src/std/args.rs b/src/wasm-lib/kcl/src/std/args.rs index 34c233c5ab..431e76b8cf 100644 --- a/src/wasm-lib/kcl/src/std/args.rs +++ b/src/wasm-lib/kcl/src/std/args.rs @@ -1,21 +1,21 @@ -use std::any::type_name; +use std::{any::type_name, num::NonZeroU32}; use anyhow::Result; use kcmc::{websocket::OkWebSocketResponseData, ModelingCmd}; use kittycad_modeling_cmds as kcmc; -use serde::de::DeserializeOwned; -use serde_json::Value as JValue; use crate::{ - ast::types::{execute::parse_json_number_as_f64, TagNode}, + ast::types::TagNode, errors::{KclError, KclErrorDetails}, executor::{ ExecState, ExecutorContext, ExtrudeSurface, KclValue, Metadata, Sketch, SketchSet, SketchSurface, Solid, - SolidSet, SourceRange, TagIdentifier, UserVal, + SolidSet, SourceRange, TagIdentifier, }, - std::{shapes::SketchOrSurface, sketch::FaceTag, FnAsArg}, + std::{shapes::SketchOrSurface, sketch::FaceTag, types::Uint, FnAsArg}, }; +use super::shapes::PolygonType; + #[derive(Debug, Clone)] pub struct Args { pub args: Vec, @@ -181,39 +181,58 @@ impl Args { Ok(()) } - fn make_user_val_from_json(&self, j: serde_json::Value) -> KclValue { - KclValue::UserVal(crate::executor::UserVal { - value: j, - meta: vec![Metadata { - source_range: self.source_range, - }], + pub(crate) fn make_user_val_from_point(&self, p: [f64; 2]) -> Result { + let meta = Metadata { + source_range: self.source_range, + }; + let x = KclValue::Number { + value: p[0], + meta: vec![meta], + }; + let y = KclValue::Number { + value: p[1], + meta: vec![meta], + }; + Ok(KclValue::Array { + value: vec![x, y], + meta: vec![meta], }) } - pub(crate) fn make_null_user_val(&self) -> KclValue { - self.make_user_val_from_json(serde_json::Value::Null) + pub(crate) fn make_user_val_from_f64(&self, f: f64) -> KclValue { + KclValue::from_number( + f, + vec![Metadata { + source_range: self.source_range, + }], + ) } pub(crate) fn make_user_val_from_i64(&self, n: i64) -> KclValue { - self.make_user_val_from_json(serde_json::Value::Number(serde_json::Number::from(n))) - } - - pub(crate) fn make_user_val_from_f64(&self, f: f64) -> Result { - f64_to_jnum(f, vec![self.source_range]).map(|x| self.make_user_val_from_json(x)) - } - - pub(crate) fn make_user_val_from_point(&self, p: [f64; 2]) -> Result { - let x = f64_to_jnum(p[0], vec![self.source_range])?; - let y = f64_to_jnum(p[1], vec![self.source_range])?; - let array = serde_json::Value::Array(vec![x, y]); - Ok(self.make_user_val_from_json(array)) + KclValue::Int { + value: n, + meta: vec![Metadata { + source_range: self.source_range, + }], + } } pub(crate) fn make_user_val_from_f64_array(&self, f: Vec) -> Result { - f.into_iter() - .map(|n| f64_to_jnum(n, vec![self.source_range])) - .collect::, _>>() - .map(|arr| self.make_user_val_from_json(serde_json::Value::Array(arr))) + let array = f + .into_iter() + .map(|n| KclValue::Number { + value: n, + meta: vec![Metadata { + source_range: self.source_range, + }], + }) + .collect::>(); + Ok(KclValue::Array { + value: array, + meta: vec![Metadata { + source_range: self.source_range, + }], + }) } pub(crate) fn get_number(&self) -> Result { @@ -221,11 +240,19 @@ impl Args { } pub(crate) fn get_number_array(&self) -> Result, KclError> { - let mut numbers: Vec = Vec::new(); - for arg in &self.args { - let parsed = arg.get_json_value()?; - numbers.push(parse_json_number_as_f64(&parsed, self.source_range)?); - } + let numbers = self + .args + .iter() + .map(|arg| { + let Some(num) = f64::from_mem_item(arg) else { + return Err(KclError::Semantic(KclErrorDetails { + source_ranges: arg.metadata().iter().map(|x| x.source_range).collect(), + message: format!("Expected a number but found {}", arg.human_friendly_type()), + })); + }; + Ok(num) + }) + .collect::>()?; Ok(numbers) } @@ -474,6 +501,25 @@ pub trait FromKclValue<'a>: Sized { fn from_mem_item(arg: &'a KclValue) -> Option; } +impl<'a> FromArgs<'a> for Vec { + fn from_args(args: &'a Args, i: usize) -> Result { + let Some(arg) = args.args.get(i) else { + return Err(KclError::Semantic(KclErrorDetails { + message: format!("Expected an argument at index {i}"), + source_ranges: vec![args.source_range], + })); + }; + let KclValue::Array { value: array, meta: _ } = arg else { + let message = format!("Expected an array but found {}", arg.human_friendly_type()); + return Err(KclError::Type(KclErrorDetails { + source_ranges: arg.metadata().into_iter().map(|m| m.source_range).collect(), + message, + })); + }; + Ok(array.to_owned()) + } +} + impl<'a, T> FromArgs<'a> for T where T: FromKclValue<'a> + Sized, @@ -511,7 +557,7 @@ where let Some(val) = T::from_mem_item(arg) else { return Err(KclError::Semantic(KclErrorDetails { message: format!( - "Argument at index {i} was supposed to be type {} but found {}", + "Argument at index {i} was supposed to be type Option<{}> but found {}", type_name::(), arg.human_friendly_type() ), @@ -563,31 +609,50 @@ where } } -impl<'a> FromKclValue<'a> for &'a str { +impl<'a> FromKclValue<'a> for [f64; 2] { fn from_mem_item(arg: &'a KclValue) -> Option { - arg.as_user_val().and_then(|uv| uv.value.as_str()) - } -} - -impl<'a> FromKclValue<'a> for i64 { - fn from_mem_item(arg: &'a KclValue) -> Option { - arg.as_user_val() - .and_then(|uv| uv.value.as_number()) - .and_then(|num| num.as_i64()) + let KclValue::Array { value, meta: _ } = arg else { + return None; + }; + if value.len() != 2 { + return None; + } + let v0 = value.first()?; + let v1 = value.get(1)?; + let array = [v0.as_f64()?, v1.as_f64()?]; + Some(array) } } -impl<'a> FromKclValue<'a> for UserVal { +impl<'a> FromKclValue<'a> for [usize; 3] { fn from_mem_item(arg: &'a KclValue) -> Option { - arg.as_user_val().map(|x| x.to_owned()) + let KclValue::Array { value, meta: _ } = arg else { + return None; + }; + if value.len() != 3 { + return None; + } + let v0 = value.first()?; + let v1 = value.get(1)?; + let v2 = value.get(2)?; + let array = [v0.as_usize()?, v1.as_usize()?, v2.as_usize()?]; + Some(array) } } -impl<'a> FromKclValue<'a> for Vec { +impl<'a> FromKclValue<'a> for [f64; 3] { fn from_mem_item(arg: &'a KclValue) -> Option { - arg.as_user_val() - .and_then(|uv| uv.value.as_array()) - .map(ToOwned::to_owned) + let KclValue::Array { value, meta: _ } = arg else { + return None; + }; + if value.len() != 3 { + return None; + } + let v0 = value.first()?; + let v1 = value.get(1)?; + let v2 = value.get(2)?; + let array = [v0.as_f64()?, v1.as_f64()?, v2.as_f64()?]; + Some(array) } } @@ -609,90 +674,767 @@ impl<'a> FromKclValue<'a> for KclValue { } } -macro_rules! impl_from_arg_via_json { - ($typ:path) => { - impl<'a> FromKclValue<'a> for $typ { - fn from_mem_item(arg: &'a KclValue) -> Option { - from_user_val(arg) +macro_rules! fields { + ($obj:ident, $typ:ident, $($field:ident),+) => { + $( + let $field = $obj.get(stringify!($field))?.$typ()?; + )+ + }; + ($obj:ident, $typ:ident, $($field:ident $key:literal),+) => { + $( + let $field = $obj.get($key)?.$typ()?; + )+ + }; +} + +macro_rules! fields_opt { + ($obj:ident, $typ:ident, $($field:ident),+) => { + $( + let $field = $obj.get(stringify!($field)).and_then(|x|x.$typ()); + )+ + }; +} + +macro_rules! fields_recurse { + ($obj:ident, $field:ident) => { + let $field = $obj.get(stringify!($field)).and_then(FromKclValue::from_mem_item)?; + }; + ($obj:ident, $field:ident $k:literal) => { + let $field = $obj.get($k).and_then(FromKclValue::from_mem_item)?; + }; +} + +macro_rules! fields_recurse_opt { + ($obj:ident, $field:ident) => { + let $field = $obj.get(stringify!($field)).and_then(FromKclValue::from_mem_item); + }; + ($obj:ident, $field:ident, $k:literal) => { + let $field = $obj.get($k).and_then(FromKclValue::from_mem_item); + }; +} + +impl<'a> FromKclValue<'a> for crate::std::import::ImportFormat { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_str, typ "type"); + match typ { + "fbx" => Some(Self::Fbx {}), + "gltf" => Some(Self::Gltf {}), + "sldprt" => Some(Self::Sldprt {}), + "step" => Some(Self::Step {}), + "stl" => { + fields_recurse_opt!(obj, coords); + fields_recurse!(obj, units); + Some(Self::Stl { coords, units }) } + "obj" => { + fields_recurse_opt!(obj, coords); + fields_recurse!(obj, units); + Some(Self::Obj { coords, units }) + } + "ply" => { + fields_recurse_opt!(obj, coords); + fields_recurse!(obj, units); + Some(Self::Ply { coords, units }) + } + _ => None, } - }; + } } -impl<'a, T> FromKclValue<'a> for Vec -where - T: serde::de::DeserializeOwned + FromKclValue<'a>, -{ +impl<'a> FromKclValue<'a> for super::sketch::AngledLineThatIntersectsData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields_recurse!(obj, angle); + fields_recurse!(obj, intersect_tag "intersectTag"); + fields_recurse_opt!(obj, offset); + Some(Self { + angle, + intersect_tag, + offset, + }) + } +} + +impl<'a> FromKclValue<'a> for super::shapes::PolygonData { fn from_mem_item(arg: &'a KclValue) -> Option { - from_user_val(arg) + let obj = arg.as_object()?; + fields_recurse!(obj, radius); + fields_recurse!(obj, num_sides "numSides"); + fields_recurse!(obj, center); + fields_recurse!(obj, inscribed); + let polygon_type = if inscribed { + PolygonType::Inscribed + } else { + PolygonType::Circumscribed + }; + Some(Self { + radius, + num_sides, + center, + polygon_type, + inscribed, + }) + } +} + +impl<'a> FromKclValue<'a> for crate::std::polar::PolarCoordsData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields_recurse!(obj, angle); + fields_recurse!(obj, length); + Some(Self { angle, length }) + } +} + +impl<'a> FromKclValue<'a> for crate::std::loft::LoftData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields_recurse_opt!(obj, v_degree); + fields_recurse_opt!(obj, bez_approximate_rational); + fields_recurse_opt!(obj, base_curve_index); + fields_recurse_opt!(obj, tolerance); + Some(Self { + v_degree, + bez_approximate_rational, + base_curve_index, + tolerance, + }) + } +} + +impl<'a> FromKclValue<'a> for crate::std::planes::StandardPlane { + fn from_mem_item(arg: &'a KclValue) -> Option { + let s = arg.as_str()?; + match s { + "XY" | "xy" => Some(Self::XY), + "-XY" | "-xy" => Some(Self::NegXY), + "XZ" | "xz" => Some(Self::XZ), + "-XZ" | "-xz" => Some(Self::NegXZ), + "YZ" | "yz" => Some(Self::YZ), + "-YZ" | "-yz" => Some(Self::NegYZ), + _ => None, + } + } +} + +impl<'a> FromKclValue<'a> for kittycad_modeling_cmds::units::UnitLength { + fn from_mem_item(arg: &'a KclValue) -> Option { + let s = arg.as_str()?; + s.parse().ok() + } +} + +impl<'a> FromKclValue<'a> for kittycad_modeling_cmds::coord::System { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields_recurse!(obj, forward); + fields_recurse!(obj, up); + Some(Self { forward, up }) + } +} + +impl<'a> FromKclValue<'a> for kittycad_modeling_cmds::coord::AxisDirectionPair { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields_recurse!(obj, axis); + fields_recurse!(obj, direction); + Some(Self { axis, direction }) + } +} + +impl<'a> FromKclValue<'a> for kittycad_modeling_cmds::coord::Axis { + fn from_mem_item(arg: &'a KclValue) -> Option { + let s = arg.as_str()?; + match s { + "y" => Some(Self::Y), + "z" => Some(Self::Z), + _ => None, + } + } +} + +impl<'a> FromKclValue<'a> for PolygonType { + fn from_mem_item(arg: &'a KclValue) -> Option { + let s = arg.as_str()?; + match s { + "inscribed" => Some(Self::Inscribed), + _ => Some(Self::Circumscribed), + } + } +} + +impl<'a> FromKclValue<'a> for kittycad_modeling_cmds::coord::Direction { + fn from_mem_item(arg: &'a KclValue) -> Option { + let s = arg.as_str()?; + match s { + "positive" => Some(Self::Positive), + "negative" => Some(Self::Negative), + _ => None, + } + } +} + +impl<'a> FromKclValue<'a> for super::patterns::CircularPattern3dData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, instances); + fields!(obj, as_f64, arc_degrees "arcDegrees"); + fields!(obj, as_bool, rotate_duplicates "rotateDuplicates"); + let instances = Uint::new(instances); + fields_recurse!(obj, axis); + fields_recurse!(obj, center); + Some(Self { + instances, + axis, + center, + arc_degrees, + rotate_duplicates, + }) + } +} + +impl<'a> FromKclValue<'a> for super::patterns::CircularPattern2dData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, instances); + fields!(obj, as_f64, arc_degrees "arcDegrees"); + fields!(obj, as_bool, rotate_duplicates "rotateDuplicates"); + let instances = Uint::new(instances); + fields_recurse!(obj, center); + Some(Self { + instances, + center, + arc_degrees, + rotate_duplicates, + }) + } +} + +impl<'a> FromKclValue<'a> for super::patterns::LinearPattern3dData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, instances, distance); + let instances = Uint::new(instances); + fields_recurse!(obj, axis); + Some(Self { + instances, + distance, + axis, + }) + } +} + +impl<'a> FromKclValue<'a> for super::patterns::LinearPattern2dData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, instances, distance); + let instances = Uint::new(instances); + fields_recurse!(obj, axis); + Some(Self { + instances, + distance, + axis, + }) + } +} + +impl<'a> FromKclValue<'a> for super::sketch::BezierData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_point2d, to, control1, control2); + Some(Self { to, control1, control2 }) + } +} + +impl<'a> FromKclValue<'a> for super::shell::ShellData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, thickness); + fields_recurse!(obj, faces); + Some(Self { thickness, faces }) + } +} + +impl<'a> FromKclValue<'a> for super::chamfer::ChamferData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, length); + fields_recurse!(obj, tags); + Some(Self { length, tags }) + } +} + +impl<'a> FromKclValue<'a> for super::fillet::FilletData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, radius); + fields_opt!(obj, as_f64, tolerance); + fields_recurse!(obj, tags); + Some(Self { + radius, + tolerance, + tags, + }) + } +} + +impl<'a> FromKclValue<'a> for super::helix::HelixData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, revolutions); + fields_opt!(obj, as_f64, length); + fields_opt!(obj, as_bool, ccw); + let ccw = ccw.unwrap_or_default(); + let angle_start = obj.get("angleStart").or_else(|| obj.get("angle_start"))?.as_f64()?; + Some(Self { + revolutions, + angle_start, + ccw, + length, + }) + } +} + +impl<'a> FromKclValue<'a> for FaceTag { + fn from_mem_item(arg: &'a KclValue) -> Option { + let case1 = || match arg.as_str() { + Some("start" | "START") => Some(Self::StartOrEnd(super::sketch::StartOrEnd::Start)), + Some("end" | "END") => Some(Self::StartOrEnd(super::sketch::StartOrEnd::End)), + _ => None, + }; + let case2 = || { + let tag = TagIdentifier::from_mem_item(arg)?; + Some(Self::Tag(Box::new(tag))) + }; + case1().or_else(case2) } } -macro_rules! impl_from_arg_for_array { - ($n:literal) => { - impl<'a, T> FromKclValue<'a> for [T; $n] - where - T: serde::de::DeserializeOwned + FromKclValue<'a>, - { +impl<'a> FromKclValue<'a> for super::sketch::AngledLineToData { + fn from_mem_item(arg: &'a KclValue) -> Option { + // Deserialize from an {angle, to} object. + let case1 = || { + let obj = arg.as_object()?; + fields!(obj, as_f64, angle, to); + Some(Self { angle, to }) + }; + // Deserialize from an [angle, to] array. + let case2 = || { + let [angle, to] = arg.as_point2d()?; + Some(Self { angle, to }) + }; + case1().or_else(case2) + } +} + +impl<'a> FromKclValue<'a> for super::sketch::ArcData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, radius); + let case1 = || { + let angle_start = obj.get("angleStart").or_else(|| obj.get("angle_start"))?.as_f64()?; + let angle_end = obj.get("angleEnd").or_else(|| obj.get("angle_end"))?.as_f64()?; + Some(Self::AnglesAndRadius { + angle_start, + angle_end, + radius, + }) + }; + let case2 = || { + let obj = arg.as_object()?; + fields!(obj, as_point2d, center, to); + Some(Self::CenterToRadius { center, to, radius }) + }; + case1().or_else(case2) + } +} + +impl<'a> FromKclValue<'a> for super::revolve::RevolveData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + let angle = obj.get("angle").and_then(|x| x.as_f64()); + let tolerance = obj.get("tolerance").and_then(|x| x.as_f64()); + fields_recurse!(obj, axis); + Some(Self { angle, axis, tolerance }) + } +} + +impl<'a> FromKclValue<'a> for super::shapes::CircleData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_point2d, center); + fields!(obj, as_f64, radius); + Some(Self { center, radius }) + } +} + +impl<'a> FromKclValue<'a> for super::sketch::TangentialArcData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_f64, radius, offset); + Some(Self::RadiusAndOffset { radius, offset }) + } +} + +impl<'a> FromKclValue<'a> for crate::executor::Point3d { + fn from_mem_item(arg: &'a KclValue) -> Option { + // Case 1: object with x/y/z fields + if let Some(obj) = arg.as_object() { + fields!(obj, as_f64, x, y, z); + return Some(Self { x, y, z }); + } + // Case 2: Array of 3 numbers. + let [x, y, z]: [f64; 3] = FromKclValue::from_mem_item(arg)?; + Some(Self { x, y, z }) + } +} + +impl<'a> FromKclValue<'a> for super::sketch::PlaneData { + fn from_mem_item(arg: &'a KclValue) -> Option { + // Case 0: actual plane + if let KclValue::Plane(p) = arg { + return Some(Self::Plane { + origin: Box::new(p.origin), + x_axis: Box::new(p.x_axis), + y_axis: Box::new(p.y_axis), + z_axis: Box::new(p.z_axis), + }); + } + // Case 1: predefined plane + if let Some(s) = arg.as_str() { + return match s { + "XY" | "xy" => Some(Self::XY), + "-XY" | "-xy" => Some(Self::NegXY), + "XZ" | "xz" => Some(Self::XZ), + "-XZ" | "-xz" => Some(Self::NegXZ), + "YZ" | "yz" => Some(Self::YZ), + "-YZ" | "-yz" => Some(Self::NegYZ), + _ => None, + }; + } + // Case 2: custom plane + let obj = arg.as_object()?; + fields!(obj, as_object, plane); + let origin = plane + .get("origin") + .and_then(FromKclValue::from_mem_item) + .map(Box::new)?; + let x_axis = plane + .get("xAxis") + .or_else(|| plane.get("x_axis")) + .and_then(FromKclValue::from_mem_item) + .map(Box::new)?; + let y_axis = plane + .get("yAxis") + .or_else(|| plane.get("y_axis")) + .and_then(FromKclValue::from_mem_item) + .map(Box::new)?; + let z_axis = plane + .get("zAxis") + .or_else(|| plane.get("z_axis")) + .and_then(FromKclValue::from_mem_item) + .map(Box::new)?; + Some(Self::Plane { + origin, + x_axis, + y_axis, + z_axis, + }) + } +} + +impl<'a> FromKclValue<'a> for crate::executor::ExtrudePlane { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_uuid, face_id "faceId"); + let tag = FromKclValue::from_mem_item(obj.get("tag")?); + fields_recurse!(obj, geo_meta "geoMeta"); + Some(Self { face_id, tag, geo_meta }) + } +} + +impl<'a> FromKclValue<'a> for crate::executor::ExtrudeArc { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_uuid, face_id "faceId"); + let tag = FromKclValue::from_mem_item(obj.get("tag")?); + fields_recurse!(obj, geo_meta "geoMeta"); + Some(Self { face_id, tag, geo_meta }) + } +} + +impl<'a> FromKclValue<'a> for crate::executor::GeoMeta { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_uuid, id); + fields_recurse!(obj, source_range "sourceRange"); + let source_range = SourceRange(source_range); + Some(Self { + id, + metadata: Metadata { source_range }, + }) + } +} + +impl<'a> FromKclValue<'a> for crate::executor::ChamferSurface { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_uuid, face_id "faceId"); + let tag = FromKclValue::from_mem_item(obj.get("tag")?); + fields_recurse!(obj, geo_meta "geoMeta"); + Some(Self { face_id, tag, geo_meta }) + } +} + +impl<'a> FromKclValue<'a> for crate::executor::FilletSurface { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_uuid, face_id "faceId"); + let tag = FromKclValue::from_mem_item(obj.get("tag")?); + fields_recurse!(obj, geo_meta "geoMeta"); + Some(Self { face_id, tag, geo_meta }) + } +} + +impl<'a> FromKclValue<'a> for ExtrudeSurface { + fn from_mem_item(arg: &'a KclValue) -> Option { + let case1 = crate::executor::ExtrudePlane::from_mem_item; + let case2 = crate::executor::ExtrudeArc::from_mem_item; + let case3 = crate::executor::ChamferSurface::from_mem_item; + let case4 = crate::executor::FilletSurface::from_mem_item; + case1(arg) + .map(Self::ExtrudePlane) + .or_else(|| case2(arg).map(Self::ExtrudeArc)) + .or_else(|| case3(arg).map(Self::Chamfer)) + .or_else(|| case4(arg).map(Self::Fillet)) + } +} + +impl<'a> FromKclValue<'a> for crate::executor::EdgeCut { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields!(obj, as_str, typ "type"); + let tag = Box::new(obj.get("tag").and_then(FromKclValue::from_mem_item)); + fields!(obj, as_uuid, edge_id "edgeId"); + fields!(obj, as_uuid, id); + match typ { + "fillet" => { + fields!(obj, as_f64, radius); + Some(Self::Fillet { + edge_id, + tag, + id, + radius, + }) + } + "chamfer" => { + fields!(obj, as_f64, length); + Some(Self::Chamfer { + id, + length, + edge_id, + tag, + }) + } + _ => None, + } + } +} + +macro_rules! impl_from_kcl_for_vec { + ($typ:path) => { + impl<'a> FromKclValue<'a> for Vec<$typ> { fn from_mem_item(arg: &'a KclValue) -> Option { - from_user_val(arg) + arg.as_array()? + .iter() + .map(|value| FromKclValue::from_mem_item(value)) + .collect::>() } } }; } -fn from_user_val(arg: &KclValue) -> Option { - let v = match arg { - KclValue::UserVal(v) => v.value.clone(), - other => serde_json::to_value(other).ok()?, - }; - serde_json::from_value(v).ok() -} - -impl_from_arg_via_json!(super::sketch::AngledLineData); -impl_from_arg_via_json!(super::sketch::AngledLineToData); -impl_from_arg_via_json!(super::sketch::AngledLineThatIntersectsData); -impl_from_arg_via_json!(super::shapes::CircleData); -impl_from_arg_via_json!(super::shapes::PolygonData); -impl_from_arg_via_json!(super::sketch::ArcData); -impl_from_arg_via_json!(super::sketch::TangentialArcData); -impl_from_arg_via_json!(super::sketch::BezierData); -impl_from_arg_via_json!(super::chamfer::ChamferData); -impl_from_arg_via_json!(super::patterns::LinearPattern3dData); -impl_from_arg_via_json!(super::patterns::CircularPattern3dData); -impl_from_arg_via_json!(super::patterns::LinearPattern2dData); -impl_from_arg_via_json!(super::patterns::CircularPattern2dData); -impl_from_arg_via_json!(super::helix::HelixData); -impl_from_arg_via_json!(super::shell::ShellData); -impl_from_arg_via_json!(super::fillet::FilletData); -impl_from_arg_via_json!(super::revolve::RevolveData); -impl_from_arg_via_json!(super::sketch::SketchData); -impl_from_arg_via_json!(crate::std::import::ImportFormat); -impl_from_arg_via_json!(crate::std::polar::PolarCoordsData); -impl_from_arg_via_json!(crate::std::loft::LoftData); -impl_from_arg_via_json!(crate::std::planes::StandardPlane); -impl_from_arg_via_json!(crate::std::mirror::Mirror2dData); -impl_from_arg_via_json!(Sketch); -impl_from_arg_via_json!(FaceTag); -impl_from_arg_via_json!(String); -impl_from_arg_via_json!(crate::ast::types::KclNone); -impl_from_arg_via_json!(u32); -impl_from_arg_via_json!(u64); -impl_from_arg_via_json!(f64); -impl_from_arg_via_json!(bool); - -impl_from_arg_for_array!(2); -impl_from_arg_for_array!(3); +impl_from_kcl_for_vec!(FaceTag); +impl_from_kcl_for_vec!(crate::executor::EdgeCut); +impl_from_kcl_for_vec!(crate::executor::Metadata); +impl_from_kcl_for_vec!(super::fillet::EdgeReference); +impl_from_kcl_for_vec!(ExtrudeSurface); +impl_from_kcl_for_vec!(Sketch); -impl<'a> FromKclValue<'a> for SketchSet { +impl<'a> FromKclValue<'a> for crate::executor::SourceRange { + fn from_mem_item(arg: &'a KclValue) -> Option { + FromKclValue::from_mem_item(arg).map(crate::executor::SourceRange) + } +} + +impl<'a> FromKclValue<'a> for crate::executor::Metadata { + fn from_mem_item(arg: &'a KclValue) -> Option { + FromKclValue::from_mem_item(arg).map(|sr| Self { source_range: sr }) + } +} + +impl<'a> FromKclValue<'a> for crate::executor::Solid { + fn from_mem_item(arg: &'a KclValue) -> Option { + arg.as_solid().cloned() + } +} + +impl<'a> FromKclValue<'a> for super::sketch::SketchData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let case1 = super::sketch::PlaneData::from_mem_item; + let case2 = crate::executor::Solid::from_mem_item; + case1(arg) + .map(Self::Plane) + .or_else(|| case2(arg).map(Box::new).map(Self::Solid)) + } +} + +impl<'a> FromKclValue<'a> for super::revolve::AxisAndOrigin { + fn from_mem_item(arg: &'a KclValue) -> Option { + // Case 1: predefined planes. + if let Some(s) = arg.as_str() { + return match s { + "X" | "x" => Some(Self::X), + "Y" | "y" => Some(Self::Y), + "-X" | "-x" => Some(Self::NegX), + "-Y" | "-y" => Some(Self::NegY), + _ => None, + }; + } + // Case 2: custom planes. + let obj = arg.as_object()?; + fields!(obj, as_object, custom); + fields!(custom, as_point2d, axis, origin); + Some(Self::Custom { axis, origin }) + } +} + +impl<'a> FromKclValue<'a> for super::fillet::EdgeReference { + fn from_mem_item(arg: &'a KclValue) -> Option { + let id = arg.as_uuid().map(Self::Uuid); + let tag = || TagIdentifier::from_mem_item(arg).map(Box::new).map(Self::Tag); + id.or_else(tag) + } +} + +impl<'a> FromKclValue<'a> for super::revolve::AxisOrEdgeReference { + fn from_mem_item(arg: &'a KclValue) -> Option { + let case1 = super::revolve::AxisAndOrigin::from_mem_item; + let case2 = super::fillet::EdgeReference::from_mem_item; + case1(arg).map(Self::Axis).or_else(|| case2(arg).map(Self::Edge)) + } +} + +impl<'a> FromKclValue<'a> for super::mirror::Mirror2dData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let obj = arg.as_object()?; + fields_recurse!(obj, axis); + Some(Self { axis }) + } +} + +impl<'a> FromKclValue<'a> for super::sketch::AngledLineData { + fn from_mem_item(arg: &'a KclValue) -> Option { + let case1 = |arg: &KclValue| { + let obj = arg.as_object()?; + fields!(obj, as_f64, angle, length); + Some(Self::AngleAndLengthNamed { angle, length }) + }; + let case2 = |arg: &KclValue| { + let array = arg.as_array()?; + let ang = array.first()?.as_f64()?; + let len = array.get(1)?.as_f64()?; + Some(Self::AngleAndLengthPair([ang, len])) + }; + case1(arg).or_else(|| case2(arg)) + } +} + +impl<'a> FromKclValue<'a> for i64 { fn from_mem_item(arg: &'a KclValue) -> Option { - let KclValue::UserVal(uv) = arg else { + let KclValue::Int { value, meta: _ } = arg else { return None; }; - if let Some((x, _meta)) = uv.get::() { - Some(SketchSet::from(x)) - } else { - uv.get::>().map(|x| x.0).map(SketchSet::from) + Some(*value) + } +} + +impl<'a> FromKclValue<'a> for u32 { + fn from_mem_item(arg: &'a KclValue) -> Option { + let KclValue::Int { value, meta: _ } = arg else { + return None; + }; + Some(*value as u32) + } +} + +impl<'a> FromKclValue<'a> for NonZeroU32 { + fn from_mem_item(arg: &'a KclValue) -> Option { + u32::from_mem_item(arg).and_then(|x| x.try_into().ok()) + } +} + +impl<'a> FromKclValue<'a> for u64 { + fn from_mem_item(arg: &'a KclValue) -> Option { + let KclValue::Int { value, meta: _ } = arg else { + return None; + }; + Some(*value as u64) + } +} +impl<'a> FromKclValue<'a> for f64 { + fn from_mem_item(arg: &'a KclValue) -> Option { + match arg { + KclValue::Number { value, meta: _ } => Some(*value), + KclValue::Int { value, meta: _ } => Some(*value as f64), + _ => None, + } + } +} +impl<'a> FromKclValue<'a> for Sketch { + fn from_mem_item(arg: &'a KclValue) -> Option { + let KclValue::Sketch { value } = arg else { + return None; + }; + Some(value.as_ref().to_owned()) + } +} +impl<'a> FromKclValue<'a> for String { + fn from_mem_item(arg: &'a KclValue) -> Option { + let KclValue::String { value, meta: _ } = arg else { + return None; + }; + Some(value.to_owned()) + } +} +impl<'a> FromKclValue<'a> for crate::ast::types::KclNone { + fn from_mem_item(arg: &'a KclValue) -> Option { + let KclValue::KclNone { value, meta: _ } = arg else { + return None; + }; + Some(value.to_owned()) + } +} +impl<'a> FromKclValue<'a> for bool { + fn from_mem_item(arg: &'a KclValue) -> Option { + let KclValue::Bool { value, meta: _ } = arg else { + return None; + }; + Some(*value) + } +} + +impl<'a> FromKclValue<'a> for SketchSet { + fn from_mem_item(arg: &'a KclValue) -> Option { + match arg { + KclValue::Sketch { value: sketch } => Some(SketchSet::from(sketch.to_owned())), + KclValue::Sketches { value } => Some(SketchSet::from(value.to_owned())), + KclValue::Array { .. } => { + let v: Option> = FromKclValue::from_mem_item(arg); + Some(SketchSet::Sketches(v?.iter().cloned().map(Box::new).collect())) + } + _ => None, } } } @@ -717,16 +1459,11 @@ impl<'a> FromKclValue<'a> for SolidSet { arg.get_solid_set().ok() } } + impl<'a> FromKclValue<'a> for SketchOrSurface { fn from_mem_item(arg: &'a KclValue) -> Option { match arg { - KclValue::UserVal(uv) => { - if let Some((sg, _meta)) = uv.get() { - Some(Self::Sketch(sg)) - } else { - None - } - } + KclValue::Sketch { value: sg } => Some(Self::Sketch(sg.to_owned())), KclValue::Plane(sg) => Some(Self::SketchSurface(SketchSurface::Plane(sg.clone()))), KclValue::Face(sg) => Some(Self::SketchSurface(SketchSurface::Face(sg.clone()))), _ => None, @@ -743,13 +1480,18 @@ impl<'a> FromKclValue<'a> for SketchSurface { } } -fn f64_to_jnum(f: f64, source_ranges: Vec) -> Result { - serde_json::Number::from_f64(f) - .ok_or_else(|| { - KclError::Type(KclErrorDetails { - message: format!("Failed to convert `{f}` to a number"), - source_ranges, - }) - }) - .map(serde_json::Value::Number) +impl From for Metadata { + fn from(value: Args) -> Self { + Self { + source_range: value.source_range, + } + } +} + +impl From for Vec { + fn from(value: Args) -> Self { + vec![Metadata { + source_range: value.source_range, + }] + } } diff --git a/src/wasm-lib/kcl/src/std/array.rs b/src/wasm-lib/kcl/src/std/array.rs index ee41092ca5..f96966f57c 100644 --- a/src/wasm-lib/kcl/src/std/array.rs +++ b/src/wasm-lib/kcl/src/std/array.rs @@ -1,47 +1,25 @@ use derive_docs::stdlib; -use serde_json::Value as JValue; use super::{args::FromArgs, Args, FnAsArg}; use crate::{ errors::{KclError, KclErrorDetails}, - executor::{ExecState, KclValue, SourceRange, UserVal}, + executor::{ExecState, KclValue, SourceRange}, function_param::FunctionParam, }; /// Apply a function to each element of an array. pub async fn map(exec_state: &mut ExecState, args: Args) -> Result { - let (array, f): (Vec, FnAsArg<'_>) = FromArgs::from_args(&args, 0)?; - let array: Vec = array - .into_iter() - .map(|jval| { - KclValue::UserVal(UserVal { - value: jval, - meta: vec![args.source_range.into()], - }) - }) - .collect(); + let (array, f): (Vec, FnAsArg<'_>) = FromArgs::from_args(&args, 0)?; + let meta = vec![args.source_range.into()]; let map_fn = FunctionParam { inner: f.func, fn_expr: f.expr, - meta: vec![args.source_range.into()], + meta: meta.clone(), ctx: args.ctx.clone(), memory: *f.memory, }; let new_array = inner_map(array, map_fn, exec_state, &args).await?; - let unwrapped = new_array - .clone() - .into_iter() - .map(|k| match k { - KclValue::UserVal(user_val) => Ok(user_val.value), - _ => Err(()), - }) - .collect::, _>>(); - if let Ok(unwrapped) = unwrapped { - let uv = UserVal::new(vec![args.source_range.into()], unwrapped); - return Ok(KclValue::UserVal(uv)); - } - let uv = UserVal::new(vec![args.source_range.into()], new_array); - Ok(KclValue::UserVal(uv)) + Ok(KclValue::Array { value: new_array, meta }) } /// Apply a function to every element of a list. @@ -110,16 +88,7 @@ async fn call_map_closure<'a>( /// For each item in an array, update a value. pub async fn reduce(exec_state: &mut ExecState, args: Args) -> Result { - let (array, start, f): (Vec, KclValue, FnAsArg<'_>) = FromArgs::from_args(&args, 0)?; - let array: Vec = array - .into_iter() - .map(|jval| { - KclValue::UserVal(UserVal { - value: jval, - meta: vec![args.source_range.into()], - }) - }) - .collect(); + let (array, start, f): (Vec, KclValue, FnAsArg<'_>) = FromArgs::from_args(&args, 0)?; let reduce_fn = FunctionParam { inner: f.func, fn_expr: f.expr, @@ -206,50 +175,26 @@ async fn call_reduce_closure<'a>( #[stdlib { name = "push", }] -async fn inner_push(array: Vec, elem: KclValue, args: &Args) -> Result { +async fn inner_push(mut array: Vec, elem: KclValue, args: &Args) -> Result { // Unwrap the KclValues to JValues for manipulation - let mut unwrapped_array = array - .into_iter() - .map(|k| match k { - KclValue::UserVal(user_val) => Ok(user_val.value), - _ => Err(KclError::Semantic(KclErrorDetails { - message: "Expected a UserVal in array".to_string(), - source_ranges: vec![args.source_range], - })), - }) - .collect::, _>>()?; - - // Unwrap the element - let unwrapped_elem = match elem { - KclValue::UserVal(user_val) => user_val.value, - _ => { - return Err(KclError::Semantic(KclErrorDetails { - message: "Expected a UserVal as element".to_string(), - source_ranges: vec![args.source_range], - })); - } - }; - - // Append the element to the array - unwrapped_array.push(unwrapped_elem); - - // Wrap the new array into a UserVal with the source range metadata - let uv = UserVal::new(vec![args.source_range.into()], unwrapped_array); - - // Return the new array wrapped as a KclValue::UserVal - Ok(KclValue::UserVal(uv)) + array.push(elem); + Ok(KclValue::Array { + value: array, + meta: vec![args.source_range.into()], + }) } pub async fn push(_exec_state: &mut ExecState, args: Args) -> Result { // Extract the array and the element from the arguments - let (array_jvalues, elem): (Vec, KclValue) = FromArgs::from_args(&args, 0)?; - - // Convert the array of JValue into Vec - let array: Vec = array_jvalues - .into_iter() - .map(|jval| KclValue::UserVal(UserVal::new(vec![args.source_range.into()], jval))) - .collect(); - - // Call the inner_push function + let (val, elem): (KclValue, KclValue) = FromArgs::from_args(&args, 0)?; + + let meta = vec![args.source_range]; + let KclValue::Array { value: array, meta: _ } = val else { + let actual_type = val.human_friendly_type(); + return Err(KclError::Semantic(KclErrorDetails { + source_ranges: meta, + message: format!("You can't push to a value of type {actual_type}, only an array"), + })); + }; inner_push(array, elem, &args).await } diff --git a/src/wasm-lib/kcl/src/std/assert.rs b/src/wasm-lib/kcl/src/std/assert.rs index 5a71efe73c..6f7a849237 100644 --- a/src/wasm-lib/kcl/src/std/assert.rs +++ b/src/wasm-lib/kcl/src/std/assert.rs @@ -24,7 +24,7 @@ async fn _assert(value: bool, message: &str, args: &Args) -> Result<(), KclError pub async fn assert(_exec_state: &mut ExecState, args: Args) -> Result { let (data, description): (bool, String) = args.get_data()?; inner_assert(data, &description, &args).await?; - Ok(args.make_null_user_val()) + Ok(args.make_user_val_from_f64(0.0)) // TODO: Add a new Void enum for fns that don't return anything. } /// Check a value at runtime, and raise an error if the argument provided @@ -44,7 +44,7 @@ async fn inner_assert(data: bool, message: &str, args: &Args) -> Result<(), KclE pub async fn assert_lt(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_lt(left, right, &description, &args).await?; - Ok(args.make_null_user_val()) + Ok(args.make_user_val_from_f64(0.0)) // TODO: Add a new Void enum for fns that don't return anything. } /// Check that a numerical value is less than to another at runtime, @@ -63,7 +63,7 @@ async fn inner_assert_lt(left: f64, right: f64, message: &str, args: &Args) -> R pub async fn assert_gt(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_gt(left, right, &description, &args).await?; - Ok(args.make_null_user_val()) + Ok(args.make_user_val_from_f64(0.0)) // TODO: Add a new Void enum for fns that don't return anything. } /// Check that a numerical value equals another at runtime, @@ -96,7 +96,7 @@ async fn inner_assert_equal(left: f64, right: f64, epsilon: f64, message: &str, pub async fn assert_equal(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, epsilon, description): (f64, f64, f64, String) = args.get_data()?; inner_assert_equal(left, right, epsilon, &description, &args).await?; - Ok(args.make_null_user_val()) + Ok(args.make_user_val_from_f64(0.0)) // TODO: Add a new Void enum for fns that don't return anything. } /// Check that a numerical value is greater than another at runtime, @@ -115,7 +115,7 @@ async fn inner_assert_gt(left: f64, right: f64, message: &str, args: &Args) -> R pub async fn assert_lte(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_lte(left, right, &description, &args).await?; - Ok(args.make_null_user_val()) + Ok(args.make_user_val_from_f64(0.0)) // TODO: Add a new Void enum for fns that don't return anything. } /// Check that a numerical value is less than or equal to another at runtime, @@ -135,7 +135,7 @@ async fn inner_assert_lte(left: f64, right: f64, message: &str, args: &Args) -> pub async fn assert_gte(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_gte(left, right, &description, &args).await?; - Ok(args.make_null_user_val()) + Ok(args.make_user_val_from_f64(0.0)) // TODO: Add a new Void enum for fns that don't return anything. } /// Check that a numerical value is greater than or equal to another at runtime, diff --git a/src/wasm-lib/kcl/src/std/extrude.rs b/src/wasm-lib/kcl/src/std/extrude.rs index f4e2776055..d8f2b82444 100644 --- a/src/wasm-lib/kcl/src/std/extrude.rs +++ b/src/wasm-lib/kcl/src/std/extrude.rs @@ -233,7 +233,7 @@ pub(crate) async fn do_post_extrude( tag: path.get_base().tag.clone(), geo_meta: GeoMeta { id: path.get_base().geo_meta.id, - metadata: path.get_base().geo_meta.metadata.clone(), + metadata: path.get_base().geo_meta.metadata, }, }); Some(extrude_surface) @@ -244,7 +244,7 @@ pub(crate) async fn do_post_extrude( tag: path.get_base().tag.clone(), geo_meta: GeoMeta { id: path.get_base().geo_meta.id, - metadata: path.get_base().geo_meta.metadata.clone(), + metadata: path.get_base().geo_meta.metadata, }, }); Some(extrude_surface) @@ -259,7 +259,7 @@ pub(crate) async fn do_post_extrude( tag: path.get_base().tag.clone(), geo_meta: GeoMeta { id: path.get_base().geo_meta.id, - metadata: path.get_base().geo_meta.metadata.clone(), + metadata: path.get_base().geo_meta.metadata, }, }); Some(extrude_surface) diff --git a/src/wasm-lib/kcl/src/std/fillet.rs b/src/wasm-lib/kcl/src/std/fillet.rs index 758bcbad47..029fee7181 100644 --- a/src/wasm-lib/kcl/src/std/fillet.rs +++ b/src/wasm-lib/kcl/src/std/fillet.rs @@ -14,7 +14,7 @@ use uuid::Uuid; use crate::{ ast::types::TagNode, errors::{KclError, KclErrorDetails}, - executor::{EdgeCut, ExecState, ExtrudeSurface, FilletSurface, GeoMeta, KclValue, Solid, TagIdentifier, UserVal}, + executor::{EdgeCut, ExecState, ExtrudeSurface, FilletSurface, GeoMeta, KclValue, Solid, TagIdentifier}, settings::types::UnitLength, std::Args, }; @@ -186,15 +186,10 @@ pub async fn get_opposite_edge(exec_state: &mut ExecState, args: Args) -> Result let tag: TagIdentifier = args.get_data()?; let edge = inner_get_opposite_edge(tag, exec_state, args.clone()).await?; - Ok(KclValue::UserVal(UserVal { - value: serde_json::to_value(edge).map_err(|e| { - KclError::Type(KclErrorDetails { - message: format!("Failed to convert Uuid to json: {}", e), - source_ranges: vec![args.source_range], - }) - })?, + Ok(KclValue::Uuid { + value: edge, meta: vec![args.source_range.into()], - })) + }) } /// Get the opposite edge to the edge given. @@ -264,15 +259,10 @@ pub async fn get_next_adjacent_edge(exec_state: &mut ExecState, args: Args) -> R let tag: TagIdentifier = args.get_data()?; let edge = inner_get_next_adjacent_edge(tag, exec_state, args.clone()).await?; - Ok(KclValue::UserVal(UserVal { - value: serde_json::to_value(edge).map_err(|e| { - KclError::Type(KclErrorDetails { - message: format!("Failed to convert Uuid to json: {}", e), - source_ranges: vec![args.source_range], - }) - })?, + Ok(KclValue::Uuid { + value: edge, meta: vec![args.source_range.into()], - })) + }) } /// Get the next adjacent edge to the edge given. @@ -354,15 +344,10 @@ pub async fn get_previous_adjacent_edge(exec_state: &mut ExecState, args: Args) let tag: TagIdentifier = args.get_data()?; let edge = inner_get_previous_adjacent_edge(tag, exec_state, args.clone()).await?; - Ok(KclValue::UserVal(UserVal { - value: serde_json::to_value(edge).map_err(|e| { - KclError::Type(KclErrorDetails { - message: format!("Failed to convert Uuid to json: {}", e), - source_ranges: vec![args.source_range], - }) - })?, + Ok(KclValue::Uuid { + value: edge, meta: vec![args.source_range.into()], - })) + }) } /// Get the previous adjacent edge to the edge given. diff --git a/src/wasm-lib/kcl/src/std/math.rs b/src/wasm-lib/kcl/src/std/math.rs index a81c1ce835..734352e40b 100644 --- a/src/wasm-lib/kcl/src/std/math.rs +++ b/src/wasm-lib/kcl/src/std/math.rs @@ -40,7 +40,7 @@ pub async fn cos(_exec_state: &mut ExecState, args: Args) -> Result Result Result Result { pub async fn pi(_exec_state: &mut ExecState, args: Args) -> Result { let result = inner_pi()?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Return the value of `pi`. Archimedes’ constant (Ï€). @@ -155,7 +155,7 @@ pub async fn sqrt(_exec_state: &mut ExecState, args: Args) -> Result Result Result Result Result Result Result Result Result Result Result Result Result Result Result { pub async fn e(_exec_state: &mut ExecState, args: Args) -> Result { let result = inner_e()?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Return the value of Euler’s number `e`. @@ -648,7 +648,7 @@ fn inner_e() -> Result { pub async fn tau(_exec_state: &mut ExecState, args: Args) -> Result { let result = inner_tau()?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Return the value of `tau`. The full circle constant (Ï„). Equal to 2Ï€. @@ -678,7 +678,7 @@ pub async fn to_radians(_exec_state: &mut ExecState, args: Args) -> Result Result Result { let (hypotenuse, leg) = args.get_hypotenuse_leg()?; let result = inner_leg_length(hypotenuse, leg); - args.make_user_val_from_f64(result) + Ok(KclValue::from_number(result, vec![args.into()])) } /// Compute the length of the given leg. @@ -264,7 +264,7 @@ fn inner_leg_length(hypotenuse: f64, leg: f64) -> f64 { pub async fn leg_angle_x(_exec_state: &mut ExecState, args: Args) -> Result { let (hypotenuse, leg) = args.get_hypotenuse_leg()?; let result = inner_leg_angle_x(hypotenuse, leg); - args.make_user_val_from_f64(result) + Ok(KclValue::from_number(result, vec![args.into()])) } /// Compute the angle of the given leg for x. @@ -284,7 +284,7 @@ fn inner_leg_angle_x(hypotenuse: f64, leg: f64) -> f64 { pub async fn leg_angle_y(_exec_state: &mut ExecState, args: Args) -> Result { let (hypotenuse, leg) = args.get_hypotenuse_leg()?; let result = inner_leg_angle_y(hypotenuse, leg); - args.make_user_val_from_f64(result) + Ok(KclValue::from_number(result, vec![args.into()])) } /// Compute the angle of the given leg for y. diff --git a/src/wasm-lib/kcl/src/std/patterns.rs b/src/wasm-lib/kcl/src/std/patterns.rs index 25c5726715..80e4be3d20 100644 --- a/src/wasm-lib/kcl/src/std/patterns.rs +++ b/src/wasm-lib/kcl/src/std/patterns.rs @@ -14,13 +14,10 @@ use kittycad_modeling_cmds::{ }; use schemars::JsonSchema; use serde::{Deserialize, Serialize}; -use serde_json::Value as JValue; use crate::{ errors::{KclError, KclErrorDetails}, - executor::{ - ExecState, Geometries, Geometry, KclValue, Point3d, Sketch, SketchSet, Solid, SolidSet, SourceRange, UserVal, - }, + executor::{ExecState, Geometries, Geometry, KclValue, Point3d, Sketch, SketchSet, Solid, SolidSet, SourceRange}, function_param::FunctionParam, std::{types::Uint, Args}, }; @@ -361,10 +358,10 @@ async fn make_transform<'a>( exec_state: &mut ExecState, ) -> Result { // Call the transform fn for this repetition. - let repetition_num = KclValue::UserVal(UserVal { - value: JValue::Number(i.into()), + let repetition_num = KclValue::Int { + value: i.into(), meta: vec![source_range.into()], - }); + }; let transform_fn_args = vec![repetition_num]; let transform_fn_return = transform_function.call(exec_state, transform_fn_args).await?; @@ -376,7 +373,7 @@ async fn make_transform<'a>( source_ranges: source_ranges.clone(), }) })?; - let KclValue::UserVal(transform) = transform_fn_return else { + let KclValue::Object { value: transform, meta } = transform_fn_return else { return Err(KclError::Semantic(KclErrorDetails { message: "Transform function must return a transform object".to_string(), source_ranges: source_ranges.clone(), @@ -384,9 +381,9 @@ async fn make_transform<'a>( }; // Apply defaults to the transform. - let replicate = match transform.value.get("replicate") { - Some(JValue::Bool(true)) => true, - Some(JValue::Bool(false)) => false, + let replicate = match transform.get("replicate") { + Some(KclValue::Bool { value: true, .. }) => true, + Some(KclValue::Bool { value: false, .. }) => false, Some(_) => { return Err(KclError::Semantic(KclErrorDetails { message: "The 'replicate' key must be a bool".to_string(), @@ -395,38 +392,43 @@ async fn make_transform<'a>( } None => true, }; - let scale = match transform.value.get("scale") { + let scale = match transform.get("scale") { Some(x) => array_to_point3d(x, source_ranges.clone())?, None => Point3d { x: 1.0, y: 1.0, z: 1.0 }, }; - let translate = match transform.value.get("translate") { + let translate = match transform.get("translate") { Some(x) => array_to_point3d(x, source_ranges.clone())?, None => Point3d { x: 0.0, y: 0.0, z: 0.0 }, }; let mut rotation = Rotation::default(); - if let Some(rot) = transform.value.get("rotation") { + if let Some(rot) = transform.get("rotation") { + let KclValue::Object { value: rot, meta: _ } = rot else { + return Err(KclError::Semantic(KclErrorDetails { + message: "The 'rotation' key must be an object (with optional fields 'angle', 'axis' and 'origin')" + .to_string(), + source_ranges: source_ranges.clone(), + })); + }; if let Some(axis) = rot.get("axis") { rotation.axis = array_to_point3d(axis, source_ranges.clone())?.into(); } if let Some(angle) = rot.get("angle") { match angle { - JValue::Number(number) => { - if let Some(number) = number.as_f64() { - rotation.angle = Angle::from_degrees(number); - } + KclValue::Number { value: number, meta: _ } => { + rotation.angle = Angle::from_degrees(*number); } _ => { return Err(KclError::Semantic(KclErrorDetails { message: "The 'rotation.angle' key must be a number (of degrees)".to_string(), - source_ranges: source_ranges.clone(), + source_ranges: meta.iter().map(|m| m.source_range).collect(), })); } } } if let Some(origin) = rot.get("origin") { rotation.origin = match origin { - JValue::String(s) if s == "local" => OriginType::Local, - JValue::String(s) if s == "global" => OriginType::Global, + KclValue::String { value: s, meta: _ } if s == "local" => OriginType::Local, + KclValue::String { value: s, meta: _ } if s == "global" => OriginType::Global, other => { let origin = array_to_point3d(other, source_ranges.clone())?.into(); OriginType::Custom { origin } @@ -443,8 +445,8 @@ async fn make_transform<'a>( Ok(t) } -fn array_to_point3d(json: &JValue, source_ranges: Vec) -> Result { - let JValue::Array(arr) = json else { +fn array_to_point3d(val: &KclValue, source_ranges: Vec) -> Result { + let KclValue::Array { value: arr, meta } = val else { return Err(KclError::Semantic(KclErrorDetails { message: "Expected an array of 3 numbers (i.e. a 3D point)".to_string(), source_ranges, @@ -457,17 +459,21 @@ fn array_to_point3d(json: &JValue, source_ranges: Vec) -> Result for PlaneData { /// Offset a plane by a distance along its normal. pub async fn offset_plane(exec_state: &mut ExecState, args: Args) -> Result { let (std_plane, offset): (StandardPlane, f64) = args.get_data_and_float()?; - - let plane = inner_offset_plane(std_plane, offset, exec_state).await?; - - Ok(KclValue::UserVal(UserVal::new( - vec![Metadata { - source_range: args.source_range, - }], - plane, - ))) + let plane_data = inner_offset_plane(std_plane, offset, exec_state).await?; + let plane = Plane::from_plane_data(plane_data, exec_state); + Ok(KclValue::Plane(Box::new(plane))) } /// Offset a plane by a distance along its normal. @@ -129,6 +123,20 @@ pub async fn offset_plane(exec_state: &mut ExecState, args: Args) -> Result startProfileAt([0, 0], %) +/// |> circle({radius: 10, center: [0, 0]}, %) +/// +/// // Triangle on the plane 4 units above +/// startSketchOn(offsetPlane("XY", 4)) +/// |> startProfileAt([0, 0], %) +/// |> line([10, 0], %) +/// |> line([0, 10], %) +/// |> close(%) +/// ``` + #[stdlib { name = "offsetPlane", }] diff --git a/src/wasm-lib/kcl/src/std/segment.rs b/src/wasm-lib/kcl/src/std/segment.rs index d5a13f8d0e..965cd274fb 100644 --- a/src/wasm-lib/kcl/src/std/segment.rs +++ b/src/wasm-lib/kcl/src/std/segment.rs @@ -60,7 +60,7 @@ pub async fn segment_end_x(exec_state: &mut ExecState, args: Args) -> Result Result Result Result Result Result Result { pub async fn segment_length(exec_state: &mut ExecState, args: Args) -> Result { let tag: TagIdentifier = args.get_data()?; let result = inner_segment_length(&tag, exec_state, args.clone())?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Compute the length of the provided line segment. @@ -376,7 +376,7 @@ pub async fn segment_angle(exec_state: &mut ExecState, args: Args) -> Result Result { let (tag, to, sketch) = args.get_tag_to_number_sketch()?; let result = inner_angle_to_match_length_x(&tag, to, sketch, exec_state, args.clone())?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Returns the angle to match the given length for x. @@ -478,7 +478,7 @@ fn inner_angle_to_match_length_x( pub async fn angle_to_match_length_y(exec_state: &mut ExecState, args: Args) -> Result { let (tag, to, sketch) = args.get_tag_to_number_sketch()?; let result = inner_angle_to_match_length_y(&tag, to, sketch, exec_state, args.clone())?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Returns the angle to match the given length for y. diff --git a/src/wasm-lib/kcl/src/std/shapes.rs b/src/wasm-lib/kcl/src/std/shapes.rs index 43a28df698..32fd72b7d2 100644 --- a/src/wasm-lib/kcl/src/std/shapes.rs +++ b/src/wasm-lib/kcl/src/std/shapes.rs @@ -48,7 +48,9 @@ pub async fn circle(exec_state: &mut ExecState, args: Args) -> Result Result Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_line_to(to, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a line from the current origin to some absolute (x, y) point. @@ -164,7 +166,9 @@ pub async fn x_line_to(exec_state: &mut ExecState, args: Args) -> Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_x_line_to(to, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a line parallel to the X axis, that ends at the given X. @@ -212,7 +216,9 @@ pub async fn y_line_to(exec_state: &mut ExecState, args: Args) -> Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_y_line_to(to, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a line parallel to the Y axis, that ends at the given Y. @@ -252,7 +258,9 @@ pub async fn line(exec_state: &mut ExecState, args: Args) -> Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_line(delta, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a line relative to the current origin to a specified (x, y) away @@ -333,7 +341,9 @@ pub async fn x_line(exec_state: &mut ExecState, args: Args) -> Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_x_line(length, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a line relative to the current origin to a specified distance away @@ -376,7 +386,9 @@ pub async fn y_line(exec_state: &mut ExecState, args: Args) -> Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_y_line(length, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a line relative to the current origin to a specified distance away @@ -430,7 +442,9 @@ pub async fn angled_line(exec_state: &mut ExecState, args: Args) -> Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_angled_line(data, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a line segment relative to the current origin using the polar @@ -515,7 +529,9 @@ pub async fn angled_line_of_x_length(exec_state: &mut ExecState, args: Args) -> let (data, sketch, tag): (AngledLineData, Sketch, Option) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_angled_line_of_x_length(data, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Create a line segment from the current 2-dimensional sketch origin @@ -573,9 +589,9 @@ async fn inner_angled_line_of_x_length( #[serde(rename_all = "camelCase")] pub struct AngledLineToData { /// The angle of the line. - angle: f64, + pub angle: f64, /// The point to draw to. - to: f64, + pub to: f64, } /// Draw an angled line to a given x coordinate. @@ -583,7 +599,9 @@ pub async fn angled_line_to_x(exec_state: &mut ExecState, args: Args) -> Result< let (data, sketch, tag): (AngledLineToData, Sketch, Option) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_angled_line_to_x(data, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Create a line segment from the current 2-dimensional sketch origin @@ -641,7 +659,9 @@ pub async fn angled_line_of_y_length(exec_state: &mut ExecState, args: Args) -> let new_sketch = inner_angled_line_of_y_length(data, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Create a line segment from the current 2-dimensional sketch origin @@ -700,7 +720,9 @@ pub async fn angled_line_to_y(exec_state: &mut ExecState, args: Args) -> Result< let (data, sketch, tag): (AngledLineToData, Sketch, Option) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_angled_line_to_y(data, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Create a line segment from the current 2-dimensional sketch origin @@ -771,7 +793,9 @@ pub async fn angled_line_that_intersects(exec_state: &mut ExecState, args: Args) let (data, sketch, tag): (AngledLineThatIntersectsData, Sketch, Option) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_angled_line_that_intersects(data, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw an angled line from the current origin, constructing a line segment @@ -828,7 +852,9 @@ pub async fn start_sketch_at(exec_state: &mut ExecState, args: Args) -> Result Result< args.get_data_and_sketch_surface()?; let sketch = inner_start_profile_at(start, sketch_surface, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(sketch.meta.clone(), sketch)) + Ok(KclValue::Sketch { + value: Box::new(sketch), + }) } /// Start a new profile at a given point. @@ -1262,7 +1290,7 @@ pub(crate) async fn inner_start_profile_at( pub async fn profile_start_x(_exec_state: &mut ExecState, args: Args) -> Result { let sketch: Sketch = args.get_sketch()?; let x = inner_profile_start_x(sketch)?; - args.make_user_val_from_f64(x) + Ok(args.make_user_val_from_f64(x)) } /// Extract the provided 2-dimensional sketch's profile's origin's 'x' @@ -1286,7 +1314,7 @@ pub(crate) fn inner_profile_start_x(sketch: Sketch) -> Result { pub async fn profile_start_y(_exec_state: &mut ExecState, args: Args) -> Result { let sketch: Sketch = args.get_sketch()?; let x = inner_profile_start_y(sketch)?; - args.make_user_val_from_f64(x) + Ok(args.make_user_val_from_f64(x)) } /// Extract the provided 2-dimensional sketch's profile's origin's 'y' @@ -1309,15 +1337,7 @@ pub(crate) fn inner_profile_start_y(sketch: Sketch) -> Result { pub async fn profile_start(_exec_state: &mut ExecState, args: Args) -> Result { let sketch: Sketch = args.get_sketch()?; let point = inner_profile_start(sketch)?; - Ok(KclValue::UserVal(UserVal { - value: serde_json::to_value(point).map_err(|e| { - KclError::Type(KclErrorDetails { - message: format!("Failed to convert point to json: {}", e), - source_ranges: vec![args.source_range], - }) - })?, - meta: Default::default(), - })) + Ok(KclValue::from_point2d(point, args.into())) } /// Extract the provided 2-dimensional sketch's profile's origin @@ -1345,7 +1365,9 @@ pub async fn close(exec_state: &mut ExecState, args: Args) -> Result Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_arc(data, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a curved line segment along an imaginary circle. @@ -1573,7 +1597,9 @@ pub async fn tangential_arc(exec_state: &mut ExecState, args: Args) -> Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_tangential_arc(data, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a curved line segment along part of an imaginary circle. @@ -1701,7 +1727,9 @@ pub async fn tangential_arc_to(exec_state: &mut ExecState, args: Args) -> Result let (to, sketch, tag): ([f64; 2], Sketch, Option) = super::args::FromArgs::from_args(&args, 0)?; let new_sketch = inner_tangential_arc_to(to, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a tangential arc to point some distance away.. @@ -1709,7 +1737,9 @@ pub async fn tangential_arc_to_relative(exec_state: &mut ExecState, args: Args) let (delta, sketch, tag): ([f64; 2], Sketch, Option) = super::args::FromArgs::from_args(&args, 0)?; let new_sketch = inner_tangential_arc_to_relative(delta, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Starting at the current sketch's origin, draw a curved line segment along @@ -1873,11 +1903,11 @@ async fn inner_tangential_arc_to_relative( #[serde(rename_all = "camelCase")] pub struct BezierData { /// The to point. - to: [f64; 2], + pub to: [f64; 2], /// The first control point. - control1: [f64; 2], + pub control1: [f64; 2], /// The second control point. - control2: [f64; 2], + pub control2: [f64; 2], } /// Draw a bezier curve. @@ -1885,7 +1915,9 @@ pub async fn bezier_curve(exec_state: &mut ExecState, args: Args) -> Result) = args.get_data_and_sketch_and_tag()?; let new_sketch = inner_bezier_curve(data, sketch, tag, exec_state, args).await?; - Ok(KclValue::new_user_val(new_sketch.meta.clone(), new_sketch)) + Ok(KclValue::Sketch { + value: Box::new(new_sketch), + }) } /// Draw a smooth, continuous, curved line segment from the current origin to @@ -1965,7 +1997,9 @@ pub async fn hole(exec_state: &mut ExecState, args: Args) -> Result Result { let result = inner_mm(&args)?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Millimeters conversion factor for current projects units. @@ -55,7 +55,7 @@ fn inner_mm(args: &Args) -> Result { pub async fn inch(_exec_state: &mut ExecState, args: Args) -> Result { let result = inner_inch(&args)?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Inches conversion factor for current projects units. @@ -96,7 +96,7 @@ fn inner_inch(args: &Args) -> Result { pub async fn ft(_exec_state: &mut ExecState, args: Args) -> Result { let result = inner_ft(&args)?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Feet conversion factor for current projects units. @@ -138,7 +138,7 @@ fn inner_ft(args: &Args) -> Result { pub async fn m(_exec_state: &mut ExecState, args: Args) -> Result { let result = inner_m(&args)?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Meters conversion factor for current projects units. @@ -180,7 +180,7 @@ fn inner_m(args: &Args) -> Result { pub async fn cm(_exec_state: &mut ExecState, args: Args) -> Result { let result = inner_cm(&args)?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Centimeters conversion factor for current projects units. @@ -222,7 +222,7 @@ fn inner_cm(args: &Args) -> Result { pub async fn yd(_exec_state: &mut ExecState, args: Args) -> Result { let result = inner_yd(&args)?; - args.make_user_val_from_f64(result) + Ok(args.make_user_val_from_f64(result)) } /// Yards conversion factor for current projects units. diff --git a/src/wasm-lib/kcl/tests/add_lots/program_memory.snap b/src/wasm-lib/kcl/tests/add_lots/program_memory.snap index d15686eaca..c91b27db5b 100644 --- a/src/wasm-lib/kcl/tests/add_lots/program_memory.snap +++ b/src/wasm-lib/kcl/tests/add_lots/program_memory.snap @@ -8,27 +8,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "f": { @@ -89,27 +85,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] } }, @@ -130,14 +122,1420 @@ snapshot_kind: text ] }, "x": { - "type": "UserVal", - "type": "UserVal", + "type": "Number", "value": 10100.0, "__meta": [ { "sourceRange": [ - 38, - 834, + 40, + 41, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 47, + 48, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 54, + 55, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 61, + 62, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 68, + 69, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 75, + 76, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 82, + 83, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 89, + 90, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 96, + 97, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 103, + 104, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 110, + 112, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 118, + 120, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 126, + 128, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 134, + 136, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 142, + 144, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 150, + 152, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 158, + 160, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 166, + 168, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 174, + 176, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 182, + 184, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 190, + 192, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 198, + 200, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 206, + 208, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 214, + 216, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 222, + 224, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 230, + 232, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 238, + 240, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 246, + 248, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 254, + 256, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 262, + 264, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 270, + 272, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 278, + 280, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 286, + 288, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 294, + 296, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 302, + 304, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 310, + 312, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 318, + 320, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 326, + 328, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 334, + 336, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 342, + 344, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 350, + 352, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 358, + 360, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 366, + 368, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 374, + 376, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 382, + 384, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 390, + 392, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 398, + 400, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 406, + 408, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 414, + 416, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 422, + 424, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 430, + 432, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 438, + 440, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 446, + 448, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 454, + 456, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 462, + 464, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 470, + 472, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 478, + 480, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 486, + 488, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 494, + 496, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 502, + 504, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 510, + 512, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 518, + 520, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 526, + 528, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 534, + 536, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 542, + 544, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 550, + 552, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 558, + 560, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 566, + 568, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 574, + 576, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 582, + 584, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 590, + 592, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 598, + 600, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 606, + 608, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 614, + 616, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 622, + 624, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 630, + 632, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 638, + 640, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 646, + 648, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 654, + 656, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 662, + 664, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 670, + 672, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 678, + 680, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 686, + 688, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 694, + 696, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 702, + 704, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 710, + 712, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 718, + 720, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 726, + 728, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 734, + 736, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 742, + 744, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 750, + 752, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 758, + 760, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 766, + 768, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 774, + 776, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 782, + 784, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 790, + 792, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 798, + 800, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 806, + 808, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 814, + 816, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 822, + 824, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, + 0 + ] + }, + { + "sourceRange": [ + 830, + 833, + 0 + ] + }, + { + "sourceRange": [ + 29, + 30, 0 ] } diff --git a/src/wasm-lib/kcl/tests/array_elem_push/program_memory.snap b/src/wasm-lib/kcl/tests/array_elem_push/program_memory.snap index 7eb65a5fca..1e67c7f256 100644 --- a/src/wasm-lib/kcl/tests/array_elem_push/program_memory.snap +++ b/src/wasm-lib/kcl/tests/array_elem_push/program_memory.snap @@ -8,36 +8,67 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "arr": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 1, - 2, - 3 + { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 7, + 8, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 2, + "__meta": [ + { + "sourceRange": [ + 10, + 11, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 3, + "__meta": [ + { + "sourceRange": [ + 13, + 14, + 0 + ] + } + ] + } ], "__meta": [ { @@ -50,13 +81,60 @@ snapshot_kind: text ] }, "new_arr1": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 1, - 2, - 3, - 4 + { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 7, + 8, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 2, + "__meta": [ + { + "sourceRange": [ + 10, + 11, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 3, + "__meta": [ + { + "sourceRange": [ + 13, + 14, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 4, + "__meta": [ + { + "sourceRange": [ + 37, + 38, + 0 + ] + } + ] + } ], "__meta": [ { @@ -69,14 +147,73 @@ snapshot_kind: text ] }, "new_arr2": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 1, - 2, - 3, - 4, - 5 + { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 7, + 8, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 2, + "__meta": [ + { + "sourceRange": [ + 10, + 11, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 3, + "__meta": [ + { + "sourceRange": [ + 13, + 14, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 4, + "__meta": [ + { + "sourceRange": [ + 37, + 38, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 5, + "__meta": [ + { + "sourceRange": [ + 66, + 67, + 0 + ] + } + ] + } ], "__meta": [ { diff --git a/src/wasm-lib/kcl/tests/array_range_expr/program_memory.snap b/src/wasm-lib/kcl/tests/array_range_expr/program_memory.snap index 90b6736b8a..5b778464e5 100644 --- a/src/wasm-lib/kcl/tests/array_range_expr/program_memory.snap +++ b/src/wasm-lib/kcl/tests/array_range_expr/program_memory.snap @@ -8,32 +8,27 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "five": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 5, "__meta": [ { @@ -46,8 +41,7 @@ snapshot_kind: text ] }, "four": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 4, "__meta": [ { @@ -60,14 +54,73 @@ snapshot_kind: text ] }, "r1": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 0, - 1, - 2, - 3, - 4 + { + "type": "Int", + "value": 0, + "__meta": [ + { + "sourceRange": [ + 5, + 11, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 5, + 11, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 2, + "__meta": [ + { + "sourceRange": [ + 5, + 11, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 3, + "__meta": [ + { + "sourceRange": [ + 5, + 11, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 4, + "__meta": [ + { + "sourceRange": [ + 5, + 11, + 0 + ] + } + ] + } ], "__meta": [ { @@ -80,14 +133,73 @@ snapshot_kind: text ] }, "r2": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 0, - 1, - 2, - 3, - 4 + { + "type": "Int", + "value": 0, + "__meta": [ + { + "sourceRange": [ + 95, + 107, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 95, + 107, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 2, + "__meta": [ + { + "sourceRange": [ + 95, + 107, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 3, + "__meta": [ + { + "sourceRange": [ + 95, + 107, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 4, + "__meta": [ + { + "sourceRange": [ + 95, + 107, + 0 + ] + } + ] + } ], "__meta": [ { @@ -100,15 +212,86 @@ snapshot_kind: text ] }, "r3": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 0, - 1, - 2, - 3, - 4, - 5 + { + "type": "Int", + "value": 0, + "__meta": [ + { + "sourceRange": [ + 194, + 206, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 194, + 206, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 2, + "__meta": [ + { + "sourceRange": [ + 194, + 206, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 3, + "__meta": [ + { + "sourceRange": [ + 194, + 206, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 4, + "__meta": [ + { + "sourceRange": [ + 194, + 206, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 5, + "__meta": [ + { + "sourceRange": [ + 194, + 206, + 0 + ] + } + ] + } ], "__meta": [ { @@ -121,13 +304,60 @@ snapshot_kind: text ] }, "r4": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 1, - 2, - 3, - 4 + { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 341, + 373, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 2, + "__meta": [ + { + "sourceRange": [ + 341, + 373, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 3, + "__meta": [ + { + "sourceRange": [ + 341, + 373, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 4, + "__meta": [ + { + "sourceRange": [ + 341, + 373, + 0 + ] + } + ] + } ], "__meta": [ { @@ -140,8 +370,7 @@ snapshot_kind: text ] }, "zero": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 0, "__meta": [ { diff --git a/src/wasm-lib/kcl/tests/array_range_negative_expr/program_memory.snap b/src/wasm-lib/kcl/tests/array_range_negative_expr/program_memory.snap index b62ff7e493..4b1b8bfa62 100644 --- a/src/wasm-lib/kcl/tests/array_range_negative_expr/program_memory.snap +++ b/src/wasm-lib/kcl/tests/array_range_negative_expr/program_memory.snap @@ -8,44 +8,171 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "xs": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - -5, - -4, - -3, - -2, - -1, - 0, - 1, - 2, - 3, - 4, - 5 + { + "type": "Int", + "value": -5, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": -4, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": -3, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": -2, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": -1, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 0, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 2, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 3, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 4, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 5, + "__meta": [ + { + "sourceRange": [ + 5, + 19, + 0 + ] + } + ] + } ], "__meta": [ { diff --git a/src/wasm-lib/kcl/tests/comparisons/program_memory.snap b/src/wasm-lib/kcl/tests/comparisons/program_memory.snap index 1d8fb2ed8f..3ed819d9c9 100644 --- a/src/wasm-lib/kcl/tests/comparisons/program_memory.snap +++ b/src/wasm-lib/kcl/tests/comparisons/program_memory.snap @@ -8,27 +8,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] } }, diff --git a/src/wasm-lib/kcl/tests/cube/program_memory.snap b/src/wasm-lib/kcl/tests/cube/program_memory.snap index 5f76caf6be..719ca526fa 100644 --- a/src/wasm-lib/kcl/tests/cube/program_memory.snap +++ b/src/wasm-lib/kcl/tests/cube/program_memory.snap @@ -8,27 +8,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "cube": { @@ -717,27 +713,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] } }, diff --git a/src/wasm-lib/kcl/tests/double_map_fn/program_memory.snap b/src/wasm-lib/kcl/tests/double_map_fn/program_memory.snap index 6ac6c1eaf0..14431a553f 100644 --- a/src/wasm-lib/kcl/tests/double_map_fn/program_memory.snap +++ b/src/wasm-lib/kcl/tests/double_map_fn/program_memory.snap @@ -8,27 +8,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "increment": { @@ -89,27 +85,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] } }, @@ -130,12 +122,47 @@ snapshot_kind: text ] }, "xs": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 0, - 1, - 2 + { + "type": "Int", + "value": 0, + "__meta": [ + { + "sourceRange": [ + 47, + 53, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 47, + 53, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 2, + "__meta": [ + { + "sourceRange": [ + 47, + 53, + 0 + ] + } + ] + } ], "__meta": [ { @@ -148,12 +175,89 @@ snapshot_kind: text ] }, "ys": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 2.0, - 3.0, - 4.0 + { + "type": "Number", + "value": 2.0, + "__meta": [ + { + "sourceRange": [ + 47, + 53, + 0 + ] + }, + { + "sourceRange": [ + 37, + 38, + 0 + ] + }, + { + "sourceRange": [ + 37, + 38, + 0 + ] + } + ] + }, + { + "type": "Number", + "value": 3.0, + "__meta": [ + { + "sourceRange": [ + 47, + 53, + 0 + ] + }, + { + "sourceRange": [ + 37, + 38, + 0 + ] + }, + { + "sourceRange": [ + 37, + 38, + 0 + ] + } + ] + }, + { + "type": "Number", + "value": 4.0, + "__meta": [ + { + "sourceRange": [ + 47, + 53, + 0 + ] + }, + { + "sourceRange": [ + 37, + 38, + 0 + ] + }, + { + "sourceRange": [ + 37, + 38, + 0 + ] + } + ] + } ], "__meta": [ { diff --git a/src/wasm-lib/kcl/tests/helix_ccw/program_memory.snap b/src/wasm-lib/kcl/tests/helix_ccw/program_memory.snap index 6824b22cb0..58924587a2 100644 --- a/src/wasm-lib/kcl/tests/helix_ccw/program_memory.snap +++ b/src/wasm-lib/kcl/tests/helix_ccw/program_memory.snap @@ -8,27 +8,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "part001": { diff --git a/src/wasm-lib/kcl/tests/if_else/program_memory.snap b/src/wasm-lib/kcl/tests/if_else/program_memory.snap index 1a6a881f8f..c72b2632c2 100644 --- a/src/wasm-lib/kcl/tests/if_else/program_memory.snap +++ b/src/wasm-lib/kcl/tests/if_else/program_memory.snap @@ -8,32 +8,27 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "a": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 3, "__meta": [ { @@ -46,8 +41,7 @@ snapshot_kind: text ] }, "b": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 4, "__meta": [ { @@ -60,8 +54,7 @@ snapshot_kind: text ] }, "c": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 5, "__meta": [ { diff --git a/src/wasm-lib/kcl/tests/index_of_array/program_memory.snap b/src/wasm-lib/kcl/tests/index_of_array/program_memory.snap index 06c3f72782..9ce06297de 100644 --- a/src/wasm-lib/kcl/tests/index_of_array/program_memory.snap +++ b/src/wasm-lib/kcl/tests/index_of_array/program_memory.snap @@ -8,36 +8,67 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "array": { - "type": "UserVal", - "type": "UserVal", + "type": "Array", "value": [ - 90, - 91, - 92 + { + "type": "Int", + "value": 90, + "__meta": [ + { + "sourceRange": [ + 44, + 46, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 91, + "__meta": [ + { + "sourceRange": [ + 48, + 50, + 0 + ] + } + ] + }, + { + "type": "Int", + "value": 92, + "__meta": [ + { + "sourceRange": [ + 52, + 54, + 0 + ] + } + ] + } ], "__meta": [ { @@ -50,8 +81,7 @@ snapshot_kind: text ] }, "i": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 1, "__meta": [ { @@ -64,28 +94,26 @@ snapshot_kind: text ] }, "result0": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 91, "__meta": [ { "sourceRange": [ - 93, - 101, + 48, + 50, 0 ] } ] }, "result1": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 91, "__meta": [ { "sourceRange": [ - 277, - 285, + 48, + 50, 0 ] } diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane4.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane4.png index 701b46c98e..dea293e351 100644 Binary files a/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane4.png and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane4.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_polygon1.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_polygon1.png index 3b3c303c7e..7b262d824e 100644 Binary files a/src/wasm-lib/kcl/tests/outputs/serial_test_example_polygon1.png and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_polygon1.png differ diff --git a/src/wasm-lib/kcl/tests/property_of_object/program_memory.snap b/src/wasm-lib/kcl/tests/property_of_object/program_memory.snap index 35a8f4ef15..f03732d627 100644 --- a/src/wasm-lib/kcl/tests/property_of_object/program_memory.snap +++ b/src/wasm-lib/kcl/tests/property_of_object/program_memory.snap @@ -8,35 +8,54 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "obj": { - "type": "UserVal", - "type": "UserVal", + "type": "Object", "value": { - "bar": 0, - "foo": 1 + "bar": { + "type": "Int", + "value": 0, + "__meta": [ + { + "sourceRange": [ + 71, + 72, + 0 + ] + } + ] + }, + "foo": { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 63, + 64, + 0 + ] + } + ] + } }, "__meta": [ { @@ -49,12 +68,47 @@ snapshot_kind: text ] }, "obj2": { - "type": "UserVal", - "type": "UserVal", + "type": "Object", "value": { "inner": { - "bar": 0, - "foo": 1 + "type": "Object", + "value": { + "bar": { + "type": "Int", + "value": 0, + "__meta": [ + { + "sourceRange": [ + 71, + 72, + 0 + ] + } + ] + }, + "foo": { + "type": "Int", + "value": 1, + "__meta": [ + { + "sourceRange": [ + 63, + 64, + 0 + ] + } + ] + } + }, + "__meta": [ + { + "sourceRange": [ + 56, + 74, + 0 + ] + } + ] } }, "__meta": [ @@ -68,64 +122,59 @@ snapshot_kind: text ] }, "one_a": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 1, "__meta": [ { "sourceRange": [ - 122, - 132, + 63, + 64, 0 ] } ] }, "one_b": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 1, "__meta": [ { "sourceRange": [ - 356, - 362, + 63, + 64, 0 ] } ] }, "one_c": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 1, "__meta": [ { "sourceRange": [ - 553, - 570, + 63, + 64, 0 ] } ] }, "one_d": { - "type": "UserVal", - "type": "UserVal", + "type": "Int", "value": 1, "__meta": [ { "sourceRange": [ - 757, - 770, + 63, + 64, 0 ] } ] }, "p": { - "type": "UserVal", - "type": "UserVal", + "type": "String", "value": "foo", "__meta": [ { diff --git a/src/wasm-lib/kcl/tests/sketch_in_object/program_memory.snap b/src/wasm-lib/kcl/tests/sketch_in_object/program_memory.snap index 7ae5862740..67147fe9d7 100644 --- a/src/wasm-lib/kcl/tests/sketch_in_object/program_memory.snap +++ b/src/wasm-lib/kcl/tests/sketch_in_object/program_memory.snap @@ -8,27 +8,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "test": { @@ -295,27 +291,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] } }, @@ -637,27 +629,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] }, "test": { @@ -924,27 +912,23 @@ snapshot_kind: text { "bindings": { "HALF_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 180, + "type": "Number", + "value": 180.0, "__meta": [] }, "QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 90, + "type": "Number", + "value": 90.0, "__meta": [] }, "THREE_QUARTER_TURN": { - "type": "UserVal", - "type": "UserVal", - "value": 270, + "type": "Number", + "value": 270.0, "__meta": [] }, "ZERO": { - "type": "UserVal", - "type": "UserVal", - "value": 0, + "type": "Number", + "value": 0.0, "__meta": [] } }, @@ -982,45 +966,10 @@ snapshot_kind: text ] }, "x": { - "type": "UserVal", - "type": "UserVal", + "type": "Sketch", "value": { - "__meta": [ - { - "sourceRange": [ - 52, - 77, - 0 - ] - } - ], + "type": "Sketch", "id": "[uuid]", - "on": { - "__meta": [], - "id": "[uuid]", - "origin": { - "x": 0.0, - "y": 0.0, - "z": 0.0 - }, - "type": "plane", - "value": "XY", - "xAxis": { - "x": 1.0, - "y": 0.0, - "z": 0.0 - }, - "yAxis": { - "x": 0.0, - "y": 1.0, - "z": 0.0 - }, - "zAxis": { - "x": 0.0, - "y": 0.0, - "z": 1.0 - } - }, "paths": [ { "__geoMeta": { @@ -1103,182 +1052,221 @@ snapshot_kind: text "type": "ToPoint" } ], - "start": { - "__geoMeta": { - "id": "[uuid]", - "sourceRange": [ - 52, - 77, - 0 - ] + "on": { + "type": "plane", + "id": "[uuid]", + "value": "XY", + "origin": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "xAxis": { + "x": 1.0, + "y": 0.0, + "z": 0.0 + }, + "yAxis": { + "x": 0.0, + "y": 1.0, + "z": 0.0 + }, + "zAxis": { + "x": 0.0, + "y": 0.0, + "z": 1.0 }, + "__meta": [] + }, + "start": { "from": [ 0.0, 0.0 ], - "tag": null, "to": [ 0.0, 0.0 - ] + ], + "tag": null, + "__geoMeta": { + "id": "[uuid]", + "sourceRange": [ + 52, + 77, + 0 + ] + } }, - "type": "Sketch" - }, - "__meta": [ - { - "sourceRange": [ - 52, - 77, - 0 - ] - } - ] + "__meta": [ + { + "sourceRange": [ + 52, + 77, + 0 + ] + } + ] + } }, "x2": { - "type": "UserVal", - "type": "UserVal", + "type": "Object", "value": { "thing1": { - "thing2": { - "__meta": [ - { - "sourceRange": [ - 242, - 267, - 0 - ] - } - ], - "id": "[uuid]", - "on": { - "__meta": [], - "id": "[uuid]", - "origin": { - "x": 0.0, - "y": 0.0, - "z": 0.0 - }, - "type": "plane", - "value": "XY", - "xAxis": { - "x": 1.0, - "y": 0.0, - "z": 0.0 - }, - "yAxis": { - "x": 0.0, - "y": 1.0, - "z": 0.0 - }, - "zAxis": { - "x": 0.0, - "y": 0.0, - "z": 1.0 - } - }, - "paths": [ - { - "__geoMeta": { - "id": "[uuid]", - "sourceRange": [ - 277, - 292, - 0 - ] - }, - "from": [ - 0.0, - 0.0 - ], - "tag": null, - "to": [ - 0.0, - 1.0 - ], - "type": "ToPoint" - }, - { - "__geoMeta": { - "id": "[uuid]", - "sourceRange": [ - 302, - 317, - 0 - ] - }, - "from": [ - 0.0, - 1.0 - ], - "tag": null, - "to": [ - 1.0, - 1.0 + "type": "Object", + "value": { + "thing2": { + "type": "Sketch", + "value": { + "type": "Sketch", + "id": "[uuid]", + "paths": [ + { + "__geoMeta": { + "id": "[uuid]", + "sourceRange": [ + 277, + 292, + 0 + ] + }, + "from": [ + 0.0, + 0.0 + ], + "tag": null, + "to": [ + 0.0, + 1.0 + ], + "type": "ToPoint" + }, + { + "__geoMeta": { + "id": "[uuid]", + "sourceRange": [ + 302, + 317, + 0 + ] + }, + "from": [ + 0.0, + 1.0 + ], + "tag": null, + "to": [ + 1.0, + 1.0 + ], + "type": "ToPoint" + }, + { + "__geoMeta": { + "id": "[uuid]", + "sourceRange": [ + 327, + 343, + 0 + ] + }, + "from": [ + 1.0, + 1.0 + ], + "tag": null, + "to": [ + 1.0, + 0.0 + ], + "type": "ToPoint" + }, + { + "__geoMeta": { + "id": "[uuid]", + "sourceRange": [ + 353, + 361, + 0 + ] + }, + "from": [ + 1.0, + 0.0 + ], + "tag": null, + "to": [ + 0.0, + 0.0 + ], + "type": "ToPoint" + } ], - "type": "ToPoint" - }, - { - "__geoMeta": { + "on": { + "type": "plane", "id": "[uuid]", - "sourceRange": [ - 327, - 343, - 0 - ] + "value": "XY", + "origin": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "xAxis": { + "x": 1.0, + "y": 0.0, + "z": 0.0 + }, + "yAxis": { + "x": 0.0, + "y": 1.0, + "z": 0.0 + }, + "zAxis": { + "x": 0.0, + "y": 0.0, + "z": 1.0 + }, + "__meta": [] }, - "from": [ - 1.0, - 1.0 - ], - "tag": null, - "to": [ - 1.0, - 0.0 - ], - "type": "ToPoint" - }, - { - "__geoMeta": { - "id": "[uuid]", - "sourceRange": [ - 353, - 361, - 0 - ] + "start": { + "from": [ + 0.0, + 0.0 + ], + "to": [ + 0.0, + 0.0 + ], + "tag": null, + "__geoMeta": { + "id": "[uuid]", + "sourceRange": [ + 242, + 267, + 0 + ] + } }, - "from": [ - 1.0, - 0.0 - ], - "tag": null, - "to": [ - 0.0, - 0.0 - ], - "type": "ToPoint" - } - ], - "start": { - "__geoMeta": { - "id": "[uuid]", - "sourceRange": [ - 242, - 267, - 0 + "__meta": [ + { + "sourceRange": [ + 242, + 267, + 0 + ] + } ] - }, - "from": [ - 0.0, - 0.0 - ], - "tag": null, - "to": [ - 0.0, - 0.0 + } + } + }, + "__meta": [ + { + "sourceRange": [ + 199, + 365, + 0 ] - }, - "type": "Sketch" - } + } + ] } }, "__meta": [ diff --git a/src/wasm-lib/tests/executor/inputs/member_expression_sketch.kcl b/src/wasm-lib/tests/executor/inputs/member_expression_sketch.kcl index 5e18848c8f..6d4a3b7659 100644 --- a/src/wasm-lib/tests/executor/inputs/member_expression_sketch.kcl +++ b/src/wasm-lib/tests/executor/inputs/member_expression_sketch.kcl @@ -13,5 +13,5 @@ sq = square([0,0], 10) cb = square([3,3], 4) |> extrude(10, %) -pt1 = sq.paths[0] -pt2 = cb.value[0] +// pt1 = sq.paths[0] +// pt2 = cb.value[0] diff --git a/src/wasm-lib/tests/executor/main.rs b/src/wasm-lib/tests/executor/main.rs index 034adb81d3..9b261acb70 100644 --- a/src/wasm-lib/tests/executor/main.rs +++ b/src/wasm-lib/tests/executor/main.rs @@ -1354,7 +1354,7 @@ secondSketch = startSketchOn(part001, '') assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([260, 286, 0])], message: "Argument at index 1 was supposed to be type kcl_lib::std::sketch::FaceTag but found string (text)" }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([260, 286, 0])], message: "Argument at index 1 was supposed to be type Option but found string (text)" }"# ); } diff --git a/src/wasm-lib/tests/executor/no_visuals.rs b/src/wasm-lib/tests/executor/no_visuals.rs index 8f573fbfcb..8714aaafe2 100644 --- a/src/wasm-lib/tests/executor/no_visuals.rs +++ b/src/wasm-lib/tests/executor/no_visuals.rs @@ -73,7 +73,7 @@ gen_test_fail!( ); gen_test_fail!( invalid_index_negative, - "semantic: '-1' is not a valid index, indices must be whole positive numbers" + "semantic: '-1' is negative, so you can't index an array with it" ); gen_test_fail!( invalid_index_fractional, @@ -81,7 +81,7 @@ gen_test_fail!( ); gen_test_fail!( invalid_member_object, - "semantic: Only arrays and objects can be indexed, but you're trying to index a number" + "semantic: Only arrays and objects can be indexed, but you're trying to index an integer" ); gen_test_fail!( invalid_member_object_prop, @@ -107,7 +107,10 @@ gen_test_fail!( // if_else_no_expr, // "syntax: blocks inside an if/else expression must end in an expression" // ); -gen_test_fail!(comparisons_multiple, "syntax: Invalid number: true"); +gen_test_fail!( + comparisons_multiple, + "semantic: Expected a number, but found a boolean (true/false value)" +); gen_test_fail!( import_cycle1, "import cycle: circular import of modules is not allowed: tests/executor/inputs/no_visuals/import_cycle2.kcl -> tests/executor/inputs/no_visuals/import_cycle3.kcl -> tests/executor/inputs/no_visuals/import_cycle1.kcl -> tests/executor/inputs/no_visuals/import_cycle2.kcl" diff --git a/src/wasm-lib/tests/modify/main.rs b/src/wasm-lib/tests/modify/main.rs index bf18aa734a..6bb9389da7 100644 --- a/src/wasm-lib/tests/modify/main.rs +++ b/src/wasm-lib/tests/modify/main.rs @@ -4,7 +4,7 @@ use kcl_lib::{ modify::modify_ast_for_sketch, types::{ModuleId, Node, Program}, }, - executor::{ExecutorContext, IdGenerator, KclValue, PlaneType, Sketch, SourceRange}, + executor::{ExecutorContext, IdGenerator, KclValue, PlaneType, SourceRange}, }; use kittycad_modeling_cmds::{each_cmd as mcmd, length_unit::LengthUnit, shared::Point3d, ModelingCmd}; use pretty_assertions::assert_eq; @@ -18,12 +18,9 @@ async fn setup(code: &str, name: &str) -> Result<(ExecutorContext, Node // We need to get the sketch ID. // Get the sketch ID from memory. - let KclValue::UserVal(user_val) = exec_state.memory.get(name, SourceRange::default()).unwrap() else { + let KclValue::Sketch { value: sketch } = exec_state.memory.get(name, SourceRange::default()).unwrap() else { anyhow::bail!("part001 not found in memory: {:?}", exec_state.memory); }; - let Some((sketch, _meta)) = user_val.get::() else { - anyhow::bail!("part001 was not a Sketch"); - }; let sketch_id = sketch.id; let plane_id = uuid::Uuid::new_v4();