initial commit

This commit is contained in:
Jonas Almeida 2025-03-10 10:35:00 -03:00
commit 819744f3b4
16 changed files with 3023 additions and 0 deletions

24
.github/workflows/npm-publish.yml vendored Normal file
View File

@ -0,0 +1,24 @@
name: NPM Package Publish
on:
workflow_dispatch:
jobs:
publish-npm:
runs-on: ubuntu-latest
permissions:
contents: read
id-token: write
steps:
- uses: actions/checkout@v4
- uses: oven-sh/setup-bun@v2
- uses: actions/setup-node@v4
with:
node-version: "20.x"
registry-url: "https://registry.npmjs.org"
- run: bun install
- run: bun run build
- name: Publish to NPM
run: npm publish --provenance --access public
env:
NODE_AUTH_TOKEN: ${{secrets.NPM_TOKEN}}

177
.gitignore vendored Normal file
View File

@ -0,0 +1,177 @@
# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore
# Logs
logs
_.log
npm-debug.log_
yarn-debug.log*
yarn-error.log*
lerna-debug.log*
.pnpm-debug.log*
# Caches
.cache
# Diagnostic reports (https://nodejs.org/api/report.html)
report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json
# Runtime data
pids
_.pid
_.seed
*.pid.lock
# Directory for instrumented libs generated by jscoverage/JSCover
lib-cov
# Coverage directory used by tools like istanbul
coverage
*.lcov
# nyc test coverage
.nyc_output
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
.grunt
# Bower dependency directory (https://bower.io/)
bower_components
# node-waf configuration
.lock-wscript
# Compiled binary addons (https://nodejs.org/api/addons.html)
build/Release
# Dependency directories
node_modules/
jspm_packages/
# Snowpack dependency directory (https://snowpack.dev/)
web_modules/
# TypeScript cache
*.tsbuildinfo
# Optional npm cache directory
.npm
# Optional eslint cache
.eslintcache
# Optional stylelint cache
.stylelintcache
# Microbundle cache
.rpt2_cache/
.rts2_cache_cjs/
.rts2_cache_es/
.rts2_cache_umd/
# Optional REPL history
.node_repl_history
# Output of 'npm pack'
*.tgz
# Yarn Integrity file
.yarn-integrity
# dotenv environment variable files
.env
.env.development.local
.env.test.local
.env.production.local
.env.local
# parcel-bundler cache (https://parceljs.org/)
.parcel-cache
# Next.js build output
.next
out
# Nuxt.js build / generate output
.nuxt
dist
# Gatsby files
# Comment in the public line in if your project uses Gatsby and not Next.js
# https://nextjs.org/blog/next-9-1#public-directory-support
# public
# vuepress build output
.vuepress/dist
# vuepress v2.x temp and cache directory
.temp
# Docusaurus cache and generated files
.docusaurus
# Serverless directories
.serverless/
# FuseBox cache
.fusebox/
# DynamoDB Local files
.dynamodb/
# TernJS port file
.tern-port
# Stores VSCode versions used for testing VSCode extensions
.vscode-test
# yarn v2
.yarn/cache
.yarn/unplugged
.yarn/build-state.yml
.yarn/install-state.gz
.pnp.*
# IntelliJ based IDEs
.idea
# Finder (MacOS) folder config
.DS_Store
extensions_support/

57
CHANGELOG.md Normal file
View File

@ -0,0 +1,57 @@
# template-ext-worker
## 0.0.10
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.5
## 0.0.9
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.4
## 0.0.8
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.2
## 0.0.7
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.1
## 0.0.6
### Patch Changes
- Updated dependencies
- @kksh/api@0.1.0
## 0.0.5
### Patch Changes
- Updated dependencies
- @kksh/api@0.0.53
## 0.0.4
### Patch Changes
- Updated dependencies
- @kksh/api@0.0.48
## 0.0.3
### Patch Changes
- Updated dependencies
- @kksh/api@0.0.47

7
README.md Normal file
View File

