Compare commits
	
		
			1 Commits
		
	
	
		
			updater-te
			...
			hotfix-v0.
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 76f1d09abb | 
| @ -8,8 +8,8 @@ import { UnitLength_type } from '@kittycad/lib/dist/types/src/models' | |||||||
|  |  | ||||||
| test.describe('Testing in-app sample loading', () => { | test.describe('Testing in-app sample loading', () => { | ||||||
|   /** |   /** | ||||||
|    * Note this test implicitly depends on the KCL sample "car-wheel.kcl", |    * Note this test implicitly depends on the KCL sample "a-parametric-bearing-pillow-block", | ||||||
|    * its title, and its units settings. https://github.com/KittyCAD/kcl-samples/blob/main/car-wheel/car-wheel.kcl |    * its title, and its units settings. https://github.com/KittyCAD/kcl-samples/blob/main/a-parametric-bearing-pillow-block/main.kcl | ||||||
|    */ |    */ | ||||||
|   test('Web: should overwrite current code, cannot create new file', async ({ |   test('Web: should overwrite current code, cannot create new file', async ({ | ||||||
|     editor, |     editor, | ||||||
| @ -29,8 +29,8 @@ test.describe('Testing in-app sample loading', () => { | |||||||
|  |  | ||||||
|     // Locators and constants |     // Locators and constants | ||||||
|     const newSample = { |     const newSample = { | ||||||
|       file: 'car-wheel' + FILE_EXT, |       file: 'a-parametric-bearing-pillow-block' + FILE_EXT, | ||||||
|       title: 'Car Wheel', |       title: 'A Parametric Bearing Pillow Block', | ||||||
|     } |     } | ||||||
|     const commandBarButton = page.getByRole('button', { name: 'Commands' }) |     const commandBarButton = page.getByRole('button', { name: 'Commands' }) | ||||||
|     const samplesCommandOption = page.getByRole('option', { |     const samplesCommandOption = page.getByRole('option', { | ||||||
| @ -75,8 +75,8 @@ test.describe('Testing in-app sample loading', () => { | |||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Note this test implicitly depends on the KCL samples: |    * Note this test implicitly depends on the KCL samples: | ||||||
|    * "car-wheel.kcl": https://github.com/KittyCAD/kcl-samples/blob/main/car-wheel/car-wheel.kcl |    * "a-parametric-bearing-pillow-block": https://github.com/KittyCAD/kcl-samples/blob/main/a-parametric-bearing-pillow-block/main.kcl | ||||||
|    * "gear-rack.kcl": https://github.com/KittyCAD/kcl-samples/blob/main/gear-rack/gear-rack.kcl |    * "gear-rack": https://github.com/KittyCAD/kcl-samples/blob/main/gear-rack/main.kcl | ||||||
|    */ |    */ | ||||||
|   test( |   test( | ||||||
|     'Desktop: should create new file by default, optionally overwrite', |     'Desktop: should create new file by default, optionally overwrite', | ||||||
| @ -93,8 +93,8 @@ test.describe('Testing in-app sample loading', () => { | |||||||
|  |  | ||||||
|       // Locators and constants |       // Locators and constants | ||||||
|       const sampleOne = { |       const sampleOne = { | ||||||
|         file: 'car-wheel' + FILE_EXT, |         file: 'a-parametric-bearing-pillow-block' + FILE_EXT, | ||||||
|         title: 'Car Wheel', |         title: 'A Parametric Bearing Pillow Block', | ||||||
|       } |       } | ||||||
|       const sampleTwo = { |       const sampleTwo = { | ||||||
|         file: 'gear-rack' + FILE_EXT, |         file: 'gear-rack' + FILE_EXT, | ||||||
|  | |||||||
| @ -1,166 +1,211 @@ | |||||||
| [ | [ | ||||||
|   { |   { | ||||||
|     "file": "80-20-rail.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "80-20-rail/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "80/20 Rail", |     "title": "80/20 Rail", | ||||||
|     "description": "An 80/20 extruded aluminum linear rail. T-slot profile adjustable by profile height, rail length, and origin position" |     "description": "An 80/20 extruded aluminum linear rail. T-slot profile adjustable by profile height, rail length, and origin position" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "a-parametric-bearing-pillow-block.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "a-parametric-bearing-pillow-block/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "A Parametric Bearing Pillow Block", |     "title": "A Parametric Bearing Pillow Block", | ||||||
|     "description": "A bearing pillow block, also known as a plummer block or pillow block bearing, is a pedestal used to provide support for a rotating shaft with the help of compatible bearings and various accessories. Housing a bearing, the pillow block provides a secure and stable foundation that allows the shaft to rotate smoothly within its machinery setup. These components are essential in a wide range of mechanical systems and machinery, playing a key role in reducing friction and supporting radial and axial loads." |     "description": "A bearing pillow block, also known as a plummer block or pillow block bearing, is a pedestal used to provide support for a rotating shaft with the help of compatible bearings and various accessories. Housing a bearing, the pillow block provides a secure and stable foundation that allows the shaft to rotate smoothly within its machinery setup. These components are essential in a wide range of mechanical systems and machinery, playing a key role in reducing friction and supporting radial and axial loads." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "ball-bearing.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "ball-bearing/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Ball Bearing", |     "title": "Ball Bearing", | ||||||
|     "description": "A ball bearing is a type of rolling-element bearing that uses balls to maintain the separation between the bearing races. The primary purpose of a ball bearing is to reduce rotational friction and support radial and axial loads." |     "description": "A ball bearing is a type of rolling-element bearing that uses balls to maintain the separation between the bearing races. The primary purpose of a ball bearing is to reduce rotational friction and support radial and axial loads." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "bracket.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "bracket/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Shelf Bracket", |     "title": "Shelf Bracket", | ||||||
|     "description": "This is a bracket that holds a shelf. It is made of aluminum and is designed to hold a force of 300 lbs. The bracket is 6 inches wide and the force is applied at the end of the shelf, 12 inches from the wall. The bracket has a factor of safety of 1.2. The legs of the bracket are 5 inches and 2 inches long. The thickness of the bracket is calculated from the constraints provided." |     "description": "This is a bracket that holds a shelf. It is made of aluminum and is designed to hold a force of 300 lbs. The bracket is 6 inches wide and the force is applied at the end of the shelf, 12 inches from the wall. The bracket has a factor of safety of 1.2. The legs of the bracket are 5 inches and 2 inches long. The thickness of the bracket is calculated from the constraints provided." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "brake-caliper.kcl", |     "file": "main.kcl", | ||||||
|     "title": "Brake Caliper", |     "pathFromProjectDirectoryToFirstFile": "car-wheel-assembly/main.kcl", | ||||||
|     "description": "Brake calipers are used to squeeze the brake pads against the rotor, causing larger and larger amounts of friction depending on how hard the brakes are pressed." |     "multipleFiles": true, | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "file": "car-wheel.kcl", |  | ||||||
|     "title": "Car Wheel", |  | ||||||
|     "description": "A sports car wheel with a circular lug pattern and spokes." |  | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "file": "car-wheel-assembly.kcl", |  | ||||||
|     "title": "Car Wheel Assembly", |     "title": "Car Wheel Assembly", | ||||||
|     "description": "A car wheel assembly with a rotor, tire, and lug nuts." |     "description": "A car wheel assembly with a rotor, tire, and lug nuts." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "dodecahedron.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "dodecahedron/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Hollow Dodecahedron", |     "title": "Hollow Dodecahedron", | ||||||
|     "description": "A regular dodecahedron or pentagonal dodecahedron is a dodecahedron composed of regular pentagonal faces, three meeting at each vertex. This example shows constructing the individual faces of the dodecahedron and extruding inwards." |     "description": "A regular dodecahedron or pentagonal dodecahedron is a dodecahedron composed of regular pentagonal faces, three meeting at each vertex. This example shows constructing the individual faces of the dodecahedron and extruding inwards." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "enclosure.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "enclosure/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Enclosure", |     "title": "Enclosure", | ||||||
|     "description": "An enclosure body and sealing lid for storing items" |     "description": "An enclosure body and sealing lid for storing items" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "flange-with-patterns.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "flange-with-patterns/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Flange", |     "title": "Flange", | ||||||
|     "description": "A flange is a flat rim, collar, or rib, typically forged or cast, that is used to strengthen an object, guide it, or attach it to another object. Flanges are known for their use in various applications, including piping, plumbing, and mechanical engineering, among others." |     "description": "A flange is a flat rim, collar, or rib, typically forged or cast, that is used to strengthen an object, guide it, or attach it to another object. Flanges are known for their use in various applications, including piping, plumbing, and mechanical engineering, among others." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "flange-xy.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "flange-xy/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Flange with XY coordinates", |     "title": "Flange with XY coordinates", | ||||||
|     "description": "A flange is a flat rim, collar, or rib, typically forged or cast, that is used to strengthen an object, guide it, or attach it to another object. Flanges are known for their use in various applications, including piping, plumbing, and mechanical engineering, among others." |     "description": "A flange is a flat rim, collar, or rib, typically forged or cast, that is used to strengthen an object, guide it, or attach it to another object. Flanges are known for their use in various applications, including piping, plumbing, and mechanical engineering, among others." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "focusrite-scarlett-mounting-bracket.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "focusrite-scarlett-mounting-bracket/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "A mounting bracket for the Focusrite Scarlett Solo audio interface", |     "title": "A mounting bracket for the Focusrite Scarlett Solo audio interface", | ||||||
|     "description": "This is a bracket that holds an audio device underneath a desk or shelf. The audio device has dimensions of 144mm wide, 80mm length and 45mm depth with fillets of 6mm. This mounting bracket is designed to be 3D printed with PLA material" |     "description": "This is a bracket that holds an audio device underneath a desk or shelf. The audio device has dimensions of 144mm wide, 80mm length and 45mm depth with fillets of 6mm. This mounting bracket is designed to be 3D printed with PLA material" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "food-service-spatula.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "food-service-spatula/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Food Service Spatula", |     "title": "Food Service Spatula", | ||||||
|     "description": "Use these spatulas for mixing, flipping, and scraping." |     "description": "Use these spatulas for mixing, flipping, and scraping." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "french-press.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "french-press/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "French Press", |     "title": "French Press", | ||||||
|     "description": "A french press immersion coffee maker" |     "description": "A french press immersion coffee maker" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "gear.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "gear/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Spur Gear", |     "title": "Spur Gear", | ||||||
|     "description": "A rotating machine part having cut teeth or, in the case of a cogwheel, inserted teeth (called cogs), which mesh with another toothed part to transmit torque. Geared devices can change the speed, torque, and direction of a power source. The two elements that define a gear are its circular shape and the teeth that are integrated into its outer edge, which are designed to fit into the teeth of another gear." |     "description": "A rotating machine part having cut teeth or, in the case of a cogwheel, inserted teeth (called cogs), which mesh with another toothed part to transmit torque. Geared devices can change the speed, torque, and direction of a power source. The two elements that define a gear are its circular shape and the teeth that are integrated into its outer edge, which are designed to fit into the teeth of another gear." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "gear-rack.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "gear-rack/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "100mm Gear Rack", |     "title": "100mm Gear Rack", | ||||||
|     "description": "A flat bar or rail that is engraved with teeth along its length. These teeth are designed to mesh with the teeth of a gear, known as a pinion. When the pinion, a small cylindrical gear, rotates, its teeth engage with the teeth on the rack, causing the rack to move linearly. Conversely, linear motion applied to the rack will cause the pinion to rotate." |     "description": "A flat bar or rail that is engraved with teeth along its length. These teeth are designed to mesh with the teeth of a gear, known as a pinion. When the pinion, a small cylindrical gear, rotates, its teeth engage with the teeth on the rack, causing the rack to move linearly. Conversely, linear motion applied to the rack will cause the pinion to rotate." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "hex-nut.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "hex-nut/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Hex nut", |     "title": "Hex nut", | ||||||
|     "description": "A hex nut is a type of fastener with a threaded hole and a hexagonal outer shape, used in a wide variety of applications to secure parts together. The hexagonal shape allows for a greater torque to be applied with wrenches or tools, making it one of the most common nut types in hardware." |     "description": "A hex nut is a type of fastener with a threaded hole and a hexagonal outer shape, used in a wide variety of applications to secure parts together. The hexagonal shape allows for a greater torque to be applied with wrenches or tools, making it one of the most common nut types in hardware." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "i-beam.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "i-beam/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "I-beam", |     "title": "I-beam", | ||||||
|     "description": "A structural metal beam with an I shaped cross section. Often used in construction" |     "description": "A structural metal beam with an I shaped cross section. Often used in construction" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "kitt.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "kitt/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Kitt", |     "title": "Kitt", | ||||||
|     "description": "The beloved KittyCAD mascot in a voxelized style." |     "description": "The beloved KittyCAD mascot in a voxelized style." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "lego.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "lego/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Lego Brick", |     "title": "Lego Brick", | ||||||
|     "description": "A standard Lego brick. This is a small, plastic construction block toy that can be interlocked with other blocks to build various structures, models, and figures. There are a lot of hacks used in this code." |     "description": "A standard Lego brick. This is a small, plastic construction block toy that can be interlocked with other blocks to build various structures, models, and figures. There are a lot of hacks used in this code." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "lug-nut.kcl", |     "file": "main.kcl", | ||||||
|     "title": "Lug Nut", |     "pathFromProjectDirectoryToFirstFile": "mounting-plate/main.kcl", | ||||||
|     "description": "lug Nuts are essential components used to create secure connections, whether for electrical purposes, like terminating wires or grounding, or for mechanical purposes, such as providing mounting points or reinforcing structural joints." |     "multipleFiles": false, | ||||||
|   }, |  | ||||||
|   { |  | ||||||
|     "file": "mounting-plate.kcl", |  | ||||||
|     "title": "Mounting Plate", |     "title": "Mounting Plate", | ||||||
|     "description": "A flat piece of material, often metal or plastic, that serves as a support or base for attaching, securing, or mounting various types of equipment, devices, or components." |     "description": "A flat piece of material, often metal or plastic, that serves as a support or base for attaching, securing, or mounting various types of equipment, devices, or components." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "multi-axis-robot.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "multi-axis-robot/main.kcl", | ||||||
|  |     "multipleFiles": true, | ||||||
|     "title": "Robot Arm", |     "title": "Robot Arm", | ||||||
|     "description": "A 4 axis robotic arm for industrial use. These machines can be used for assembly, packaging, organization of goods, and quality inspection processes" |     "description": "A 4 axis robotic arm for industrial use. These machines can be used for assembly, packaging, organization of goods, and quality inspection processes" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "pipe.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "pipe/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Pipe", |     "title": "Pipe", | ||||||
|     "description": "A tubular section or hollow cylinder, usually but not necessarily of circular cross-section, used mainly to convey substances that can flow." |     "description": "A tubular section or hollow cylinder, usually but not necessarily of circular cross-section, used mainly to convey substances that can flow." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "pipe-flange-assembly.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "pipe-flange-assembly/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Pipe and Flange Assembly", |     "title": "Pipe and Flange Assembly", | ||||||
|     "description": "A crucial component in various piping systems, designed to facilitate the connection, disconnection, and access to piping for inspection, cleaning, and modifications. This assembly combines pipes (long cylindrical conduits) with flanges (plate-like fittings) to create a secure yet detachable joint." |     "description": "A crucial component in various piping systems, designed to facilitate the connection, disconnection, and access to piping for inspection, cleaning, and modifications. This assembly combines pipes (long cylindrical conduits) with flanges (plate-like fittings) to create a secure yet detachable joint." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "pipe-with-bend.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "pipe-with-bend/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Pipe with bend", |     "title": "Pipe with bend", | ||||||
|     "description": "A tubular section or hollow cylinder, usually but not necessarily of circular cross-section, used mainly to convey substances that can flow." |     "description": "A tubular section or hollow cylinder, usually but not necessarily of circular cross-section, used mainly to convey substances that can flow." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "poopy-shoe.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "poopy-shoe/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Poopy Shoe", |     "title": "Poopy Shoe", | ||||||
|     "description": "poop shute for bambu labs printer - optimized for printing." |     "description": "poop shute for bambu labs printer - optimized for printing." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "router-template-cross-bar.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "router-template-cross-bar/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Router template for a cross bar", |     "title": "Router template for a cross bar", | ||||||
|     "description": "A guide for routing a notch into a cross bar." |     "description": "A guide for routing a notch into a cross bar." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "router-template-slate.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "router-template-slate/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Router template for a slate", |     "title": "Router template for a slate", | ||||||
|     "description": "A guide for routing a slate for a cross bar." |     "description": "A guide for routing a slate for a cross bar." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "sheet-metal-bracket.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "sheet-metal-bracket/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Sheet Metal Bracket", |     "title": "Sheet Metal Bracket", | ||||||
|     "description": "A component typically made from flat sheet metal through various manufacturing processes such as bending, punching, cutting, and forming. These brackets are used to support, attach, or mount other hardware components, often providing a structural or functional base for assembly." |     "description": "A component typically made from flat sheet metal through various manufacturing processes such as bending, punching, cutting, and forming. These brackets are used to support, attach, or mount other hardware components, often providing a structural or functional base for assembly." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "socket-head-cap-screw.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "socket-head-cap-screw/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Socket Head Cap Screw", |     "title": "Socket Head Cap Screw", | ||||||
|     "description": "This is for a #10-24 screw that is 1.00 inches long. A socket head cap screw is a type of fastener that is widely used in a variety of applications requiring a high strength fastening solution. It is characterized by its cylindrical head and internal hexagonal drive, which allows for tightening with an Allen wrench or hex key." |     "description": "This is for a #10-24 screw that is 1.00 inches long. A socket head cap screw is a type of fastener that is widely used in a variety of applications requiring a high strength fastening solution. It is characterized by its cylindrical head and internal hexagonal drive, which allows for tightening with an Allen wrench or hex key." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "tire.kcl", |     "file": "main.kcl", | ||||||
|     "title": "Tire", |     "pathFromProjectDirectoryToFirstFile": "walkie-talkie/main.kcl", | ||||||
|     "description": "A tire is a critical component of a vehicle that provides the necessary traction and grip between the car and the road. It supports the vehicle's weight and absorbs shocks from road irregularities." |     "multipleFiles": true, | ||||||
|  |     "title": "Walkie Talkie", | ||||||
|  |     "description": "A portable, handheld two-way radio device that allows users to communicate wirelessly over short to medium distances. It operates on specific radio frequencies and features a push-to-talk button for transmitting messages, making it ideal for quick and reliable communication in outdoor, work, or emergency settings." | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "file": "washer.kcl", |     "file": "main.kcl", | ||||||
|  |     "pathFromProjectDirectoryToFirstFile": "washer/main.kcl", | ||||||
|  |     "multipleFiles": false, | ||||||
|     "title": "Washer", |     "title": "Washer", | ||||||
|     "description": "A small, typically disk-shaped component with a hole in the middle, used in a wide range of applications, primarily in conjunction with fasteners like bolts and screws. Washers distribute the load of a fastener across a broader area. This is especially important when the fastening surface is soft or uneven, as it helps to prevent damage to the surface and ensures the load is evenly distributed, reducing the risk of the fastener becoming loose over time." |     "description": "A small, typically disk-shaped component with a hole in the middle, used in a wide range of applications, primarily in conjunction with fasteners like bolts and screws. Washers distribute the load of a fastener across a broader area. This is especially important when the fastening surface is soft or uneven, as it helps to prevent damage to the surface and ensures the load is evenly distributed, reducing the risk of the fastener becoming loose over time." | ||||||
|   }, |   }, | ||||||
|  | |||||||
| @ -57,7 +57,9 @@ export const FileMachineProvider = ({ | |||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     markOnce('code/didLoadFile') |     markOnce('code/didLoadFile') | ||||||
|     async function fetchKclSamples() { |     async function fetchKclSamples() { | ||||||
|       setKclSamples(await getKclSamplesManifest()) |       const manifest = await getKclSamplesManifest() | ||||||
|  |       const filteredFiles = manifest.filter((file) => !file.multipleFiles) | ||||||
|  |       setKclSamples(filteredFiles) | ||||||
|     } |     } | ||||||
|     fetchKclSamples().catch(reportError) |     fetchKclSamples().catch(reportError) | ||||||
|   }, []) |   }, []) | ||||||
| @ -324,7 +326,7 @@ export const FileMachineProvider = ({ | |||||||
|           } |           } | ||||||
|         }, |         }, | ||||||
|         kclSamples.map((sample) => ({ |         kclSamples.map((sample) => ({ | ||||||
|           value: sample.file, |           value: sample.pathFromProjectDirectoryToFirstFile, | ||||||
|           name: sample.title, |           name: sample.title, | ||||||
|         })) |         })) | ||||||
|       ).filter( |       ).filter( | ||||||
|  | |||||||
| @ -1,79 +1,81 @@ | |||||||
| import { assertParse, initPromise, programMemoryInit } from './wasm' | import { assertParse, initPromise, programMemoryInit } from './wasm' | ||||||
| import { enginelessExecutor } from '../lib/testHelpers' | import { enginelessExecutor } from '../lib/testHelpers' | ||||||
| // These unit tests makes web requests to a public github repository. |  | ||||||
|  | import path from 'node:path' | ||||||
|  | import fs from 'node:fs/promises' | ||||||
|  | import child_process from 'node:child_process' | ||||||
|  |  | ||||||
|  | // The purpose of these tests is to act as a first line of defense | ||||||
|  | // if something gets real screwy with our KCL ecosystem. | ||||||
|  | // THESE TESTS ONLY RUN UNDER A NODEJS ENVIRONMENT. They DO NOT | ||||||
|  | // test under our application. | ||||||
|  |  | ||||||
|  | const DIR_KCL_SAMPLES = 'kcl-samples' | ||||||
|  | const URL_GIT_KCL_SAMPLES = 'https://github.com/KittyCAD/kcl-samples.git' | ||||||
|  |  | ||||||
| interface KclSampleFile { | interface KclSampleFile { | ||||||
|   file: string |   file: string | ||||||
|  |   pathFromProjectDirectoryToFirstFile: string | ||||||
|   title: string |   title: string | ||||||
|   filename: string |   filename: string | ||||||
|   description: string |   description: string | ||||||
| } | } | ||||||
|  |  | ||||||
|  | try { | ||||||
|  |   // @ts-expect-error | ||||||
|  |   await fs.rm(DIR_KCL_SAMPLES, { recursive: true }) | ||||||
|  | } catch (e) { | ||||||
|  |   console.log(e) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | child_process.spawnSync('git', ['clone', URL_GIT_KCL_SAMPLES, DIR_KCL_SAMPLES]) | ||||||
|  |  | ||||||
|  | // @ts-expect-error | ||||||
|  | let files = await fs.readdir(DIR_KCL_SAMPLES) | ||||||
|  | // @ts-expect-error | ||||||
|  | const manifestJsonStr = await fs.readFile( | ||||||
|  |   path.resolve(DIR_KCL_SAMPLES, 'manifest.json'), | ||||||
|  |   'utf-8' | ||||||
|  | ) | ||||||
|  | const manifest = JSON.parse(manifestJsonStr) | ||||||
|  |  | ||||||
|  | process.chdir(DIR_KCL_SAMPLES) | ||||||
|  |  | ||||||
| beforeAll(async () => { | beforeAll(async () => { | ||||||
|   await initPromise |   await initPromise | ||||||
| }) | }) | ||||||
|  |  | ||||||
| // Only used to actually fetch an older version of KCL code that will break in the parser. | afterAll(async () => { | ||||||
| /* eslint-disable @typescript-eslint/no-unused-vars */ |   try { | ||||||
| async function getBrokenSampleCodeForLocalTesting() { |     process.chdir('..') | ||||||
|   const result = await fetch( |     await fs.rm(DIR_KCL_SAMPLES, { recursive: true }) | ||||||
|     'https://raw.githubusercontent.com/KittyCAD/kcl-samples/5ccd04a1773ebdbfd02684057917ce5dbe0eaab3/80-20-rail.kcl' |   } catch (e) {} | ||||||
|   ) | }) | ||||||
|   const text = await result.text() |  | ||||||
|   return text |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function getKclSampleCodeFromGithub(file: string): Promise<string> { | afterEach(() => { | ||||||
|   const result = await fetch( |   process.chdir('..') | ||||||
|     `https://raw.githubusercontent.com/KittyCAD/kcl-samples/refs/heads/main/${file}/${file}.kcl` | }) | ||||||
|   ) |  | ||||||
|   const text = await result.text() |  | ||||||
|   return text |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function getFileNamesFromManifestJSON(): Promise<KclSampleFile[]> { | // The tests have to be sequential because we need to change directories | ||||||
|   const result = await fetch( | // to support `import` working properly. | ||||||
|     'https://raw.githubusercontent.com/KittyCAD/kcl-samples/refs/heads/main/manifest.json' | // @ts-expect-error | ||||||
|   ) | describe.sequential('Test KCL Samples from public Github repository', () => { | ||||||
|   const json = await result.json() |   // @ts-expect-error | ||||||
|   json.forEach((file: KclSampleFile) => { |   describe.sequential('when performing enginelessExecutor', () => { | ||||||
|     const filenameWithoutExtension = file.file.split('.')[0] |     manifest.forEach((file: KclSampleFile) => { | ||||||
|     file.filename = filenameWithoutExtension |       // @ts-expect-error | ||||||
|   }) |       it.sequential( | ||||||
|   return json |         `should execute ${file.title} (${file.file}) successfully`, | ||||||
| } |         async () => { | ||||||
|  |           const [dirProject, fileKcl] = | ||||||
| // Value to use across all tests! |             file.pathFromProjectDirectoryToFirstFile.split('/') | ||||||
| let files: KclSampleFile[] = [] |           process.chdir(dirProject) | ||||||
|  |           const code = await fs.readFile(fileKcl, 'utf-8') | ||||||
| describe('Test KCL Samples from public Github repository', () => { |  | ||||||
|   describe('When parsing source code', () => { |  | ||||||
|     // THIS RUNS ACROSS OTHER TESTS! |  | ||||||
|     it('should fetch files', async () => { |  | ||||||
|       files = await getFileNamesFromManifestJSON() |  | ||||||
|     }) |  | ||||||
|     // 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 |  | ||||||
|     files.forEach((file: KclSampleFile) => { |  | ||||||
|       it(`should parse ${file.filename} without errors`, async () => { |  | ||||||
|         const code = await getKclSampleCodeFromGithub(file.filename) |  | ||||||
|         assertParse(code) |  | ||||||
|       }, 1000) |  | ||||||
|     }) |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   describe('when performing enginelessExecutor', () => { |  | ||||||
|     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 ast = assertParse(code) |           const ast = assertParse(code) | ||||||
|           await enginelessExecutor(ast, programMemoryInit()) |           await enginelessExecutor(ast, programMemoryInit()) | ||||||
|         } |         }, | ||||||
|       }, |         files.length * 1000 | ||||||
|       files.length * 1000 |       ) | ||||||
|     ) |     }) | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  | |||||||
| @ -1,5 +1,21 @@ | |||||||
| import { isDesktop } from 'lib/isDesktop' | import { isDesktop } from 'lib/isDesktop' | ||||||
|  |  | ||||||
|  | // Polyfill window.electron fs functions as needed when in a nodejs context | ||||||
|  | // (INTENDED FOR VITEST SHINANGANS.) | ||||||
|  | if (process.env.NODE_ENV === 'test' && process.env.VITEST) { | ||||||
|  |   const fs = require('node:fs/promises') | ||||||
|  |   const path = require('node:path') | ||||||
|  |   Object.assign(window, { | ||||||
|  |     electron: { | ||||||
|  |       readFile: fs.readFile, | ||||||
|  |       stat: fs.stat, | ||||||
|  |       readdir: fs.readdir, | ||||||
|  |       path, | ||||||
|  |       process: {}, | ||||||
|  |     }, | ||||||
|  |   }) | ||||||
|  | } | ||||||
|  |  | ||||||
| /// FileSystemManager is a class that provides a way to read files from the local file system. | /// FileSystemManager is a class that provides a way to read files from the local file system. | ||||||
| /// It assumes that you are in a project since it is solely used by the std lib | /// It assumes that you are in a project since it is solely used by the std lib | ||||||
| /// when executing code. | /// when executing code. | ||||||
| @ -19,13 +35,9 @@ class FileSystemManager { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   async readFile(path: string): Promise<Uint8Array> { |   async readFile(path: string): Promise<Uint8Array> { | ||||||
|     // Using local file system only works from desktop. |     // Using local file system only works from desktop and nodejs | ||||||
|     if (!isDesktop()) { |     if (!window?.electron?.readFile) { | ||||||
|       return Promise.reject( |       return Promise.reject(new Error('No polyfill found for this function')) | ||||||
|         new Error( |  | ||||||
|           'This function can only be called from the desktop application' |  | ||||||
|         ) |  | ||||||
|       ) |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return this.join(this.dir, path).then((filePath) => { |     return this.join(this.dir, path).then((filePath) => { | ||||||
| @ -35,12 +47,8 @@ class FileSystemManager { | |||||||
|  |  | ||||||
|   async exists(path: string): Promise<boolean | void> { |   async exists(path: string): Promise<boolean | void> { | ||||||
|     // Using local file system only works from desktop. |     // Using local file system only works from desktop. | ||||||
|     if (!isDesktop()) { |     if (!window?.electron?.stat) { | ||||||
|       return Promise.reject( |       return Promise.reject(new Error('No polyfill found for this function')) | ||||||
|         new Error( |  | ||||||
|           'This function can only be called from the desktop application' |  | ||||||
|         ) |  | ||||||
|       ) |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return this.join(this.dir, path).then(async (file) => { |     return this.join(this.dir, path).then(async (file) => { | ||||||
| @ -57,12 +65,8 @@ class FileSystemManager { | |||||||
|  |  | ||||||
|   async getAllFiles(path: string): Promise<string[] | void> { |   async getAllFiles(path: string): Promise<string[] | void> { | ||||||
|     // Using local file system only works from desktop. |     // Using local file system only works from desktop. | ||||||
|     if (!isDesktop()) { |     if (!window?.electron?.readdir) { | ||||||
|       return Promise.reject( |       return Promise.reject(new Error('No polyfill found for this function')) | ||||||
|         new Error( |  | ||||||
|           'This function can only be called from the desktop application' |  | ||||||
|         ) |  | ||||||
|       ) |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return this.join(this.dir, path).then((filepath) => { |     return this.join(this.dir, path).then((filepath) => { | ||||||
|  | |||||||
| @ -3,6 +3,8 @@ import { isDesktop } from './isDesktop' | |||||||
|  |  | ||||||
| export type KclSamplesManifestItem = { | export type KclSamplesManifestItem = { | ||||||
|   file: string |   file: string | ||||||
|  |   pathFromProjectDirectoryToFirstFile: string | ||||||
|  |   multipleFiles: boolean | ||||||
|   title: string |   title: string | ||||||
|   description: string |   description: string | ||||||
| } | } | ||||||
|  | |||||||
| @ -49,20 +49,30 @@ export function kclCommands( | |||||||
|         if (!data?.sample) { |         if (!data?.sample) { | ||||||
|           return |           return | ||||||
|         } |         } | ||||||
|  |         const pathParts = data.sample.split('/') | ||||||
|  |         const projectPathPart = pathParts[0] | ||||||
|  |         const primaryKclFile = pathParts[1] | ||||||
|         const sampleCodeUrl = `https://raw.githubusercontent.com/KittyCAD/kcl-samples/main/${encodeURIComponent( |         const sampleCodeUrl = `https://raw.githubusercontent.com/KittyCAD/kcl-samples/main/${encodeURIComponent( | ||||||
|           data.sample.replace(FILE_EXT, '') |           projectPathPart | ||||||
|         )}/${encodeURIComponent(data.sample)}` |         )}/${encodeURIComponent(primaryKclFile)}` | ||||||
|         const sampleSettingsFileUrl = `https://raw.githubusercontent.com/KittyCAD/kcl-samples/main/${encodeURIComponent( |         const sampleSettingsFileUrl = `https://raw.githubusercontent.com/KittyCAD/kcl-samples/main/${encodeURIComponent( | ||||||
|           data.sample.replace(FILE_EXT, '') |           projectPathPart | ||||||
|         )}/${PROJECT_SETTINGS_FILE_NAME}` |         )}/${PROJECT_SETTINGS_FILE_NAME}` | ||||||
|  |  | ||||||
|         Promise.all([fetch(sampleCodeUrl), fetch(sampleSettingsFileUrl)]) |         Promise.allSettled([fetch(sampleCodeUrl), fetch(sampleSettingsFileUrl)]) | ||||||
|  |           .then((results) => { | ||||||
|  |             const a = | ||||||
|  |               'value' in results[0] ? results[0].value : results[0].reason | ||||||
|  |             const b = | ||||||
|  |               'value' in results[1] ? results[1].value : results[1].reason | ||||||
|  |             return [a, b] | ||||||
|  |           }) | ||||||
|           .then( |           .then( | ||||||
|             async ([ |             async ([ | ||||||
|               codeResponse, |               codeResponse, | ||||||
|               settingsResponse, |               settingsResponse, | ||||||
|             ]): Promise<OnSubmitProps> => { |             ]): Promise<OnSubmitProps> => { | ||||||
|               if (!(codeResponse.ok && settingsResponse.ok)) { |               if (!codeResponse.ok) { | ||||||
|                 console.error( |                 console.error( | ||||||
|                   'Failed to fetch sample code:', |                   'Failed to fetch sample code:', | ||||||
|                   codeResponse.statusText |                   codeResponse.statusText | ||||||
| @ -70,20 +80,24 @@ export function kclCommands( | |||||||
|                 return Promise.reject(new Error('Failed to fetch sample code')) |                 return Promise.reject(new Error('Failed to fetch sample code')) | ||||||
|               } |               } | ||||||
|               const code = await codeResponse.text() |               const code = await codeResponse.text() | ||||||
|               const parsedProjectSettings = parseProjectSettings( |  | ||||||
|                 await settingsResponse.text() |               // It's possible that a sample doesn't have a project.toml | ||||||
|               ) |               // associated with it. | ||||||
|               let projectSettingsPayload: ReturnType< |               let projectSettingsPayload: ReturnType< | ||||||
|                 typeof projectConfigurationToSettingsPayload |                 typeof projectConfigurationToSettingsPayload | ||||||
|               > = {} |               > = {} | ||||||
|               if (!err(parsedProjectSettings)) { |               if (settingsResponse.ok) { | ||||||
|                 projectSettingsPayload = projectConfigurationToSettingsPayload( |                 const parsedProjectSettings = parseProjectSettings( | ||||||
|                   parsedProjectSettings |                   await settingsResponse.text() | ||||||
|                 ) |                 ) | ||||||
|  |                 if (!err(parsedProjectSettings)) { | ||||||
|  |                   projectSettingsPayload = | ||||||
|  |                     projectConfigurationToSettingsPayload(parsedProjectSettings) | ||||||
|  |                 } | ||||||
|               } |               } | ||||||
|  |  | ||||||
|               return { |               return { | ||||||
|                 sampleName: data.sample, |                 sampleName: data.sample.split('/')[0] + FILE_EXT, | ||||||
|                 code, |                 code, | ||||||
|                 method: data.method, |                 method: data.method, | ||||||
|                 sampleUnits: |                 sampleUnits: | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user
	