diff --git a/package-lock.json b/package-lock.json
index 3c48025c..b44d56cb 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -6612,6 +6612,10 @@
"@testing-library/dom": "9.3.1"
}
},
+ "node_modules/@mongodb-js/aggregation-stage-icons": {
+ "resolved": "packages/aggregation-stage-icons",
+ "link": true
+ },
"node_modules/@mongodb-js/compass-components": {
"version": "1.21.3",
"resolved": "https://registry.npmjs.org/@mongodb-js/compass-components/-/compass-components-1.21.3.tgz",
@@ -28302,6 +28306,49 @@
"url": "https://github.com/sponsors/colinhacks"
}
},
+ "packages/aggregation-stage-icons": {
+ "name": "@mongodb-js/aggregation-stage-icons",
+ "version": "0.1.0",
+ "license": "Apache-2.0",
+ "devDependencies": {
+ "@mongodb-js/eslint-config-devtools": "^0.11.7",
+ "@mongodb-js/mocha-config-devtools": "^1.1.2",
+ "@mongodb-js/mongodb-constants": "^0.25.0",
+ "@mongodb-js/prettier-config-devtools": "^1.0.3",
+ "@mongodb-js/tsconfig-devtools": "^1.1.2",
+ "@types/chai": "^4.2.21",
+ "@types/mocha": "^9.1.1",
+ "@types/node": "^22.15.30",
+ "@types/react": "^17.0.53",
+ "@types/react-dom": "^17.0.19",
+ "@types/sinon-chai": "^4.0.0",
+ "chai": "^4.5.0",
+ "depcheck": "^1.4.7",
+ "eslint": "^7.25.0 || ^8.0.0",
+ "gen-esm-wrapper": "^1.1.3",
+ "mocha": "^8.4.0",
+ "nyc": "^15.1.0",
+ "prettier": "^3.8.1",
+ "react": "^17.0.2",
+ "react-dom": "^17.0.2",
+ "sinon": "^9.2.3",
+ "typescript": "^5.9.3"
+ },
+ "peerDependencies": {
+ "react": "^17.0.2 || ^18 || ^19"
+ }
+ },
+ "packages/aggregation-stage-icons/node_modules/@types/sinon-chai": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/@types/sinon-chai/-/sinon-chai-4.0.0.tgz",
+ "integrity": "sha512-Uar+qk3TmeFsUWCwtqRNqNUE7vf34+MCJiQJR5M2rd4nCbhtE8RgTiHwN/mVwbfCjhmO6DiOel/MgzHkRMJJFg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/chai": "*",
+ "@types/sinon": "*"
+ }
+ },
"packages/device-id": {
"name": "@mongodb-js/device-id",
"version": "0.4.11",
@@ -35605,6 +35652,45 @@
"@testing-library/dom": "9.3.1"
}
},
+ "@mongodb-js/aggregation-stage-icons": {
+ "version": "file:packages/aggregation-stage-icons",
+ "requires": {
+ "@mongodb-js/eslint-config-devtools": "^0.11.7",
+ "@mongodb-js/mocha-config-devtools": "^1.1.2",
+ "@mongodb-js/mongodb-constants": "^0.25.0",
+ "@mongodb-js/prettier-config-devtools": "^1.0.3",
+ "@mongodb-js/tsconfig-devtools": "^1.1.2",
+ "@types/chai": "^4.2.21",
+ "@types/mocha": "^9.1.1",
+ "@types/node": "^22.15.30",
+ "@types/react": "^17.0.53",
+ "@types/react-dom": "^17.0.19",
+ "@types/sinon-chai": "^4.0.0",
+ "chai": "^4.5.0",
+ "depcheck": "^1.4.7",
+ "eslint": "^7.25.0 || ^8.0.0",
+ "gen-esm-wrapper": "^1.1.3",
+ "mocha": "^8.4.0",
+ "nyc": "^15.1.0",
+ "prettier": "^3.8.1",
+ "react": "^17.0.2",
+ "react-dom": "^17.0.2",
+ "sinon": "^9.2.3",
+ "typescript": "^5.9.3"
+ },
+ "dependencies": {
+ "@types/sinon-chai": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/@types/sinon-chai/-/sinon-chai-4.0.0.tgz",
+ "integrity": "sha512-Uar+qk3TmeFsUWCwtqRNqNUE7vf34+MCJiQJR5M2rd4nCbhtE8RgTiHwN/mVwbfCjhmO6DiOel/MgzHkRMJJFg==",
+ "dev": true,
+ "requires": {
+ "@types/chai": "*",
+ "@types/sinon": "*"
+ }
+ }
+ }
+ },
"@mongodb-js/compass-components": {
"version": "1.21.3",
"resolved": "https://registry.npmjs.org/@mongodb-js/compass-components/-/compass-components-1.21.3.tgz",
diff --git a/packages/aggregation-stage-icons/.depcheckrc b/packages/aggregation-stage-icons/.depcheckrc
new file mode 100644
index 00000000..3ee43e27
--- /dev/null
+++ b/packages/aggregation-stage-icons/.depcheckrc
@@ -0,0 +1,11 @@
+ignores:
+ - '@mongodb-js/prettier-config-devtools'
+ - '@mongodb-js/tsconfig-devtools'
+ - '@types/node'
+ - '@types/react'
+ - '@types/react-dom'
+ - '@types/chai'
+ - '@types/sinon-chai'
+ - 'sinon'
+ignore-patterns:
+ - 'dist'
diff --git a/packages/aggregation-stage-icons/.eslintignore b/packages/aggregation-stage-icons/.eslintignore
new file mode 100644
index 00000000..85a8a75e
--- /dev/null
+++ b/packages/aggregation-stage-icons/.eslintignore
@@ -0,0 +1,2 @@
+.nyc-output
+dist
diff --git a/packages/aggregation-stage-icons/.eslintrc.js b/packages/aggregation-stage-icons/.eslintrc.js
new file mode 100644
index 00000000..83296d73
--- /dev/null
+++ b/packages/aggregation-stage-icons/.eslintrc.js
@@ -0,0 +1,8 @@
+module.exports = {
+ root: true,
+ extends: ['@mongodb-js/eslint-config-devtools'],
+ parserOptions: {
+ tsconfigRootDir: __dirname,
+ project: ['./tsconfig-lint.json'],
+ },
+};
diff --git a/packages/aggregation-stage-icons/.mocharc.js b/packages/aggregation-stage-icons/.mocharc.js
new file mode 100644
index 00000000..64afeb1f
--- /dev/null
+++ b/packages/aggregation-stage-icons/.mocharc.js
@@ -0,0 +1 @@
+module.exports = require('@mongodb-js/mocha-config-devtools');
diff --git a/packages/aggregation-stage-icons/.prettierignore b/packages/aggregation-stage-icons/.prettierignore
new file mode 100644
index 00000000..4d28df66
--- /dev/null
+++ b/packages/aggregation-stage-icons/.prettierignore
@@ -0,0 +1,3 @@
+.nyc_output
+dist
+coverage
diff --git a/packages/aggregation-stage-icons/.prettierrc.json b/packages/aggregation-stage-icons/.prettierrc.json
new file mode 100644
index 00000000..dfae21d0
--- /dev/null
+++ b/packages/aggregation-stage-icons/.prettierrc.json
@@ -0,0 +1 @@
+"@mongodb-js/prettier-config-devtools"
diff --git a/packages/aggregation-stage-icons/LICENSE b/packages/aggregation-stage-icons/LICENSE
new file mode 100644
index 00000000..5e0fd33c
--- /dev/null
+++ b/packages/aggregation-stage-icons/LICENSE
@@ -0,0 +1,201 @@
+Apache License
+Version 2.0, January 2004
+http://www.apache.org/licenses/
+
+TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+1. Definitions.
+
+"License" shall mean the terms and conditions for use, reproduction,
+and distribution as defined by Sections 1 through 9 of this document.
+
+"Licensor" shall mean the copyright owner or entity authorized by
+the copyright owner that is granting the License.
+
+"Legal Entity" shall mean the union of the acting entity and all
+other entities that control, are controlled by, or are under common
+control with that entity. For the purposes of this definition,
+"control" means (i) the power, direct or indirect, to cause the
+direction or management of such entity, whether by contract or
+otherwise, or (ii) ownership of fifty percent (50%) or more of the
+outstanding shares, or (iii) beneficial ownership of such entity.
+
+"You" (or "Your") shall mean an individual or Legal Entity
+exercising permissions granted by this License.
+
+"Source" form shall mean the preferred form for making modifications,
+including but not limited to software source code, documentation
+source, and configuration files.
+
+"Object" form shall mean any form resulting from mechanical
+transformation or translation of a Source form, including but
+not limited to compiled object code, generated documentation,
+and conversions to other media types.
+
+"Work" shall mean the work of authorship, whether in Source or
+Object form, made available under the License, as indicated by a
+copyright notice that is included in or attached to the work
+(an example is provided in the Appendix below).
+
+"Derivative Works" shall mean any work, whether in Source or Object
+form, that is based on (or derived from) the Work and for which the
+editorial revisions, annotations, elaborations, or other modifications
+represent, as a whole, an original work of authorship. For the purposes
+of this License, Derivative Works shall not include works that remain
+separable from, or merely link (or bind by name) to the interfaces of,
+the Work and Derivative Works thereof.
+
+"Contribution" shall mean any work of authorship, including
+the original version of the Work and any modifications or additions
+to that Work or Derivative Works thereof, that is intentionally
+submitted to Licensor for inclusion in the Work by the copyright owner
+or by an individual or Legal Entity authorized to submit on behalf of
+the copyright owner. For the purposes of this definition, "submitted"
+means any form of electronic, verbal, or written communication sent
+to the Licensor or its representatives, including but not limited to
+communication on electronic mailing lists, source code control systems,
+and issue tracking systems that are managed by, or on behalf of, the
+Licensor for the purpose of discussing and improving the Work, but
+excluding communication that is conspicuously marked or otherwise
+designated in writing by the copyright owner as "Not a Contribution."
+
+"Contributor" shall mean Licensor and any individual or Legal Entity
+on behalf of whom a Contribution has been received by Licensor and
+subsequently incorporated within the Work.
+
+2. Grant of Copyright License. Subject to the terms and conditions of
+this License, each Contributor hereby grants to You a perpetual,
+worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+copyright license to reproduce, prepare Derivative Works of,
+publicly display, publicly perform, sublicense, and distribute the
+Work and such Derivative Works in Source or Object form.
+
+3. Grant of Patent License. Subject to the terms and conditions of
+this License, each Contributor hereby grants to You a perpetual,
+worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+(except as stated in this section) patent license to make, have made,
+use, offer to sell, sell, import, and otherwise transfer the Work,
+where such license applies only to those patent claims licensable
+by such Contributor that are necessarily infringed by their
+Contribution(s) alone or by combination of their Contribution(s)
+with the Work to which such Contribution(s) was submitted. If You
+institute patent litigation against any entity (including a
+cross-claim or counterclaim in a lawsuit) alleging that the Work
+or a Contribution incorporated within the Work constitutes direct
+or contributory patent infringement, then any patent licenses
+granted to You under this License for that Work shall terminate
+as of the date such litigation is filed.
+
+4. Redistribution. You may reproduce and distribute copies of the
+Work or Derivative Works thereof in any medium, with or without
+modifications, and in Source or Object form, provided that You
+meet the following conditions:
+
+(a) You must give any other recipients of the Work or
+Derivative Works a copy of this License; and
+
+(b) You must cause any modified files to carry prominent notices
+stating that You changed the files; and
+
+(c) You must retain, in the Source form of any Derivative Works
+that You distribute, all copyright, patent, trademark, and
+attribution notices from the Source form of the Work,
+excluding those notices that do not pertain to any part of
+the Derivative Works; and
+
+(d) If the Work includes a "NOTICE" text file as part of its
+distribution, then any Derivative Works that You distribute must
+include a readable copy of the attribution notices contained
+within such NOTICE file, excluding those notices that do not
+pertain to any part of the Derivative Works, in at least one
+of the following places: within a NOTICE text file distributed
+as part of the Derivative Works; within the Source form or
+documentation, if provided along with the Derivative Works; or,
+within a display generated by the Derivative Works, if and
+wherever such third-party notices normally appear. The contents
+of the NOTICE file are for informational purposes only and
+do not modify the License. You may add Your own attribution
+notices within Derivative Works that You distribute, alongside
+or as an addendum to the NOTICE text from the Work, provided
+that such additional attribution notices cannot be construed
+as modifying the License.
+
+You may add Your own copyright statement to Your modifications and
+may provide additional or different license terms and conditions
+for use, reproduction, or distribution of Your modifications, or
+for any such Derivative Works as a whole, provided Your use,
+reproduction, and distribution of the Work otherwise complies with
+the conditions stated in this License.
+
+5. Submission of Contributions. Unless You explicitly state otherwise,
+any Contribution intentionally submitted for inclusion in the Work
+by You to the Licensor shall be under the terms and conditions of
+this License, without any additional terms or conditions.
+Notwithstanding the above, nothing herein shall supersede or modify
+the terms of any separate license agreement you may have executed
+with Licensor regarding such Contributions.
+
+6. Trademarks. This License does not grant permission to use the trade
+names, trademarks, service marks, or product names of the Licensor,
+except as required for reasonable and customary use in describing the
+origin of the Work and reproducing the content of the NOTICE file.
+
+7. Disclaimer of Warranty. Unless required by applicable law or
+agreed to in writing, Licensor provides the Work (and each
+Contributor provides its Contributions) on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+implied, including, without limitation, any warranties or conditions
+of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+PARTICULAR PURPOSE. You are solely responsible for determining the
+appropriateness of using or redistributing the Work and assume any
+risks associated with Your exercise of permissions under this License.
+
+8. Limitation of Liability. In no event and under no legal theory,
+whether in tort (including negligence), contract, or otherwise,
+unless required by applicable law (such as deliberate and grossly
+negligent acts) or agreed to in writing, shall any Contributor be
+liable to You for damages, including any direct, indirect, special,
+incidental, or consequential damages of any character arising as a
+result of this License or out of the use or inability to use the
+Work (including but not limited to damages for loss of goodwill,
+work stoppage, computer failure or malfunction, or any and all
+other commercial damages or losses), even if such Contributor
+has been advised of the possibility of such damages.
+
+9. Accepting Warranty or Additional Liability. While redistributing
+the Work or Derivative Works thereof, You may choose to offer,
+and charge a fee for, acceptance of support, warranty, indemnity,
+or other liability obligations and/or rights consistent with this
+License. However, in accepting such obligations, You may act only
+on Your own behalf and on Your sole responsibility, not on behalf
+of any other Contributor, and only if You agree to indemnify,
+defend, and hold each Contributor harmless for any liability
+incurred by, or claims asserted against, such Contributor by reason
+of your accepting any such warranty or additional liability.
+
+END OF TERMS AND CONDITIONS
+
+APPENDIX: How to apply the Apache License to your work.
+
+To apply the Apache License to your work, attach the following
+boilerplate notice, with the fields enclosed by brackets "{}"
+replaced with your own identifying information. (Don't include
+the brackets!) The text should be enclosed in the appropriate
+comment syntax for the file format. We also recommend that a
+file or class name and description of purpose be included on the
+same "printed page" as the copyright notice for easier
+identification within third-party archives.
+
+Copyright {yyyy} {name of copyright owner}
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
diff --git a/packages/aggregation-stage-icons/README.md b/packages/aggregation-stage-icons/README.md
new file mode 100644
index 00000000..1c66c546
--- /dev/null
+++ b/packages/aggregation-stage-icons/README.md
@@ -0,0 +1,32 @@
+# @mongodb-js/aggregation-stage-icons
+
+Icon set for MongoDB aggregation pipeline stages.
+
+## Usage
+
+### React component
+
+```tsx
+import { StageIcon } from '@mongodb-js/aggregation-stage-icons';
+
+
+
+```
+
+### SVG paths
+
+`stageIconPaths` maps each stage name to its SVG path (d):
+
+```ts
+import {
+ stageIconPaths,
+ aggregationStageNames,
+} from '@mongodb-js/aggregation-stage-icons';
+
+stageIcons['$match']; // 'M18.5 6.25C18.791 6.25...'
+aggregationStageNames; // ['$addFields', '$bucket', ...]
+```
+
+## Adding or updating icons
+
+Drop the `.svg` file into [`src/icons/`](./src/icons) named after its stage (e.g. `$match.svg`), then run `npm run extract-icon-paths` to regenerate the typed icon data.
diff --git a/packages/aggregation-stage-icons/package.json b/packages/aggregation-stage-icons/package.json
new file mode 100644
index 00000000..64c321b8
--- /dev/null
+++ b/packages/aggregation-stage-icons/package.json
@@ -0,0 +1,83 @@
+{
+ "name": "@mongodb-js/aggregation-stage-icons",
+ "description": "MongoDB aggregation stage icon set",
+ "author": {
+ "name": "MongoDB Inc",
+ "email": "compass@mongodb.com"
+ },
+ "publishConfig": {
+ "access": "public"
+ },
+ "bugs": {
+ "url": "https://jira.mongodb.org/projects/COMPASS/issues",
+ "email": "compass@mongodb.com"
+ },
+ "homepage": "https://github.com/mongodb-js/devtools-shared",
+ "version": "0.1.0",
+ "repository": {
+ "type": "git",
+ "url": "https://github.com/mongodb-js/devtools-shared.git"
+ },
+ "files": [
+ "dist"
+ ],
+ "license": "Apache-2.0",
+ "main": "dist/index.js",
+ "types": "./dist/index.d.ts",
+ "exports": {
+ "require": {
+ "default": "./dist/index.js",
+ "types": "./dist/index.d.ts"
+ },
+ "import": {
+ "default": "./dist/.esm-wrapper.mjs",
+ "types": "./dist/index.d.ts"
+ }
+ },
+ "scripts": {
+ "bootstrap": "npm run compile",
+ "prepublishOnly": "npm run compile",
+ "extract-icon-paths": "ts-node scripts/extract-icon-paths.ts",
+ "extract-icon-paths:check": "ts-node scripts/extract-icon-paths.ts --check",
+ "compile": "npm run extract-icon-paths && tsc -p tsconfig.json && gen-esm-wrapper . ./dist/.esm-wrapper.mjs",
+ "typecheck": "tsc --noEmit",
+ "eslint": "eslint",
+ "prettier": "prettier",
+ "lint": "npm run eslint . && npm run prettier -- --check .",
+ "depcheck": "depcheck",
+ "check": "npm run extract-icon-paths:check && npm run typecheck && npm run lint && npm run depcheck",
+ "check-ci": "npm run check",
+ "test": "mocha",
+ "test-cov": "nyc -x \"**/*.spec.*\" --reporter=lcov --reporter=text --reporter=html npm run test",
+ "test-watch": "npm run test -- --watch",
+ "test-ci": "npm run test-cov",
+ "reformat": "npm run prettier -- --write ."
+ },
+ "peerDependencies": {
+ "react": "^17.0.2 || ^18 || ^19"
+ },
+ "devDependencies": {
+ "@mongodb-js/eslint-config-devtools": "^0.11.7",
+ "@mongodb-js/mocha-config-devtools": "^1.1.2",
+ "@mongodb-js/mongodb-constants": "^0.25.0",
+ "@mongodb-js/prettier-config-devtools": "^1.0.3",
+ "@mongodb-js/tsconfig-devtools": "^1.1.2",
+ "@types/chai": "^4.2.21",
+ "@types/mocha": "^9.1.1",
+ "@types/node": "^22.15.30",
+ "@types/react": "^17.0.53",
+ "@types/react-dom": "^17.0.19",
+ "@types/sinon-chai": "^4.0.0",
+ "chai": "^4.5.0",
+ "depcheck": "^1.4.7",
+ "eslint": "^7.25.0 || ^8.0.0",
+ "gen-esm-wrapper": "^1.1.3",
+ "mocha": "^8.4.0",
+ "nyc": "^15.1.0",
+ "prettier": "^3.8.1",
+ "react": "^17.0.2",
+ "react-dom": "^17.0.2",
+ "sinon": "^9.2.3",
+ "typescript": "^5.9.3"
+ }
+}
diff --git a/packages/aggregation-stage-icons/scripts/extract-icon-paths.ts b/packages/aggregation-stage-icons/scripts/extract-icon-paths.ts
new file mode 100644
index 00000000..3598d906
--- /dev/null
+++ b/packages/aggregation-stage-icons/scripts/extract-icon-paths.ts
@@ -0,0 +1,64 @@
+import { readFileSync, readdirSync, writeFileSync } from 'fs';
+import { join } from 'path';
+import { format, resolveConfig } from 'prettier';
+import { STAGE_OPERATOR_NAMES } from '@mongodb-js/mongodb-constants';
+
+const iconsDir = join(__dirname, '..', 'src', 'icons');
+const outFile = join(__dirname, '..', 'src', 'stage-icon-data.ts');
+
+const knownStages = new Set(STAGE_OPERATOR_NAMES);
+
+function buildSource(): string {
+ const entries = readdirSync(iconsDir)
+ .filter((file) => file.endsWith('.svg'))
+ .sort((a, b) => a.localeCompare(b))
+ .map((file) => {
+ const svg = readFileSync(join(iconsDir, file), 'utf8');
+ const paths = [...svg.matchAll(/]*\bd="([^"]+)"/g)];
+ if (paths.length !== 1) {
+ throw new Error(
+ `Expected exactly one in ${file}, found ${paths.length}`,
+ );
+ }
+ const name = file.slice(0, -'.svg'.length);
+ if (!knownStages.has(name)) {
+ throw new Error(
+ `${file} does not correspond to a known aggregation stage. ` +
+ `Its name must match one of the stage operators in @mongodb-js/mongodb-constants.`,
+ );
+ }
+ return ` ${JSON.stringify(name)}: ${JSON.stringify(paths[0][1])},`;
+ });
+
+ return (
+ `// Generated by scripts/extract-icon-paths.ts. Do not edit by hand.\n\n` +
+ `export const stageIconPaths = {\n${entries.join('\n')}\n} as const;\n`
+ );
+}
+
+async function main(): Promise {
+ const check = process.argv.includes('--check');
+
+ const config = await resolveConfig(outFile);
+ const formatted = await format(buildSource(), {
+ ...config,
+ parser: 'typescript',
+ });
+
+ if (check) {
+ if (readFileSync(outFile, 'utf8') !== formatted) {
+ throw new Error(
+ `${outFile} is out of date. Run \`npm run extract-icon-paths\` and commit the result.`,
+ );
+ }
+ return;
+ }
+
+ writeFileSync(outFile, formatted);
+}
+
+main().catch((error) => {
+ // eslint-disable-next-line no-console
+ console.error(error instanceof Error ? error.message : error);
+ process.exit(1);
+});
diff --git a/packages/aggregation-stage-icons/src/icons/$addFields.svg b/packages/aggregation-stage-icons/src/icons/$addFields.svg
new file mode 100644
index 00000000..2ce39ff4
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$addFields.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$bucket.svg b/packages/aggregation-stage-icons/src/icons/$bucket.svg
new file mode 100644
index 00000000..f0e945de
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$bucket.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$bucketAuto.svg b/packages/aggregation-stage-icons/src/icons/$bucketAuto.svg
new file mode 100644
index 00000000..a5bd5692
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$bucketAuto.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$collStats.svg b/packages/aggregation-stage-icons/src/icons/$collStats.svg
new file mode 100644
index 00000000..ff26c2d0
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$collStats.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$count.svg b/packages/aggregation-stage-icons/src/icons/$count.svg
new file mode 100644
index 00000000..efea0f90
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$count.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$densify.svg b/packages/aggregation-stage-icons/src/icons/$densify.svg
new file mode 100644
index 00000000..4f998722
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$densify.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$facet.svg b/packages/aggregation-stage-icons/src/icons/$facet.svg
new file mode 100644
index 00000000..221a60c5
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$facet.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$fill.svg b/packages/aggregation-stage-icons/src/icons/$fill.svg
new file mode 100644
index 00000000..f64e071c
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$fill.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$geoNear.svg b/packages/aggregation-stage-icons/src/icons/$geoNear.svg
new file mode 100644
index 00000000..5bb69e6d
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$geoNear.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$graphLookup.svg b/packages/aggregation-stage-icons/src/icons/$graphLookup.svg
new file mode 100644
index 00000000..1cc9f069
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$graphLookup.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$group.svg b/packages/aggregation-stage-icons/src/icons/$group.svg
new file mode 100644
index 00000000..ad5f29ed
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$group.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$indexStats.svg b/packages/aggregation-stage-icons/src/icons/$indexStats.svg
new file mode 100644
index 00000000..2f525ad8
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$indexStats.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$limit.svg b/packages/aggregation-stage-icons/src/icons/$limit.svg
new file mode 100644
index 00000000..9bcd0723
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$limit.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$lookup.svg b/packages/aggregation-stage-icons/src/icons/$lookup.svg
new file mode 100644
index 00000000..4deac431
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$lookup.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$match.svg b/packages/aggregation-stage-icons/src/icons/$match.svg
new file mode 100644
index 00000000..7b85e82c
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$match.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$merge.svg b/packages/aggregation-stage-icons/src/icons/$merge.svg
new file mode 100644
index 00000000..f8d92811
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$merge.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$out.svg b/packages/aggregation-stage-icons/src/icons/$out.svg
new file mode 100644
index 00000000..e32d5f19
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$out.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$project.svg b/packages/aggregation-stage-icons/src/icons/$project.svg
new file mode 100644
index 00000000..dbd83b8e
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$project.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$rankFusion.svg b/packages/aggregation-stage-icons/src/icons/$rankFusion.svg
new file mode 100644
index 00000000..9dcc3f7c
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$rankFusion.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$redact.svg b/packages/aggregation-stage-icons/src/icons/$redact.svg
new file mode 100644
index 00000000..32c8c0be
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$redact.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$replaceRoot.svg b/packages/aggregation-stage-icons/src/icons/$replaceRoot.svg
new file mode 100644
index 00000000..6c441eb9
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$replaceRoot.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$replaceWith.svg b/packages/aggregation-stage-icons/src/icons/$replaceWith.svg
new file mode 100644
index 00000000..11d16edb
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$replaceWith.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$sample.svg b/packages/aggregation-stage-icons/src/icons/$sample.svg
new file mode 100644
index 00000000..4865fc0a
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$sample.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$search.svg b/packages/aggregation-stage-icons/src/icons/$search.svg
new file mode 100644
index 00000000..3d80617d
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$search.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$searchMeta.svg b/packages/aggregation-stage-icons/src/icons/$searchMeta.svg
new file mode 100644
index 00000000..2a97f7e2
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$searchMeta.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$set.svg b/packages/aggregation-stage-icons/src/icons/$set.svg
new file mode 100644
index 00000000..dd22a95c
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$set.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$setWindowFields.svg b/packages/aggregation-stage-icons/src/icons/$setWindowFields.svg
new file mode 100644
index 00000000..3c97261d
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$setWindowFields.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$skip.svg b/packages/aggregation-stage-icons/src/icons/$skip.svg
new file mode 100644
index 00000000..8fd1f14a
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$skip.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$sort.svg b/packages/aggregation-stage-icons/src/icons/$sort.svg
new file mode 100644
index 00000000..122e0a77
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$sort.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$sortByCount.svg b/packages/aggregation-stage-icons/src/icons/$sortByCount.svg
new file mode 100644
index 00000000..869ca1f9
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$sortByCount.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$unionWith.svg b/packages/aggregation-stage-icons/src/icons/$unionWith.svg
new file mode 100644
index 00000000..abf83fbc
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$unionWith.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$unset.svg b/packages/aggregation-stage-icons/src/icons/$unset.svg
new file mode 100644
index 00000000..c9969b64
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$unset.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$unwind.svg b/packages/aggregation-stage-icons/src/icons/$unwind.svg
new file mode 100644
index 00000000..6f4f247a
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$unwind.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/icons/$vectorSearch.svg b/packages/aggregation-stage-icons/src/icons/$vectorSearch.svg
new file mode 100644
index 00000000..f4702328
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/icons/$vectorSearch.svg
@@ -0,0 +1,3 @@
+
diff --git a/packages/aggregation-stage-icons/src/index.spec.tsx b/packages/aggregation-stage-icons/src/index.spec.tsx
new file mode 100644
index 00000000..e0cd5c03
--- /dev/null
+++ b/packages/aggregation-stage-icons/src/index.spec.tsx
@@ -0,0 +1,46 @@
+import React from 'react';
+import { renderToStaticMarkup } from 'react-dom/server';
+import { expect } from 'chai';
+import { StageIcon, stageIcons, aggregationStageNames } from './';
+
+describe('aggregation-stage-icons', function () {
+ it('exposes a well-formed svg string for every stage', function () {
+ expect(aggregationStageNames).to.include('$match');
+ for (const name of aggregationStageNames) {
+ expect(stageIcons[name]).to.match(/^