@ -0,0 +1,7 @@
# Random Data Generator Extension for Kunkun
Random Data Generator is an extension for [Kunkun](https://kunkun.sh/).
## Credits
[Raycast](https://www.raycast.com/loris/random) extension ported to [Kunkun](https://kunkun.sh/).

30
build.ts Normal file
View File

@ -0,0 +1,30 @@
import { watch } from "fs"
import { join } from "path"
import { refreshTemplateWorkerCommand } from "@kksh/api/dev"
import { $ } from "bun"
const entrypoints = ["./src/index.ts"]
async function build() {
try {
for (const entrypoint of entrypoints) {
await $`bun build --minify --target=browser --outdir=./dist ${entrypoint}`
}
if (Bun.argv.includes("dev")) {
await refreshTemplateWorkerCommand()
}
} catch (error) {
console.error(error)
}
}
const srcDir = join(import.meta.dir, "src")
await build()
if (Bun.argv.includes("dev")) {
console.log(`Watching ${srcDir} for changes...`)
watch(srcDir, { recursive: true }, async (event, filename) => {
await build()
})
}

71
package.json Normal file
View File

@ -0,0 +1,71 @@
{
"$schema": "https://schema.kunkun.sh",
"name": "kunkun-ext-random-data-generator",
"version": "0.0.10",
"license": "MIT",
"type": "module",
"author": "Jonas Almeida",
"repository": {
"type": "git",
"url": "https://github.com/jonasrafa/kunkun-ext-random-data-generator"
},
"kunkun": {
"name": "Random Data Generator",
"shortDescription": "Generate random data using Faker library",
"longDescription": "Generate random data using Faker library",
"identifier": "random-data-generator",
"permissions": [
"clipboard:write-text",
{
"permission": "open:url",
"allow": [
{
"url": "https://**"
},
{
"url": "http://**"
}
]
}
],
"demoImages": [
"https://raw.githubusercontent.com/jonasrafa/kunkun-ext-random-data-generator/refs/heads/main/public/screenshot1.jpg",
"https://raw.githubusercontent.com/jonasrafa/kunkun-ext-random-data-generator/refs/heads/main/public/screenshot2.jpg"
],
"icon": {
"type": "iconify",
"value": "fe:random"
},
"customUiCmds": [],
"templateUiCmds": [
{
"name": "Generate Random Data",
"description": "Generate random data using Faker library",
"main": "dist/index.js",
"cmds": []
}
]
},
"scripts": {
"dev": "bun build.ts dev",
"build": "bun build.ts"
},
"dependencies": {
"@bugsnag/cuid": "^3.2.1",
"@faker-js/faker": "^7.6.0",
"@kksh/api": "0.1.5",
"i18next": "^23.16.8",
"is-url": "^1.2.4",
"lodash": "^4.17.21"
},
"devDependencies": {
"@types/bun": "latest"
},
"peerDependencies": {
"typescript": "^5.0.0"
},
"files": [
"./dist",
".gitignore"
]
}

2349
pnpm-lock.yaml generated Normal file

File diff suppressed because it is too large Load Diff

BIN
public/screenshot1.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 56 KiB

BIN
public/screenshot2.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 67 KiB

14
src/faker.ts Normal file
View File

@ -0,0 +1,14 @@
import cuid from "@bugsnag/cuid";
import type { Faker as OriginalFaker } from "@faker-js/faker";
import { faker } from "@faker-js/faker";
// Hack since faker-js doesn't support CUIDs or a way to extend its API
(faker.datatype as unknown as { cuid: () => string }).cuid = () => cuid();
export type Faker = OriginalFaker & {
datatype: typeof faker.datatype & {
cuid: () => string;
};
};
export default faker as Faker;

5
src/i18n/en.ts Normal file
View File

@ -0,0 +1,5 @@
const en = {
welcome: "Welcome to Kunkun"
}
export default en
export type Translation = typeof en

20
src/i18n/index.ts Normal file
View File

@ -0,0 +1,20 @@
import i18next from "i18next"
import en, { type Translation } from "./en"
import zh from "./zh"
export function setupI18n(language: "en" | "zh" = "en") {
i18next.init({
resources: {
en: {
translation: en
},
zh: {
translation: zh
}
},
lng: language, // default language
fallbackLng: "en"
})
}
export const t = (key: keyof Translation, options?: any) => i18next.t(key, options)

5
src/i18n/zh.ts Normal file
View File

@ -0,0 +1,5 @@
import type { Translation } from "./en"
export default {
welcome: "欢迎来到Kunkun"
} satisfies Translation

177
src/index.ts Normal file
View File

@ -0,0 +1,177 @@
import {
Action,
clipboard,
expose,
Icon,
IconEnum,
List,
open,
Markdown,
TemplateUiCommand,
toast,
ui,
} from "@kksh/api/ui/template";
import { buildItems, type Item } from "./utils";
import faker from "./faker";
import _ from "lodash";
import isUrl from "is-url";
const Actions = {
OpenInBrowser: "Open in Browser",
CopyToClipboard: "Copy to Clipboard",
RefreshValue: "Refresh Value (Not implemented)",
PinEntry: "Pin Entry (Not implemented)",
UnpinEntry: "Unpin Entry (Not implemented)",
};
class RandomDataGenerator extends TemplateUiCommand {
private items: Item[] = [];
private groupItemsBySection(items: Item[]): { [key: string]: Item[] } {
const sectionsMap: { [key: string]: Item[] } = {};
items.forEach((item) => {
if (!sectionsMap[item.section]) {
sectionsMap[item.section] = [];
}
sectionsMap[item.section].push(item);
});
return sectionsMap;
}
private createSections(sectionsMap: {
[key: string]: Item[];
}): List.Section[] {
return Object.entries(sectionsMap).map(
([sectionName, sectionItems]) =>
new List.Section({
title: _.startCase(sectionName),
items: sectionItems.map(
(item: { section: string; id: string; value: string }) => {
const actions = [
new Action.Action({
title: Actions.CopyToClipboard,
value: Actions.CopyToClipboard,
icon: new Icon({
type: IconEnum.Iconify,
value: "material-symbols:copy-all",
}),
}),
new Action.Action({
title: Actions.RefreshValue,
value: Actions.RefreshValue,
icon: new Icon({
type: IconEnum.Iconify,
value: "material-symbols:refresh",
}),
}),
new Action.Action({
title: Actions.PinEntry,
value: Actions.PinEntry,
icon: new Icon({
type: IconEnum.Iconify,
value: "material-symbols:pin",
}),
}),
new Action.Action({
title: Actions.UnpinEntry,
value: Actions.UnpinEntry,
icon: new Icon({
type: IconEnum.Iconify,
value: "material-symbols:unpin",
}),
}),
];
if (isUrl(item.value)) {
actions.unshift(
new Action.Action({
title: Actions.OpenInBrowser,
value: Actions.OpenInBrowser,
icon: new Icon({
type: IconEnum.Iconify,
value: "material-symbols:open-in-new",
}),
})
);
}
return new List.Item({
title: _.startCase(item.id),
value: item.value,
keywords: [item.section],
// subTitle: _.truncate(item.value, {
// length: 20,
// omission: "...",
// }),
icon: new Icon({
type: IconEnum.Iconify,
value: "material-symbols:circle-outline",
}),
defaultAction: Actions.CopyToClipboard,
actions: new Action.ActionPanel({
items: actions,
}),
});
}
),
})
);
}
async load() {
faker.setLocale("en");
this.items = buildItems("", faker);
const sectionsMap = this.groupItemsBySection(this.items);
const sections = this.createSections(sectionsMap);
return ui.setSearchBarPlaceholder("Search...").then(() => {
return ui.render(
new List.List({
sections: sections,
})
);
});
}
onListItemSelected(value: string): Promise<void> {
return clipboard
.writeText(value)
.then(() => {
return toast.success("Copied to clipboard");
})
.catch((err) => {
console.error(err);
return toast.error("Failed to copy to clipboard");
});
}
async onActionSelected(actionValue: string): Promise<void> {
if (this.highlightedListItemValue) {
switch (actionValue) {
case Actions.OpenInBrowser:
return open.url(this.highlightedListItemValue);
case Actions.CopyToClipboard:
return this.onListItemSelected(this.highlightedListItemValue);
default:
break;
}
}
}
onHighlightedListItemChanged(value: string): Promise<void> {
this.highlightedListItemValue = value;
return ui.render(
new List.List({
inherits: ["items", "sections"],
detail: new List.ItemDetail({
children: [new Markdown(this.highlightedListItemValue || "")],
width: 50,
}),
})
);
}
}
expose(new RandomDataGenerator());

60
src/utils/index.ts Normal file
View File

@ -0,0 +1,60 @@
import type { Faker } from "../faker";
import _ from "lodash";
export type Item = {
section: string;
id: string;
value: string;
getValue(): string;
};
export type Pin = (item: Item) => void;
interface FakerListItemProps {
item: Item;
pin?: Pin;
unpin?: Pin;
faker: Faker;
}
const blacklistPaths = [
"locales",
"locale",
"_locale",
"localeFallback",
"_localeFallback",
"definitions",
"fake",
"faker",
"unique",
"helpers",
"mersenne",
"random",
"science",
];
export const buildItems = (path: string, faker: Faker) => {
return _.reduce(
path ? _.get(faker, path) : faker,
(acc: Item[], func, key) => {
if (blacklistPaths.includes(key)) {
return acc;
}
if (_.isFunction(func)) {
const getValue = (): string => {
const value = func();
if (_.isBoolean(value)) return value.toString();
if (!value) return "";
return value.toString();
};
acc.push({ section: path, id: key, value: getValue(), getValue });
} else if (_.isObject(func)) {
acc.push(...buildItems(path ? `${path}.${key}` : key, faker));
}
return acc;
},
[]
);
};

27
tsconfig.json Normal file
View File

@ -0,0 +1,27 @@
{
"compilerOptions": {
// Enable latest features
"lib": [
"ESNext",
"DOM"
],
"target": "ESNext",
"module": "ESNext",
"moduleDetection": "force",
"jsx": "react-jsx",
"allowJs": true,
// Bundler mode
"moduleResolution": "bundler",
"allowImportingTsExtensions": false,
"verbatimModuleSyntax": true,
"noEmit": true,
// Best practices
"strict": true,
"skipLibCheck": true,
"noFallthroughCasesInSwitch": true,
// Some stricter flags (disabled by default)
"noUnusedLocals": false,
"noUnusedParameters": false,
"noPropertyAccessFromIndexSignature": false
}
}