This commit is contained in:
Huakun Shen 2025-01-18 04:24:27 -05:00
commit 615b182421
No known key found for this signature in database
9 changed files with 469 additions and 0 deletions

25
.github/workflows/jsr-publish.yml vendored Normal file
View File

@ -0,0 +1,25 @@
name: Publish
on:
push:
branches:
- main
jobs:
publish:
runs-on: ubuntu-latest
permissions:
contents: read
id-token: write
steps:
- uses: actions/checkout@v4
- uses: oven-sh/setup-bun@v2
with:
bun-version: latest
- name: Publish package
run: |
bun install
bun run build
bunx jsr publish --allow-slow-types

175
.gitignore vendored Normal file
View File

@ -0,0 +1,175 @@
# 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

24
README.md Normal file
View File

@ -0,0 +1,24 @@
# Wifi Password Share
## Platforms
- MacOS
```bash
networksetup -listpreferredwirelessnetworks en0
security find-generic-password -D "AirPort network password" -a $SSID -w
```
- Windows
```powershell
$_, $ssid = ((netsh wlan show interface | findstr "Profile" | findstr /v "mode") -split ":",2).trim();
$_, $pass = ((netsh wlan show profile name=$ssid key=clear | findstr Key) -split ":").trim();
```
- Linux
```bash
ssid=$(nmcli device wifi show-password | grep SSID | cut -d : -f 2 | xargs)
password=$(nmcli device wifi show-password | grep Password | cut -d : -f 2 | xargs)
```
## Features
- Reveal Wifi Password from current machine
- Generate Connection QR Code, mobile devices can scan to connect directly

24
build.ts Normal file
View File

@ -0,0 +1,24 @@
import { watch } from "fs"
import { join } from "path"
import { refreshTemplateWorkerExtension } from "@kksh/api/dev"
import { $ } from "bun"
async function build() {
try {
await $`bun build --minify --target=browser --outdir=./dist ./src/index.ts`
await refreshTemplateWorkerExtension()
} 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()
})
}

BIN
bun.lockb Executable file

Binary file not shown.

1
index.ts Normal file
View File

@ -0,0 +1 @@
console.log("Hello via Bun!");

76
package.json Normal file
View File

@ -0,0 +1,76 @@
{
"$schema": "https://schema.kunkun.sh",
"name": "kunkun-ext-wifi-password",
"version": "0.0.2",
"type": "module",
"kunkun": {
"name": "Wifi Password",
"shortDescription": "Show Wifi Password",
"longDescription": "Show Wifi Password",
"identifier": "wifi-password",
"permissions": [
"clipboard:write-text",
{
"permission": "shell:execute",
"allow": [
{
"cmd": {
"program": "networksetup",
"args": [
"-listpreferredwirelessnetworks",
"en0"
]
}
},
{
"cmd": {
"program": "security",
"args": [
"find-generic-password",
"-D",
"AirPort network password",
"-a",
".+",
"-w"
]
}
}
]
}
],
"demoImages": [],
"icon": {
"type": "iconify",
"value": "material-symbols:extension"
},
"customUiCmds": [],
"templateUiCmds": [
{
"name": "Show Wifi Password",
"main": "dist/index.js",
"cmds": []
}
]
},
"scripts": {
"dev": "bun build.ts dev",
"build": "bun build.ts"
},
"dependencies": {
"@kksh/api": "^0.0.52",
"qrcode": "^1.5.4"
},
"devDependencies": {
"@types/bun": "latest",
"@types/qrcode": "^1.5.5"
},
"peerDependencies": {
"typescript": "^5.0.0"
},
"files": [
"./dist",
".gitignore"
],
"packageManager": "pnpm@9.15.3+sha512.1f79bc245a66eb0b07c5d4d83131240774642caaa86ef7d0434ab47c0d16f66b04e21e0c086eb61e62c77efc4d7f7ec071afad3796af64892fae66509173893a",
"module": "index.ts"
}

117
src/index.ts Normal file
View File

@ -0,0 +1,117 @@
import {
Action,
clipboard,
expose,
Form,
fs,
Icon,
IconEnum,
List,
Markdown,
os,
path,
shell,
toast,
ui,
WorkerExtension
} from "@kksh/api/ui/worker"
import qrcode from "qrcode"
class ExtensionTemplate extends WorkerExtension {
networks: string[] = []
get listItems() {
return this.networks.map(
(x) =>
new List.Item({
title: x,
value: x,
icon: new Icon({
type: IconEnum.Iconify,
value: "mdi:wifi"
})
})
)
}
async load() {
ui.setSearchBarPlaceholder("Search for wifi and press enter to get password")
const platform = await os.platform()
if (platform === "macos") {
const cmd = shell.createCommand("networksetup", ["-listpreferredwirelessnetworks", "en0"])
const result = await cmd.execute()
if (result.code !== 0) {
toast.error("Failed to get wifi password")
return
}
this.networks = result.stdout
.trim()
.split("\n")
.slice(1)
.map((x) => x.trim())
}
return ui.render(
new List.List({
items: this.listItems
})
)
}
async onListItemSelected(ssid: string): Promise<void> {
ui.render(
new List.List({
inherits: ["items"],
detail: undefined
})
)
const cmd = shell.createCommand("security", [
"find-generic-password",
"-D",
`AirPort network password`,
"-a",
ssid,
"-w"
])
const result = await cmd.execute()
if (result.code !== 0) {
console.log(result.stderr)
toast.error("Failed to get wifi password")
return
}
const wifiPassword = result.stdout.trim()
toast.success(`Wifi password: ${wifiPassword}, written to clipboard`)
clipboard.writeText(wifiPassword)
const wifiConnectUrl = `WIFI:S:${ssid};T:WPA;P:${wifiPassword};;`
let qrcodeSvg: string | undefined
try {
qrcodeSvg = await qrcode.toString(wifiConnectUrl)
} catch (error) {
toast.error(`Failed to generate QR code: ${error}`)
console.error(error)
}
// <img src="${qrcodeImageBase64}" style="display: block; margin: 0 auto;" />
if (!qrcodeSvg) {
return toast.error("Failed to generate QR code")
}
const markdown = `
<div style="display: flex; flex-direction: column; justify-content: center; align-items: center; padding: 2em;">
<div style="width: 10em; height: 10em; display: block;">
${qrcodeSvg}
</div>
<p style="text-align: center;">Wifi Password: <strong>${wifiPassword}</strong></p>
<p style="text-align: center;">Connect URL: <strong>${wifiConnectUrl}</strong></p>
</div>
`
return ui.render(
new List.List({
inherits: ["items"],
detail: new List.ItemDetail({
width: 60,
children: [new Markdown(markdown)]
})
})
)
}
}
expose(new ExtensionTemplate())

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
}
}