stackscan 0.1.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +222 -0
- package/dist/add.d.mts +3 -0
- package/dist/add.d.ts +3 -0
- package/dist/add.js +83 -0
- package/dist/add.mjs +7 -0
- package/dist/batch-import.d.mts +6 -0
- package/dist/batch-import.d.ts +6 -0
- package/dist/batch-import.js +5724 -0
- package/dist/batch-import.mjs +12 -0
- package/dist/chunk-22J2UHU7.mjs +1980 -0
- package/dist/chunk-2KG7JDHY.mjs +19 -0
- package/dist/chunk-2O2NMD7M.mjs +127 -0
- package/dist/chunk-3HEYJTUE.mjs +157 -0
- package/dist/chunk-3LPZV7KA.mjs +15 -0
- package/dist/chunk-3O2C34CJ.mjs +174 -0
- package/dist/chunk-3R4TXOZA.mjs +179 -0
- package/dist/chunk-3RJHZIII.mjs +174 -0
- package/dist/chunk-3XKWCBY6.mjs +105 -0
- package/dist/chunk-4OMBB3R6.mjs +52 -0
- package/dist/chunk-4THN5TT4.mjs +185 -0
- package/dist/chunk-4WCVZS6B.mjs +178 -0
- package/dist/chunk-5MNAT56X.mjs +23 -0
- package/dist/chunk-5VH7N3X6.mjs +109 -0
- package/dist/chunk-6A4WEZDZ.mjs +14 -0
- package/dist/chunk-6MIRBLO4.mjs +15 -0
- package/dist/chunk-6RCTNUF4.mjs +17 -0
- package/dist/chunk-7F4KWGMM.mjs +45 -0
- package/dist/chunk-7KF2HEU2.mjs +56 -0
- package/dist/chunk-7KONWCSX.mjs +15 -0
- package/dist/chunk-7TIXAYVC.mjs +10122 -0
- package/dist/chunk-ADMVFPX7.mjs +150 -0
- package/dist/chunk-AFJCX5SK.mjs +22 -0
- package/dist/chunk-ATNEVZPV.mjs +25 -0
- package/dist/chunk-BBP27ZGZ.mjs +4841 -0
- package/dist/chunk-BGT3PANW.mjs +64 -0
- package/dist/chunk-BJIQZNKU.mjs +184 -0
- package/dist/chunk-BKQW4ZNZ.mjs +19 -0
- package/dist/chunk-BMQTYZOD.mjs +144 -0
- package/dist/chunk-CZADURZD.mjs +674 -0
- package/dist/chunk-D2DOGM2Y.mjs +19 -0
- package/dist/chunk-D32C3FU6.mjs +45 -0
- package/dist/chunk-D6FFDX4P.mjs +158 -0
- package/dist/chunk-DUAFJVBN.mjs +44 -0
- package/dist/chunk-E24PAHCZ.mjs +146 -0
- package/dist/chunk-E3OD2BXY.mjs +14 -0
- package/dist/chunk-E4K5XYFI.mjs +210767 -0
- package/dist/chunk-EH2SEQZP.mjs +10119 -0
- package/dist/chunk-EH4FWTP5.mjs +21 -0
- package/dist/chunk-EOKQCSHI.mjs +10 -0
- package/dist/chunk-EQ4WRCPP.mjs +16 -0
- package/dist/chunk-EYXTSAAW.mjs +64 -0
- package/dist/chunk-F6Z4WXOF.mjs +146 -0
- package/dist/chunk-FM5BCBKT.mjs +77 -0
- package/dist/chunk-GXKIDBAX.mjs +146 -0
- package/dist/chunk-H6HA3IZE.mjs +1980 -0
- package/dist/chunk-H7WNQ3SS.mjs +23 -0
- package/dist/chunk-HBJEQ63D.mjs +205 -0
- package/dist/chunk-HTTDBNRT.mjs +146 -0
- package/dist/chunk-HYUKJUPF.mjs +177 -0
- package/dist/chunk-IFEEO473.mjs +49 -0
- package/dist/chunk-IILZ43EE.mjs +25 -0
- package/dist/chunk-ITGRHGUT.mjs +158 -0
- package/dist/chunk-J5JVDMED.mjs +45 -0
- package/dist/chunk-JH3SW4KH.mjs +4896 -0
- package/dist/chunk-JHVWSHNY.mjs +32 -0
- package/dist/chunk-JLFOQXQG.mjs +26 -0
- package/dist/chunk-K4BNSVJP.mjs +16 -0
- package/dist/chunk-KCDUZXVN.mjs +157 -0
- package/dist/chunk-KMDEPBKS.mjs +685 -0
- package/dist/chunk-L4AQWDD2.mjs +92 -0
- package/dist/chunk-L7I4UC35.mjs +158 -0
- package/dist/chunk-LB3L25FS.mjs +1980 -0
- package/dist/chunk-LDA354A3.mjs +158 -0
- package/dist/chunk-LSUI3VI4.mjs +19 -0
- package/dist/chunk-MB2X3DXF.mjs +45 -0
- package/dist/chunk-MBL4CIFG.mjs +28 -0
- package/dist/chunk-MLAUEAUQ.mjs +27 -0
- package/dist/chunk-NGEKE4DQ.mjs +23 -0
- package/dist/chunk-NJ4FK5BO.mjs +19 -0
- package/dist/chunk-OJZZPNXQ.mjs +1985 -0
- package/dist/chunk-P24PKCUA.mjs +25 -0
- package/dist/chunk-PFQIKEIL.mjs +15 -0
- package/dist/chunk-Q5ASM7HE.mjs +45 -0
- package/dist/chunk-Q7X2R7N7.mjs +16 -0
- package/dist/chunk-QAIJCQWH.mjs +157 -0
- package/dist/chunk-QQAUZGVG.mjs +2166 -0
- package/dist/chunk-QUIF3FJ2.mjs +55 -0
- package/dist/chunk-RDFZ4GVM.mjs +23 -0
- package/dist/chunk-RFRKMUNA.mjs +49 -0
- package/dist/chunk-RIIQA6S6.mjs +2204 -0
- package/dist/chunk-SGOSAU2P.mjs +51 -0
- package/dist/chunk-SLYSEMYE.mjs +21 -0
- package/dist/chunk-SNKLZ7VB.mjs +36 -0
- package/dist/chunk-TKKUEAN6.mjs +45 -0
- package/dist/chunk-TMH7EUXF.mjs +179 -0
- package/dist/chunk-TP55TF47.mjs +1980 -0
- package/dist/chunk-U4AFTAZI.mjs +1980 -0
- package/dist/chunk-U4EXCWGR.mjs +174 -0
- package/dist/chunk-U4OINWJM.mjs +42 -0
- package/dist/chunk-U63TGBAS.mjs +179 -0
- package/dist/chunk-UESFH6US.mjs +210773 -0
- package/dist/chunk-UITLHBZA.mjs +164 -0
- package/dist/chunk-VFD26CTJ.mjs +16 -0
- package/dist/chunk-VSWWWPO7.mjs +205 -0
- package/dist/chunk-VUVPGH7A.mjs +157 -0
- package/dist/chunk-W34IFT2J.mjs +45 -0
- package/dist/chunk-WPSGF5DK.mjs +92 -0
- package/dist/chunk-WPU2XV5B.mjs +25 -0
- package/dist/chunk-WTPR7TOE.mjs +146 -0
- package/dist/chunk-X3ATBJ5Q.mjs +14 -0
- package/dist/chunk-XMZATJRD.mjs +23 -0
- package/dist/chunk-Y2Y225MR.mjs +117 -0
- package/dist/chunk-YNDKJ5QJ.mjs +89 -0
- package/dist/chunk-YPQNYUP3.mjs +92 -0
- package/dist/chunk-ZH22D4SN.mjs +179 -0
- package/dist/chunk-ZSJJGBQR.mjs +39 -0
- package/dist/chunk-ZSMMBZHO.mjs +64 -0
- package/dist/chunk-ZTXCA3NN.mjs +74 -0
- package/dist/chunk-ZYMQR24Z.mjs +145 -0
- package/dist/cli.d.mts +1 -0
- package/dist/cli.d.ts +1 -0
- package/dist/cli.js +5770 -0
- package/dist/cli.mjs +25 -0
- package/dist/config.d.mts +5 -0
- package/dist/config.d.ts +5 -0
- package/dist/config.js +41 -0
- package/dist/config.mjs +7 -0
- package/dist/defaults.d.mts +3 -0
- package/dist/defaults.d.ts +3 -0
- package/dist/defaults.js +47 -0
- package/dist/defaults.mjs +7 -0
- package/dist/detectors/ci.d.mts +5 -0
- package/dist/detectors/ci.d.ts +5 -0
- package/dist/detectors/ci.js +59 -0
- package/dist/detectors/ci.mjs +7 -0
- package/dist/detectors/docker.d.mts +5 -0
- package/dist/detectors/docker.d.ts +5 -0
- package/dist/detectors/docker.js +50 -0
- package/dist/detectors/docker.mjs +7 -0
- package/dist/detectors/next.d.mts +7 -0
- package/dist/detectors/next.d.ts +7 -0
- package/dist/detectors/next.js +49 -0
- package/dist/detectors/next.mjs +7 -0
- package/dist/detectors/prisma.d.mts +7 -0
- package/dist/detectors/prisma.d.ts +7 -0
- package/dist/detectors/prisma.js +48 -0
- package/dist/detectors/prisma.mjs +7 -0
- package/dist/import.d.mts +2 -0
- package/dist/import.d.ts +2 -0
- package/dist/import.js +2062 -0
- package/dist/import.mjs +53 -0
- package/dist/output.d.mts +7 -0
- package/dist/output.d.ts +7 -0
- package/dist/output.js +3619 -0
- package/dist/output.mjs +13 -0
- package/dist/scan.d.mts +7 -0
- package/dist/scan.d.ts +7 -0
- package/dist/scan.js +5725 -0
- package/dist/scan.mjs +12 -0
- package/dist/scanner.d.mts +5 -0
- package/dist/scanner.d.ts +5 -0
- package/dist/scanner.js +5514 -0
- package/dist/scanner.mjs +14 -0
- package/dist/simple-icons-hex.d.mts +6751 -0
- package/dist/simple-icons-hex.d.ts +6751 -0
- package/dist/simple-icons-hex.js +10118 -0
- package/dist/simple-icons-hex.mjs +6736 -0
- package/dist/sync.d.mts +7 -0
- package/dist/sync.d.ts +7 -0
- package/dist/sync.js +5725 -0
- package/dist/sync.mjs +160 -0
- package/dist/techDefinitions.d.mts +5 -0
- package/dist/techDefinitions.d.ts +5 -0
- package/dist/techDefinitions.js +2004 -0
- package/dist/techDefinitions.mjs +7 -0
- package/dist/techMap.d.mts +5 -0
- package/dist/techMap.d.ts +5 -0
- package/dist/techMap.js +2018 -0
- package/dist/techMap.mjs +8 -0
- package/dist/types.d.mts +24 -0
- package/dist/types.d.ts +24 -0
- package/dist/types.js +18 -0
- package/dist/types.mjs +1 -0
- package/dist/typescript-7Q5RIVZY.mjs +6 -0
- package/dist/typescript-T5HXSZZS.mjs +5 -0
- package/dist/watch.d.mts +3 -0
- package/dist/watch.d.ts +3 -0
- package/dist/watch.js +2122 -0
- package/dist/watch.mjs +13 -0
- package/package.json +57 -0
- package/public/assets/logos/ai/huggingface.svg +1 -0
- package/public/assets/logos/ai/langchain.svg +1 -0
- package/public/assets/logos/ai/openai.svg +1 -0
- package/public/assets/logos/ai/replicate.svg +1 -0
- package/public/assets/logos/ai/tensorflow.svg +1 -0
- package/public/assets/logos/api/apollographql.svg +1 -0
- package/public/assets/logos/api/graphql.svg +1 -0
- package/public/assets/logos/api/relay.svg +1 -0
- package/public/assets/logos/api/trpc.svg +1 -0
- package/public/assets/logos/auth/argo.svg +1 -0
- package/public/assets/logos/auth/auth0.svg +1 -0
- package/public/assets/logos/auth/authjs.svg +73 -0
- package/public/assets/logos/auth/clerk.svg +1 -0
- package/public/assets/logos/auth/firebase.svg +1 -0
- package/public/assets/logos/auth/jsonwebtokens.svg +1 -0
- package/public/assets/logos/auth/keycloak.svg +1 -0
- package/public/assets/logos/auth/passport.svg +1 -0
- package/public/assets/logos/automation/lerna.svg +1 -0
- package/public/assets/logos/automation/nx.svg +1 -0
- package/public/assets/logos/automation/turborepo.svg +1 -0
- package/public/assets/logos/backend/adonisjs.svg +1 -0
- package/public/assets/logos/backend/express.svg +1 -0
- package/public/assets/logos/backend/fastify.svg +1 -0
- package/public/assets/logos/backend/koa.svg +1 -0
- package/public/assets/logos/backend/nestjs.svg +1 -0
- package/public/assets/logos/build/babel.svg +1 -0
- package/public/assets/logos/build/esbuild.svg +1 -0
- package/public/assets/logos/build/rollupdotjs.svg +1 -0
- package/public/assets/logos/build/snowpack.svg +1 -0
- package/public/assets/logos/build/swc.svg +1 -0
- package/public/assets/logos/build/vite.svg +15 -0
- package/public/assets/logos/build/webpack.svg +1 -0
- package/public/assets/logos/ci/circleci.svg +1 -0
- package/public/assets/logos/ci/githubactions.svg +1 -0
- package/public/assets/logos/ci/gitlab.svg +1 -0
- package/public/assets/logos/ci/jenkins.svg +1 -0
- package/public/assets/logos/ci/travisci.svg +1 -0
- package/public/assets/logos/cloud/cloudflare.svg +1 -0
- package/public/assets/logos/cloud/googlecloud.svg +1 -0
- package/public/assets/logos/cloud/microsoftazure.svg +1 -0
- package/public/assets/logos/cms/contentful.svg +1 -0
- package/public/assets/logos/cms/directus.svg +1 -0
- package/public/assets/logos/cms/ghost.svg +1 -0
- package/public/assets/logos/cms/keystone.svg +1 -0
- package/public/assets/logos/cms/sanity.svg +1 -0
- package/public/assets/logos/cms/shopify.svg +1 -0
- package/public/assets/logos/cms/strapi.svg +1 -0
- package/public/assets/logos/cms/wordpress.svg +1 -0
- package/public/assets/logos/container/docker.svg +1 -0
- package/public/assets/logos/container/kubernetes.svg +1 -0
- package/public/assets/logos/css/antdesign.svg +1 -0
- package/public/assets/logos/css/bootstrap.svg +1 -0
- package/public/assets/logos/css/bulma.svg +1 -0
- package/public/assets/logos/css/chakraui.svg +1 -0
- package/public/assets/logos/css/daisyui.svg +1 -0
- package/public/assets/logos/css/fallback/mantine-logo.svg +1 -0
- package/public/assets/logos/css/fallback/shadcnui.svg +1 -0
- package/public/assets/logos/css/foundation.svg +1 -0
- package/public/assets/logos/css/headlessui.svg +1 -0
- package/public/assets/logos/css/less.svg +1 -0
- package/public/assets/logos/css/mui.svg +1 -0
- package/public/assets/logos/css/postcss.svg +1 -0
- package/public/assets/logos/css/radixui.svg +1 -0
- package/public/assets/logos/css/sass.svg +1 -0
- package/public/assets/logos/css/shadcnui.svg +1 -0
- package/public/assets/logos/css/styledcomponents.svg +1 -0
- package/public/assets/logos/css/tailwindcss.svg +1 -0
- package/public/assets/logos/database/amazondynamodb.svg +1 -0
- package/public/assets/logos/database/apachecassandra.svg +1 -0
- package/public/assets/logos/database/cockroachlabs.svg +1 -0
- package/public/assets/logos/database/duckdb.svg +1 -0
- package/public/assets/logos/database/elasticsearch.svg +1 -0
- package/public/assets/logos/database/fauna.svg +3 -0
- package/public/assets/logos/database/mariadb.svg +1 -0
- package/public/assets/logos/database/mongodb.svg +1 -0
- package/public/assets/logos/database/mysql.svg +1 -0
- package/public/assets/logos/database/planetscale.svg +1 -0
- package/public/assets/logos/database/postgresql.svg +1 -0
- package/public/assets/logos/database/redis.svg +1 -0
- package/public/assets/logos/database/sqlite.svg +1 -0
- package/public/assets/logos/database/supabase.svg +1 -0
- package/public/assets/logos/database/surrealdb.svg +1 -0
- package/public/assets/logos/devops/ansible.svg +1 -0
- package/public/assets/logos/devops/pulumi.svg +1 -0
- package/public/assets/logos/devops/serverless.svg +1 -0
- package/public/assets/logos/devops/terraform.svg +1 -0
- package/public/assets/logos/fallback/31557565.svg +32 -0
- package/public/assets/logos/fallback/Drizzle--Streamline-Simple-Icons.svg +7 -0
- package/public/assets/logos/fallback/changesets.svg +43 -0
- package/public/assets/logos/fallback/lemonsqueezy.svg +8 -0
- package/public/assets/logos/fallback/mantine-logo.svg +1 -0
- package/public/assets/logos/fallback/reactquery.svg +12 -0
- package/public/assets/logos/fallback/shadcnui.svg +21 -0
- package/public/assets/logos/fallback/stitches.svg +10 -0
- package/public/assets/logos/format/prettier.svg +1 -0
- package/public/assets/logos/frontend/alpinedotjs.svg +1 -0
- package/public/assets/logos/frontend/angular.svg +1 -0
- package/public/assets/logos/frontend/astro.svg +1 -0
- package/public/assets/logos/frontend/backbone.svg +1 -0
- package/public/assets/logos/frontend/emberdotjs.svg +1 -0
- package/public/assets/logos/frontend/htmx.svg +1 -0
- package/public/assets/logos/frontend/lit.svg +1 -0
- package/public/assets/logos/frontend/nextdotjs.svg +1 -0
- package/public/assets/logos/frontend/nuxt.svg +1 -0
- package/public/assets/logos/frontend/preact.svg +1 -0
- package/public/assets/logos/frontend/qwik.svg +1 -0
- package/public/assets/logos/frontend/react.svg +1 -0
- package/public/assets/logos/frontend/remix.svg +1 -0
- package/public/assets/logos/frontend/solid.svg +1 -0
- package/public/assets/logos/frontend/svelte.svg +1 -0
- package/public/assets/logos/frontend/vuedotjs.svg +1 -0
- package/public/assets/logos/hosting/netlify.svg +1 -0
- package/public/assets/logos/hosting/railway.svg +1 -0
- package/public/assets/logos/hosting/render.svg +1 -0
- package/public/assets/logos/hosting/vercel.svg +1 -0
- package/public/assets/logos/language/csharp.svg +1 -0
- package/public/assets/logos/language/go.svg +1 -0
- package/public/assets/logos/language/javascript.svg +1 -0
- package/public/assets/logos/language/openjdk.svg +1 -0
- package/public/assets/logos/language/php.svg +1 -0
- package/public/assets/logos/language/python.svg +1 -0
- package/public/assets/logos/language/ruby.svg +1 -0
- package/public/assets/logos/language/rust.svg +1 -0
- package/public/assets/logos/language/typescript.svg +1 -0
- package/public/assets/logos/lint/commitlint.svg +1 -0
- package/public/assets/logos/lint/eslint.svg +1 -0
- package/public/assets/logos/lint/husky.svg +3 -0
- package/public/assets/logos/lint/lintstaged.svg +3 -0
- package/public/assets/logos/mobile/capacitor.svg +1 -0
- package/public/assets/logos/mobile/expo.svg +1 -0
- package/public/assets/logos/mobile/flutter.svg +1 -0
- package/public/assets/logos/mobile/ionic.svg +1 -0
- package/public/assets/logos/mobile/react.svg +1 -0
- package/public/assets/logos/network/axios.svg +1 -0
- package/public/assets/logos/network/brandfetch.svg +1 -0
- package/public/assets/logos/network/grpc.svg +1 -0
- package/public/assets/logos/network/openapiinitiative.svg +1 -0
- package/public/assets/logos/network/swr.svg +1 -0
- package/public/assets/logos/orm/drizzle.svg +1 -0
- package/public/assets/logos/orm/mongoose.svg +1 -0
- package/public/assets/logos/orm/prisma.svg +1 -0
- package/public/assets/logos/orm/typeorm.svg +1 -0
- package/public/assets/logos/package/npm.svg +1 -0
- package/public/assets/logos/package/pnpm.svg +1 -0
- package/public/assets/logos/package/yarn.svg +1 -0
- package/public/assets/logos/payment/fallback/lemonsqueezy.svg +1 -0
- package/public/assets/logos/payment/paypal.svg +1 -0
- package/public/assets/logos/payment/stripe.svg +1 -0
- package/public/assets/logos/runtime/bun.svg +1 -0
- package/public/assets/logos/runtime/deno.svg +1 -0
- package/public/assets/logos/ssg/eleventy.svg +1 -0
- package/public/assets/logos/ssg/gatsby.svg +1 -0
- package/public/assets/logos/ssg/hugo.svg +1 -0
- package/public/assets/logos/ssg/jekyll.svg +1 -0
- package/public/assets/logos/state/mobx.svg +1 -0
- package/public/assets/logos/state/recoil.svg +1 -0
- package/public/assets/logos/state/redux.svg +1 -0
- package/public/assets/logos/state/xstate.svg +1 -0
- package/public/assets/logos/state/zustand.svg +1 -0
- package/public/assets/logos/testing/chai.svg +1 -0
- package/public/assets/logos/testing/cypress.svg +1 -0
- package/public/assets/logos/testing/jasmine.svg +1 -0
- package/public/assets/logos/testing/jest.svg +1 -0
- package/public/assets/logos/testing/mocha.svg +1 -0
- package/public/assets/logos/testing/playwright.svg +1 -0
- package/public/assets/logos/testing/puppeteer.svg +1 -0
- package/public/assets/logos/testing/storybook.svg +1 -0
- package/public/assets/logos/testing/testinglibrary.svg +1 -0
- package/public/assets/logos/testing/vitest.svg +1 -0
- package/public/assets/logos/utility/d3.svg +1 -0
- package/public/assets/logos/utility/leaflet.svg +1 -0
- package/public/assets/logos/utility/lodash.svg +1 -0
- package/public/assets/logos/utility/pyup.svg +1 -0
- package/public/assets/logos/utility/reactivex.svg +1 -0
- package/public/assets/logos/utility/threedotjs.svg +1 -0
- package/public/assets/logos/utility/zod.svg +1 -0
- package/public/stackscan-white.svg +26 -0
- package/public/stackscan.svg +26 -0
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import {
|
|
2
|
+
techDefinitions
|
|
3
|
+
} from "./chunk-U4AFTAZI.mjs";
|
|
4
|
+
|
|
5
|
+
// src/techMap.ts
|
|
6
|
+
var techMap = {};
|
|
7
|
+
for (const def of techDefinitions) {
|
|
8
|
+
for (const alias of def.aliases) {
|
|
9
|
+
techMap[alias] = {
|
|
10
|
+
name: def.name,
|
|
11
|
+
logo: def.logo,
|
|
12
|
+
type: def.category
|
|
13
|
+
};
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export {
|
|
18
|
+
techMap
|
|
19
|
+
};
|
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import {
|
|
2
|
+
DEFAULT_CATEGORY_ICONS
|
|
3
|
+
} from "./chunk-XMZATJRD.mjs";
|
|
4
|
+
|
|
5
|
+
// src/output.ts
|
|
6
|
+
import fs from "fs-extra";
|
|
7
|
+
import path from "path";
|
|
8
|
+
import chalk from "chalk";
|
|
9
|
+
async function writeOutput(outPath, techs, config, format = "json", assetsOutPath) {
|
|
10
|
+
const outDirectory = path.dirname(outPath);
|
|
11
|
+
await fs.ensureDir(outDirectory);
|
|
12
|
+
let availableLogos = /* @__PURE__ */ new Set();
|
|
13
|
+
if (assetsOutPath) {
|
|
14
|
+
availableLogos = await copyAssets(techs, assetsOutPath);
|
|
15
|
+
}
|
|
16
|
+
if (format === "json") {
|
|
17
|
+
await fs.writeJSON(
|
|
18
|
+
outPath,
|
|
19
|
+
{
|
|
20
|
+
generatedAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
21
|
+
tech: techs
|
|
22
|
+
},
|
|
23
|
+
{ spaces: 2 }
|
|
24
|
+
);
|
|
25
|
+
console.log(chalk.blue(`\u2192 Saved ${techs.length} tech entries to ${outPath}`));
|
|
26
|
+
} else if (format === "markdown") {
|
|
27
|
+
const relativeAssetsPath = assetsOutPath ? path.relative(outDirectory, assetsOutPath).replace(/\\/g, "/") : void 0;
|
|
28
|
+
const mdContent = generateMarkdown(techs, relativeAssetsPath, availableLogos);
|
|
29
|
+
const mdPath = outPath.endsWith(".json") ? outPath.replace(/\.json$/, ".md") : outPath;
|
|
30
|
+
await fs.writeFile(mdPath, mdContent);
|
|
31
|
+
console.log(chalk.blue(`\u2192 Saved markdown to ${mdPath}`));
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
async function copyAssets(techs, dest) {
|
|
35
|
+
const srcDir = path.resolve(__dirname, "../assets/logos");
|
|
36
|
+
const lucideDir = path.resolve(__dirname, "../node_modules/lucide-static/icons");
|
|
37
|
+
const copied = /* @__PURE__ */ new Set();
|
|
38
|
+
let count = 0;
|
|
39
|
+
for (const t of techs) {
|
|
40
|
+
let srcFile = t.logo ? path.join(srcDir, t.logo) : null;
|
|
41
|
+
let destFile = t.logo ? path.join(dest, t.logo) : null;
|
|
42
|
+
let isDefault = false;
|
|
43
|
+
if (!srcFile || !await fs.pathExists(srcFile)) {
|
|
44
|
+
const defaultIconName = DEFAULT_CATEGORY_ICONS[t.type];
|
|
45
|
+
if (defaultIconName) {
|
|
46
|
+
const lucideFile = path.join(lucideDir, `${defaultIconName}.svg`);
|
|
47
|
+
if (await fs.pathExists(lucideFile)) {
|
|
48
|
+
srcFile = lucideFile;
|
|
49
|
+
const newLogoPath = `defaults/${defaultIconName}.svg`;
|
|
50
|
+
destFile = path.join(dest, newLogoPath);
|
|
51
|
+
t.logo = newLogoPath;
|
|
52
|
+
isDefault = true;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
if (srcFile && destFile && await fs.pathExists(srcFile)) {
|
|
57
|
+
await fs.ensureDir(path.dirname(destFile));
|
|
58
|
+
await fs.copy(srcFile, destFile);
|
|
59
|
+
copied.add(t.logo);
|
|
60
|
+
if (!isDefault) count++;
|
|
61
|
+
} else {
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
if (count > 0) {
|
|
65
|
+
console.log(chalk.blue(`\u2192 Copied ${count} logos to ${dest}`));
|
|
66
|
+
}
|
|
67
|
+
return copied;
|
|
68
|
+
}
|
|
69
|
+
function generateMarkdown(techs, assetsPath, availableLogos) {
|
|
70
|
+
const grouped = {};
|
|
71
|
+
for (const t of techs) {
|
|
72
|
+
const type = t.type || "misc";
|
|
73
|
+
if (!grouped[type]) grouped[type] = [];
|
|
74
|
+
grouped[type].push(t);
|
|
75
|
+
}
|
|
76
|
+
let md = "# Tech Stack\n\n";
|
|
77
|
+
const order = [
|
|
78
|
+
"language",
|
|
79
|
+
"frontend",
|
|
80
|
+
"backend",
|
|
81
|
+
"framework",
|
|
82
|
+
"library",
|
|
83
|
+
"database",
|
|
84
|
+
"orm",
|
|
85
|
+
"auth",
|
|
86
|
+
"api",
|
|
87
|
+
"cloud",
|
|
88
|
+
"hosting",
|
|
89
|
+
"ci",
|
|
90
|
+
"devops",
|
|
91
|
+
"container",
|
|
92
|
+
"testing",
|
|
93
|
+
"build",
|
|
94
|
+
"tooling",
|
|
95
|
+
"misc"
|
|
96
|
+
];
|
|
97
|
+
const sortedKeys = Object.keys(grouped).sort((a, b) => {
|
|
98
|
+
const idxA = order.indexOf(a);
|
|
99
|
+
const idxB = order.indexOf(b);
|
|
100
|
+
if (idxA === -1 && idxB === -1) return a.localeCompare(b);
|
|
101
|
+
if (idxA === -1) return 1;
|
|
102
|
+
if (idxB === -1) return -1;
|
|
103
|
+
return idxA - idxB;
|
|
104
|
+
});
|
|
105
|
+
for (const type of sortedKeys) {
|
|
106
|
+
const title = type.charAt(0).toUpperCase() + type.slice(1);
|
|
107
|
+
md += `## ${title}
|
|
108
|
+
|
|
109
|
+
`;
|
|
110
|
+
for (const t of grouped[type]) {
|
|
111
|
+
if (assetsPath && t.logo && availableLogos?.has(t.logo)) {
|
|
112
|
+
const logoUrl = `${assetsPath}/${t.logo}`;
|
|
113
|
+
md += `- <img src="${logoUrl}" alt="${t.name}" width="24" height="24" /> **${t.name}**
|
|
114
|
+
`;
|
|
115
|
+
} else {
|
|
116
|
+
md += `- **${t.name}**
|
|
117
|
+
`;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
md += "\n";
|
|
121
|
+
}
|
|
122
|
+
return md;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
export {
|
|
126
|
+
writeOutput
|
|
127
|
+
};
|
|
@@ -0,0 +1,157 @@
|
|
|
1
|
+
import {
|
|
2
|
+
copyAssets,
|
|
3
|
+
generateMarkdown
|
|
4
|
+
} from "./chunk-ZH22D4SN.mjs";
|
|
5
|
+
import {
|
|
6
|
+
simple_icons_hex_default
|
|
7
|
+
} from "./chunk-EH2SEQZP.mjs";
|
|
8
|
+
import {
|
|
9
|
+
techMap
|
|
10
|
+
} from "./chunk-2KG7JDHY.mjs";
|
|
11
|
+
|
|
12
|
+
// src/sync.ts
|
|
13
|
+
import fs from "fs";
|
|
14
|
+
import path from "path";
|
|
15
|
+
var INPUT_DIR = path.join(process.cwd(), "stacksync", "input");
|
|
16
|
+
var OUTPUT_DIR = path.join(process.cwd(), "stacksync", "output");
|
|
17
|
+
function toCamelCase(str) {
|
|
18
|
+
return str.replace(/[^a-zA-Z0-9 ]/g, "").split(" ").map((word, index) => {
|
|
19
|
+
if (index === 0) return word.toLowerCase();
|
|
20
|
+
return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
|
|
21
|
+
}).join("");
|
|
22
|
+
}
|
|
23
|
+
async function sync(options = {}) {
|
|
24
|
+
console.log("\u{1F680} Starting Sync...");
|
|
25
|
+
if (options.color) {
|
|
26
|
+
console.log(`\u{1F3A8} Color mode: ${options.color}`);
|
|
27
|
+
}
|
|
28
|
+
if (!fs.existsSync(INPUT_DIR)) {
|
|
29
|
+
console.log(`Creating input directory at: ${INPUT_DIR}`);
|
|
30
|
+
fs.mkdirSync(INPUT_DIR, { recursive: true });
|
|
31
|
+
console.log('Please place your project folders inside "stacksync/input" and run this command again.');
|
|
32
|
+
process.exit(0);
|
|
33
|
+
}
|
|
34
|
+
const entries = fs.readdirSync(INPUT_DIR, { withFileTypes: true });
|
|
35
|
+
const projectDirs = entries.filter((dirent) => dirent.isDirectory());
|
|
36
|
+
if (projectDirs.length === 0) {
|
|
37
|
+
console.log('\u26A0\uFE0F No project directories found in "stacksync/input".');
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
console.log(`Found ${projectDirs.length} projects to process.
|
|
41
|
+
`);
|
|
42
|
+
const allProjects = [];
|
|
43
|
+
const allTechs = [];
|
|
44
|
+
for (const dir of projectDirs) {
|
|
45
|
+
const projectPath = path.join(INPUT_DIR, dir.name);
|
|
46
|
+
const packageJsonPath = path.join(projectPath, "package.json");
|
|
47
|
+
if (fs.existsSync(packageJsonPath)) {
|
|
48
|
+
try {
|
|
49
|
+
const content = fs.readFileSync(packageJsonPath, "utf-8");
|
|
50
|
+
const pkg = JSON.parse(content);
|
|
51
|
+
const allDeps = { ...pkg.dependencies, ...pkg.devDependencies };
|
|
52
|
+
const detectedTechs = [];
|
|
53
|
+
Object.keys(allDeps).forEach((dep) => {
|
|
54
|
+
if (techMap[dep]) {
|
|
55
|
+
const tech = techMap[dep];
|
|
56
|
+
let color = null;
|
|
57
|
+
if (options.color === "white") color = "#FFFFFF";
|
|
58
|
+
else if (options.color === "black") color = "#000000";
|
|
59
|
+
else if (options.color && options.color.startsWith("#")) color = options.color;
|
|
60
|
+
else {
|
|
61
|
+
const depSlug = dep.toLowerCase();
|
|
62
|
+
const nameSlug = tech.name.toLowerCase();
|
|
63
|
+
const nameSlugNoSpaces = tech.name.toLowerCase().replace(/\s+/g, "");
|
|
64
|
+
const hex = simple_icons_hex_default[depSlug] || simple_icons_hex_default[nameSlug] || simple_icons_hex_default[nameSlugNoSpaces];
|
|
65
|
+
if (hex) color = `#${hex}`;
|
|
66
|
+
}
|
|
67
|
+
detectedTechs.push({
|
|
68
|
+
name: tech.name,
|
|
69
|
+
slug: dep,
|
|
70
|
+
logo: `https://raw.githubusercontent.com/benjamindotdev/stacksync/main/public/assets/logos/${tech.logo}`,
|
|
71
|
+
relativePath: `public/assets/logos/${tech.logo}`,
|
|
72
|
+
color
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
});
|
|
76
|
+
const uniqueTechs = Array.from(new Set(detectedTechs.map((t) => t.slug))).map((slug) => detectedTechs.find((t) => t.slug === slug));
|
|
77
|
+
allTechs.push(...uniqueTechs);
|
|
78
|
+
const outputFolderName = toCamelCase(dir.name);
|
|
79
|
+
const projectOutputDir = path.join(OUTPUT_DIR, outputFolderName);
|
|
80
|
+
if (!fs.existsSync(projectOutputDir)) {
|
|
81
|
+
fs.mkdirSync(projectOutputDir, { recursive: true });
|
|
82
|
+
}
|
|
83
|
+
fs.writeFileSync(
|
|
84
|
+
path.join(projectOutputDir, "stack.json"),
|
|
85
|
+
JSON.stringify(uniqueTechs, null, 2)
|
|
86
|
+
);
|
|
87
|
+
const mdContent = generateMarkdown(uniqueTechs);
|
|
88
|
+
fs.writeFileSync(path.join(projectOutputDir, "stack.md"), mdContent);
|
|
89
|
+
allProjects.push({
|
|
90
|
+
name: dir.name,
|
|
91
|
+
techs: uniqueTechs
|
|
92
|
+
});
|
|
93
|
+
console.log(`\u2705 ${dir.name.padEnd(20)} -> ${outputFolderName}/stack.json (${uniqueTechs.length} techs)`);
|
|
94
|
+
} catch (err) {
|
|
95
|
+
console.error(`\u274C Error processing ${dir.name}:`, err.message);
|
|
96
|
+
}
|
|
97
|
+
} else {
|
|
98
|
+
console.warn(`\u26A0\uFE0F Skipping "${dir.name}": No package.json found.`);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
if (allProjects.length > 0) {
|
|
102
|
+
const assetsDir = path.join(process.cwd(), "assets", "logos");
|
|
103
|
+
const uniqueAllTechs = Array.from(new Set(allTechs.map((t) => t.slug))).map((slug) => allTechs.find((t) => t.slug === slug));
|
|
104
|
+
const techsForCopy = uniqueAllTechs.map((t) => ({
|
|
105
|
+
...t,
|
|
106
|
+
logo: t.relativePath.replace("public/assets/logos/", "")
|
|
107
|
+
// Restore "frameworks/react.svg"
|
|
108
|
+
}));
|
|
109
|
+
await copyAssets(techsForCopy, assetsDir, { colorMode: options.color });
|
|
110
|
+
updateRootReadme(allProjects);
|
|
111
|
+
}
|
|
112
|
+
console.log("\n\u2728 Sync complete.");
|
|
113
|
+
}
|
|
114
|
+
function updateRootReadme(projects) {
|
|
115
|
+
const readmePath = path.join(process.cwd(), "README.md");
|
|
116
|
+
if (!fs.existsSync(readmePath)) {
|
|
117
|
+
console.log("\u26A0\uFE0F No root README.md found to update.");
|
|
118
|
+
return;
|
|
119
|
+
}
|
|
120
|
+
let readmeContent = fs.readFileSync(readmePath, "utf-8");
|
|
121
|
+
const startMarker = "<!-- STACKSYNC_START -->";
|
|
122
|
+
const endMarker = "<!-- STACKSYNC_END -->";
|
|
123
|
+
let newSection = `${startMarker}
|
|
124
|
+
## My Projects
|
|
125
|
+
|
|
126
|
+
`;
|
|
127
|
+
for (const p of projects) {
|
|
128
|
+
newSection += `### ${p.name}
|
|
129
|
+
`;
|
|
130
|
+
newSection += `<p>
|
|
131
|
+
`;
|
|
132
|
+
for (const t of p.techs) {
|
|
133
|
+
const src = t.relativePath || t.logo;
|
|
134
|
+
newSection += ` <img src="${src}" alt="${t.name}" height="25" style="margin-right: 10px;" />
|
|
135
|
+
`;
|
|
136
|
+
}
|
|
137
|
+
newSection += `</p>
|
|
138
|
+
|
|
139
|
+
`;
|
|
140
|
+
}
|
|
141
|
+
newSection += `${endMarker}`;
|
|
142
|
+
if (readmeContent.includes(startMarker) && readmeContent.includes(endMarker)) {
|
|
143
|
+
const regex = new RegExp(`${startMarker}[\\s\\S]*?${endMarker}`);
|
|
144
|
+
readmeContent = readmeContent.replace(regex, newSection);
|
|
145
|
+
console.log(`\u{1F4DD} Updated root README.md with ${projects.length} projects.`);
|
|
146
|
+
} else {
|
|
147
|
+
readmeContent += `
|
|
148
|
+
|
|
149
|
+
${newSection}`;
|
|
150
|
+
console.log(`\u{1F4DD} Appended projects to root README.md.`);
|
|
151
|
+
}
|
|
152
|
+
fs.writeFileSync(readmePath, readmeContent);
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
export {
|
|
156
|
+
sync
|
|
157
|
+
};
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
// src/detectors/next.ts
|
|
2
|
+
import fs from "fs-extra";
|
|
3
|
+
import path from "path";
|
|
4
|
+
async function detectNext(repoPath) {
|
|
5
|
+
const pages = path.join(repoPath, "pages");
|
|
6
|
+
const app = path.join(repoPath, "app");
|
|
7
|
+
if (await fs.pathExists(pages) || await fs.pathExists(app)) {
|
|
8
|
+
return { name: "Next.js App", logo: "frontend/nextdotjs.svg", type: "frontend" };
|
|
9
|
+
}
|
|
10
|
+
return null;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export {
|
|
14
|
+
detectNext
|
|
15
|
+
};
|
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
import {
|
|
2
|
+
DEFAULT_CATEGORY_ICONS
|
|
3
|
+
} from "./chunk-NGEKE4DQ.mjs";
|
|
4
|
+
import {
|
|
5
|
+
simple_icons_hex_default
|
|
6
|
+
} from "./chunk-7TIXAYVC.mjs";
|
|
7
|
+
import {
|
|
8
|
+
__dirname
|
|
9
|
+
} from "./chunk-EOKQCSHI.mjs";
|
|
10
|
+
|
|
11
|
+
// src/output.ts
|
|
12
|
+
import fs from "fs-extra";
|
|
13
|
+
import path from "path";
|
|
14
|
+
import chalk from "chalk";
|
|
15
|
+
import { createRequire } from "module";
|
|
16
|
+
var require2 = createRequire(import.meta.url);
|
|
17
|
+
async function writeOutput(outPath, techs, config, format = "json", assetsOutPath) {
|
|
18
|
+
const outDirectory = path.dirname(outPath);
|
|
19
|
+
await fs.ensureDir(outDirectory);
|
|
20
|
+
let availableLogos = /* @__PURE__ */ new Set();
|
|
21
|
+
if (assetsOutPath) {
|
|
22
|
+
availableLogos = await copyAssets(techs, assetsOutPath, config);
|
|
23
|
+
}
|
|
24
|
+
if (format === "json") {
|
|
25
|
+
await fs.writeJSON(
|
|
26
|
+
outPath,
|
|
27
|
+
{
|
|
28
|
+
generatedAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
29
|
+
tech: techs
|
|
30
|
+
},
|
|
31
|
+
{ spaces: 2 }
|
|
32
|
+
);
|
|
33
|
+
console.log(chalk.blue(`\u2192 Saved ${techs.length} tech entries to ${outPath}`));
|
|
34
|
+
} else if (format === "markdown") {
|
|
35
|
+
const relativeAssetsPath = assetsOutPath ? path.relative(outDirectory, assetsOutPath).replace(/\\/g, "/") : void 0;
|
|
36
|
+
const mdContent = generateMarkdown(techs, relativeAssetsPath, availableLogos);
|
|
37
|
+
const mdPath = outPath.endsWith(".json") ? outPath.replace(/\.json$/, ".md") : outPath;
|
|
38
|
+
await fs.writeFile(mdPath, mdContent);
|
|
39
|
+
console.log(chalk.blue(`\u2192 Saved markdown to ${mdPath}`));
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
async function copyAssets(techs, dest, config) {
|
|
43
|
+
const srcDir = path.resolve(__dirname, "../assets/logos");
|
|
44
|
+
let lucideDir;
|
|
45
|
+
try {
|
|
46
|
+
lucideDir = path.join(path.dirname(require2.resolve("lucide-static/package.json")), "icons");
|
|
47
|
+
} catch (e) {
|
|
48
|
+
lucideDir = path.resolve(__dirname, "../node_modules/lucide-static/icons");
|
|
49
|
+
}
|
|
50
|
+
const copied = /* @__PURE__ */ new Set();
|
|
51
|
+
let count = 0;
|
|
52
|
+
for (const t of techs) {
|
|
53
|
+
let srcFile = t.logo ? path.join(srcDir, t.logo) : null;
|
|
54
|
+
let destFile = t.logo ? path.join(dest, t.logo) : null;
|
|
55
|
+
let isDefault = false;
|
|
56
|
+
if (!srcFile || !await fs.pathExists(srcFile)) {
|
|
57
|
+
const defaultIconName = DEFAULT_CATEGORY_ICONS[t.type];
|
|
58
|
+
if (defaultIconName) {
|
|
59
|
+
const lucideFile = path.join(lucideDir, `${defaultIconName}.svg`);
|
|
60
|
+
if (await fs.pathExists(lucideFile)) {
|
|
61
|
+
srcFile = lucideFile;
|
|
62
|
+
const newLogoPath = `defaults/${defaultIconName}.svg`;
|
|
63
|
+
destFile = path.join(dest, newLogoPath);
|
|
64
|
+
t.logo = newLogoPath;
|
|
65
|
+
isDefault = true;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
if (srcFile && destFile && await fs.pathExists(srcFile)) {
|
|
70
|
+
await fs.ensureDir(path.dirname(destFile));
|
|
71
|
+
let svgContent = await fs.readFile(srcFile, "utf8");
|
|
72
|
+
let color;
|
|
73
|
+
if (config.iconColors && config.iconColors[t.name]) {
|
|
74
|
+
color = config.iconColors[t.name];
|
|
75
|
+
} else if (config.colorMode === "white") {
|
|
76
|
+
color = "#FFFFFF";
|
|
77
|
+
} else if (config.colorMode === "black") {
|
|
78
|
+
color = "#000000";
|
|
79
|
+
} else if (config.colorMode === "custom" && config.customColor) {
|
|
80
|
+
color = config.customColor;
|
|
81
|
+
} else if (config.colorMode === "default" || !config.colorMode) {
|
|
82
|
+
if (!isDefault) {
|
|
83
|
+
const slug = path.basename(t.logo, ".svg");
|
|
84
|
+
const brandHex = simple_icons_hex_default[slug];
|
|
85
|
+
if (brandHex) {
|
|
86
|
+
color = `#${brandHex}`;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
if (color) {
|
|
91
|
+
if (isDefault) {
|
|
92
|
+
if (svgContent.includes("stroke=")) {
|
|
93
|
+
svgContent = svgContent.replace(/stroke="[^"]*"/g, `stroke="${color}"`);
|
|
94
|
+
} else {
|
|
95
|
+
svgContent = svgContent.replace("<svg", `<svg stroke="${color}"`);
|
|
96
|
+
}
|
|
97
|
+
} else {
|
|
98
|
+
if (svgContent.includes("fill=")) {
|
|
99
|
+
svgContent = svgContent.replace(/fill="[^"]*"/g, `fill="${color}"`);
|
|
100
|
+
} else {
|
|
101
|
+
svgContent = svgContent.replace("<svg", `<svg fill="${color}"`);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
await fs.writeFile(destFile, svgContent);
|
|
106
|
+
copied.add(t.logo);
|
|
107
|
+
if (!isDefault) count++;
|
|
108
|
+
} else {
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
if (count > 0) {
|
|
112
|
+
console.log(chalk.blue(`\u2192 Copied ${count} logos to ${dest}`));
|
|
113
|
+
}
|
|
114
|
+
return copied;
|
|
115
|
+
}
|
|
116
|
+
function generateMarkdown(techs, assetsPath, availableLogos) {
|
|
117
|
+
const grouped = {};
|
|
118
|
+
for (const t of techs) {
|
|
119
|
+
const type = t.type || "misc";
|
|
120
|
+
if (!grouped[type]) grouped[type] = [];
|
|
121
|
+
grouped[type].push(t);
|
|
122
|
+
}
|
|
123
|
+
let md = "# Tech Stack\n\n";
|
|
124
|
+
const order = [
|
|
125
|
+
"language",
|
|
126
|
+
"frontend",
|
|
127
|
+
"backend",
|
|
128
|
+
"framework",
|
|
129
|
+
"library",
|
|
130
|
+
"database",
|
|
131
|
+
"orm",
|
|
132
|
+
"auth",
|
|
133
|
+
"api",
|
|
134
|
+
"cloud",
|
|
135
|
+
"hosting",
|
|
136
|
+
"ci",
|
|
137
|
+
"devops",
|
|
138
|
+
"container",
|
|
139
|
+
"testing",
|
|
140
|
+
"build",
|
|
141
|
+
"tooling",
|
|
142
|
+
"misc"
|
|
143
|
+
];
|
|
144
|
+
const sortedKeys = Object.keys(grouped).sort((a, b) => {
|
|
145
|
+
const idxA = order.indexOf(a);
|
|
146
|
+
const idxB = order.indexOf(b);
|
|
147
|
+
if (idxA === -1 && idxB === -1) return a.localeCompare(b);
|
|
148
|
+
if (idxA === -1) return 1;
|
|
149
|
+
if (idxB === -1) return -1;
|
|
150
|
+
return idxA - idxB;
|
|
151
|
+
});
|
|
152
|
+
for (const type of sortedKeys) {
|
|
153
|
+
const title = type.charAt(0).toUpperCase() + type.slice(1);
|
|
154
|
+
md += `## ${title}
|
|
155
|
+
|
|
156
|
+
`;
|
|
157
|
+
for (const t of grouped[type]) {
|
|
158
|
+
if (assetsPath && t.logo && availableLogos?.has(t.logo)) {
|
|
159
|
+
const logoUrl = `${assetsPath}/${t.logo}`;
|
|
160
|
+
md += `- <img src="${logoUrl}" alt="${t.name}" width="24" height="24" /> **${t.name}**
|
|
161
|
+
`;
|
|
162
|
+
} else {
|
|
163
|
+
md += `- **${t.name}**
|
|
164
|
+
`;
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
md += "\n";
|
|
168
|
+
}
|
|
169
|
+
return md;
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
export {
|
|
173
|
+
writeOutput
|
|
174
|
+
};
|
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
import {
|
|
2
|
+
DEFAULT_CATEGORY_ICONS
|
|
3
|
+
} from "./chunk-NGEKE4DQ.mjs";
|
|
4
|
+
import {
|
|
5
|
+
__dirname
|
|
6
|
+
} from "./chunk-EOKQCSHI.mjs";
|
|
7
|
+
import {
|
|
8
|
+
simple_icons_hex_default
|
|
9
|
+
} from "./chunk-EH2SEQZP.mjs";
|
|
10
|
+
|
|
11
|
+
// src/output.ts
|
|
12
|
+
import fs from "fs-extra";
|
|
13
|
+
import path from "path";
|
|
14
|
+
import chalk from "chalk";
|
|
15
|
+
import { createRequire } from "module";
|
|
16
|
+
var require2 = createRequire(import.meta.url);
|
|
17
|
+
async function writeOutput(outPath, techs, config, format = "json", assetsOutPath) {
|
|
18
|
+
const outDirectory = path.dirname(outPath);
|
|
19
|
+
await fs.ensureDir(outDirectory);
|
|
20
|
+
let availableLogos = /* @__PURE__ */ new Set();
|
|
21
|
+
if (assetsOutPath) {
|
|
22
|
+
availableLogos = await copyAssets(techs, assetsOutPath, config);
|
|
23
|
+
}
|
|
24
|
+
if (format === "json") {
|
|
25
|
+
await fs.writeJSON(
|
|
26
|
+
outPath,
|
|
27
|
+
{
|
|
28
|
+
generatedAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
29
|
+
tech: techs
|
|
30
|
+
},
|
|
31
|
+
{ spaces: 2 }
|
|
32
|
+
);
|
|
33
|
+
console.log(chalk.blue(`\u2192 Saved ${techs.length} tech entries to ${outPath}`));
|
|
34
|
+
} else if (format === "markdown") {
|
|
35
|
+
const relativeAssetsPath = assetsOutPath ? path.relative(outDirectory, assetsOutPath).replace(/\\/g, "/") : void 0;
|
|
36
|
+
const mdContent = generateMarkdown(techs, relativeAssetsPath, availableLogos);
|
|
37
|
+
const mdPath = outPath.endsWith(".json") ? outPath.replace(/\.json$/, ".md") : outPath;
|
|
38
|
+
await fs.writeFile(mdPath, mdContent);
|
|
39
|
+
console.log(chalk.blue(`\u2192 Saved markdown to ${mdPath}`));
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
async function copyAssets(techs, dest, config) {
|
|
43
|
+
const srcDir = path.resolve(__dirname, "../assets/logos");
|
|
44
|
+
let lucideDir;
|
|
45
|
+
try {
|
|
46
|
+
lucideDir = path.join(path.dirname(require2.resolve("lucide-static/package.json")), "icons");
|
|
47
|
+
} catch (e) {
|
|
48
|
+
lucideDir = path.resolve(__dirname, "../node_modules/lucide-static/icons");
|
|
49
|
+
}
|
|
50
|
+
const copied = /* @__PURE__ */ new Set();
|
|
51
|
+
let count = 0;
|
|
52
|
+
for (const t of techs) {
|
|
53
|
+
let srcFile = t.logo ? path.join(srcDir, t.logo) : null;
|
|
54
|
+
let destFile = t.logo ? path.join(dest, t.logo) : null;
|
|
55
|
+
let isDefault = false;
|
|
56
|
+
if (!srcFile || !await fs.pathExists(srcFile)) {
|
|
57
|
+
const defaultIconName = DEFAULT_CATEGORY_ICONS[t.type];
|
|
58
|
+
if (defaultIconName) {
|
|
59
|
+
const lucideFile = path.join(lucideDir, `${defaultIconName}.svg`);
|
|
60
|
+
if (await fs.pathExists(lucideFile)) {
|
|
61
|
+
srcFile = lucideFile;
|
|
62
|
+
const newLogoPath = `defaults/${defaultIconName}.svg`;
|
|
63
|
+
destFile = path.join(dest, newLogoPath);
|
|
64
|
+
t.logo = newLogoPath;
|
|
65
|
+
isDefault = true;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
if (srcFile && destFile && await fs.pathExists(srcFile)) {
|
|
70
|
+
await fs.ensureDir(path.dirname(destFile));
|
|
71
|
+
let svgContent = await fs.readFile(srcFile, "utf8");
|
|
72
|
+
let color;
|
|
73
|
+
if (config.iconColors && config.iconColors[t.name]) {
|
|
74
|
+
color = config.iconColors[t.name];
|
|
75
|
+
} else if (config.colorMode === "white") {
|
|
76
|
+
color = "#FFFFFF";
|
|
77
|
+
} else if (config.colorMode === "black") {
|
|
78
|
+
color = "#000000";
|
|
79
|
+
} else if (config.colorMode === "custom" && config.customColor) {
|
|
80
|
+
color = config.customColor;
|
|
81
|
+
} else if (config.colorMode === "default" || !config.colorMode) {
|
|
82
|
+
if (!isDefault) {
|
|
83
|
+
const slug = path.basename(t.logo, ".svg");
|
|
84
|
+
const brandHex = simple_icons_hex_default[slug];
|
|
85
|
+
if (brandHex) {
|
|
86
|
+
color = `#${brandHex}`;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
if (color) {
|
|
91
|
+
if (isDefault) {
|
|
92
|
+
if (svgContent.includes("stroke=")) {
|
|
93
|
+
svgContent = svgContent.replace(/stroke="[^"]*"/g, `stroke="${color}"`);
|
|
94
|
+
} else {
|
|
95
|
+
svgContent = svgContent.replace("<svg", `<svg stroke="${color}"`);
|
|
96
|
+
}
|
|
97
|
+
} else {
|
|
98
|
+
if (svgContent.includes("fill=")) {
|
|
99
|
+
svgContent = svgContent.replace(/fill="[^"]*"/g, `fill="${color}"`);
|
|
100
|
+
} else {
|
|
101
|
+
svgContent = svgContent.replace("<svg", `<svg fill="${color}"`);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
await fs.writeFile(destFile, svgContent);
|
|
106
|
+
copied.add(t.logo);
|
|
107
|
+
if (!isDefault) count++;
|
|
108
|
+
} else {
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
if (count > 0) {
|
|
112
|
+
console.log(chalk.blue(`\u2192 Copied ${count} logos to ${dest}`));
|
|
113
|
+
}
|
|
114
|
+
return copied;
|
|
115
|
+
}
|
|
116
|
+
function generateMarkdown(techs, assetsPath, availableLogos) {
|
|
117
|
+
const grouped = {};
|
|
118
|
+
for (const t of techs) {
|
|
119
|
+
const type = t.type || "misc";
|
|
120
|
+
if (!grouped[type]) grouped[type] = [];
|
|
121
|
+
grouped[type].push(t);
|
|
122
|
+
}
|
|
123
|
+
let md = "# Tech Stack\n\n";
|
|
124
|
+
const order = [
|
|
125
|
+
"language",
|
|
126
|
+
"frontend",
|
|
127
|
+
"backend",
|
|
128
|
+
"framework",
|
|
129
|
+
"library",
|
|
130
|
+
"database",
|
|
131
|
+
"orm",
|
|
132
|
+
"auth",
|
|
133
|
+
"api",
|
|
134
|
+
"cloud",
|
|
135
|
+
"hosting",
|
|
136
|
+
"ci",
|
|
137
|
+
"devops",
|
|
138
|
+
"container",
|
|
139
|
+
"testing",
|
|
140
|
+
"build",
|
|
141
|
+
"tooling",
|
|
142
|
+
"misc"
|
|
143
|
+
];
|
|
144
|
+
const sortedKeys = Object.keys(grouped).sort((a, b) => {
|
|
145
|
+
const idxA = order.indexOf(a);
|
|
146
|
+
const idxB = order.indexOf(b);
|
|
147
|
+
if (idxA === -1 && idxB === -1) return a.localeCompare(b);
|
|
148
|
+
if (idxA === -1) return 1;
|
|
149
|
+
if (idxB === -1) return -1;
|
|
150
|
+
return idxA - idxB;
|
|
151
|
+
});
|
|
152
|
+
for (const type of sortedKeys) {
|
|
153
|
+
const title = type.charAt(0).toUpperCase() + type.slice(1);
|
|
154
|
+
md += `## ${title}
|
|
155
|
+
|
|
156
|
+
`;
|
|
157
|
+
for (const t of grouped[type]) {
|
|
158
|
+
if (t.logo && (t.logo.startsWith("http") || t.logo.startsWith("//"))) {
|
|
159
|
+
md += `- <img src="${t.logo}" alt="${t.name}" width="24" height="24" /> **${t.name}**
|
|
160
|
+
`;
|
|
161
|
+
} else if (assetsPath && t.logo && availableLogos?.has(t.logo)) {
|
|
162
|
+
const logoUrl = `${assetsPath}/${t.logo}`;
|
|
163
|
+
md += `- <img src="${logoUrl}" alt="${t.name}" width="24" height="24" /> **${t.name}**
|
|
164
|
+
`;
|
|
165
|
+
} else {
|
|
166
|
+
md += `- **${t.name}**
|
|
167
|
+
`;
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
md += "\n";
|
|
171
|
+
}
|
|
172
|
+
return md;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
export {
|
|
176
|
+
writeOutput,
|
|
177
|
+
copyAssets,
|
|
178
|
+
generateMarkdown
|
|
179
|
+
};
|