mirror of
https://github.com/kunkunsh/kunkun-ext-ip-info.git
synced 2025-04-03 19:06:42 +00:00
init
This commit is contained in:
commit
dd1f9f4c48
36
.github/workflows/npm-publish.yml
vendored
Normal file
36
.github/workflows/npm-publish.yml
vendored
Normal file
@ -0,0 +1,36 @@
|
||||
# This workflow will run tests using node and then publish a package to GitHub Packages when a release is created
|
||||
# For more information see: https://docs.github.com/en/actions/publishing-packages/publishing-nodejs-packages
|
||||
|
||||
name: NPM Package Publish
|
||||
|
||||
on:
|
||||
release:
|
||||
types: [created]
|
||||
workflow_dispatch:
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- uses: oven-sh/setup-bun@v2
|
||||
- name: Test
|
||||
run: |
|
||||
bun install
|
||||
bun test --coverage
|
||||
|
||||
publish-npm:
|
||||
needs: build
|
||||
runs-on: ubuntu-latest
|
||||
permissions:
|
||||
contents: read
|
||||
id-token: write
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- uses: actions/setup-node@v4
|
||||
with:
|
||||
node-version: 22
|
||||
registry-url: https://registry.npmjs.org/
|
||||
- run: npm publish --provenance --access public
|
||||
env:
|
||||
NODE_AUTH_TOKEN: ${{secrets.NPM_TOKEN}}
|
175
.gitignore
vendored
Normal file
175
.gitignore
vendored
Normal 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
|
3
README.md
Normal file
3
README.md
Normal file
@ -0,0 +1,3 @@
|
||||
# Kunkun Extension ip-info
|
||||
|
||||
Show your own ip info or search for ip info.
|
20
build.ts
Normal file
20
build.ts
Normal file
@ -0,0 +1,20 @@
|
||||
import { watch } from "fs"
|
||||
import { join } from "path"
|
||||
import { refreshTemplateWorkerExtension } from "@kksh/api/dev"
|
||||
import { $ } from "bun"
|
||||
|
||||
async function build() {
|
||||
await $`bun build --minify --target=browser --outdir=./dist ./src/index.ts`
|
||||
await refreshTemplateWorkerExtension()
|
||||
}
|
||||
|
||||
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
demo/ip-info.png
Normal file
BIN
demo/ip-info.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 181 KiB |
51
package.json
Normal file
51
package.json
Normal file
@ -0,0 +1,51 @@
|
||||
{
|
||||
"$schema": "https://schema.kunkun.sh/",
|
||||
"name": "kunkun-ext-ip-info",
|
||||
"version": "0.0.8",
|
||||
"module": "index.ts",
|
||||
"type": "module",
|
||||
"kunkun": {
|
||||
"name": "IP Info",
|
||||
"shortDescription": "Show your current IP Info and Search for IP Address",
|
||||
"longDescription": "Will display you IP Address and more information about this IP address, including Geo Location, ASN, ISP. You can also search for a specific IP.",
|
||||
"identifier": "ip-info",
|
||||
"permissions": [
|
||||
"clipboard:write-text",
|
||||
"fetch:all"
|
||||
],
|
||||
"demoImages": [
|
||||
"./demo/ip-info.png"
|
||||
],
|
||||
"icon": {
|
||||
"type": "iconify",
|
||||
"value": "mdi:ip-network"
|
||||
},
|
||||
"templateUiCmds": [
|
||||
{
|
||||
"name": "IP Info",
|
||||
"main": "dist/index.js",
|
||||
"description": "Display your current IP addres and detailed information. Search for other IP address.",
|
||||
"cmds": []
|
||||
}
|
||||
],
|
||||
"customUiCmds": []
|
||||
},
|
||||
"scripts": {
|
||||
"dev": "bun build.ts dev",
|
||||
"build": "bun build.ts"
|
||||
},
|
||||
"files": [
|
||||
"dist"
|
||||
],
|
||||
"dependencies": {
|
||||
"@kksh/api": "^0.0.52"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/bun": "latest",
|
||||
"valibot": "1.0.0-beta.12"
|
||||
},
|
||||
"peerDependencies": {
|
||||
"typescript": "^5.7.3"
|
||||
},
|
||||
"packageManager": "pnpm@9.15.4"
|
||||
}
|
2074
pnpm-lock.yaml
generated
Normal file
2074
pnpm-lock.yaml
generated
Normal file
File diff suppressed because it is too large
Load Diff
242
src/index.ts
Normal file
242
src/index.ts
Normal file
@ -0,0 +1,242 @@
|
||||
import {
|
||||
Action,
|
||||
clipboard,
|
||||
expose,
|
||||
fetch,
|
||||
Icon,
|
||||
IconEnum,
|
||||
List,
|
||||
log,
|
||||
toast,
|
||||
ui,
|
||||
WorkerExtension
|
||||
} from "@kksh/api/ui/worker"
|
||||
import { boolean, number, object, parse, safeParse, string, type InferOutput } from "valibot"
|
||||
|
||||
const IpApiJsonSchema = object({
|
||||
status: string(),
|
||||
continent: string(),
|
||||
country: string(),
|
||||
countryCode: string(),
|
||||
region: string(),
|
||||
regionName: string(),
|
||||
city: string(),
|
||||
district: string(),
|
||||
zip: string(),
|
||||
lat: number(),
|
||||
lon: number(),
|
||||
timezone: string(),
|
||||
offset: number(),
|
||||
currency: string(),
|
||||
isp: string(),
|
||||
org: string(),
|
||||
as: string(),
|
||||
asname: string(),
|
||||
reverse: string(),
|
||||
proxy: boolean(),
|
||||
hosting: boolean(),
|
||||
query: string()
|
||||
})
|
||||
|
||||
type IpListItem = {
|
||||
title: string
|
||||
value: string
|
||||
icon: string
|
||||
}
|
||||
|
||||
const Actions = {
|
||||
CopyToClipboard: "Copy to Clipboard"
|
||||
}
|
||||
|
||||
function mapIpInfoToListItem(ip: IpListItem): List.Item {
|
||||
return new List.Item({
|
||||
title: ip.title,
|
||||
value: ip.value,
|
||||
subTitle: ip.value,
|
||||
icon: new Icon({ type: IconEnum.Iconify, value: ip.icon }),
|
||||
keywords: [ip.title],
|
||||
defaultAction: "Copy",
|
||||
actions: new Action.ActionPanel({
|
||||
items: [
|
||||
new Action.Action({
|
||||
title: Actions.CopyToClipboard,
|
||||
icon: new Icon({ type: IconEnum.Iconify, value: "tabler:copy" }),
|
||||
value: Actions.CopyToClipboard
|
||||
})
|
||||
]
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
class IpInfo extends WorkerExtension {
|
||||
ip?: InferOutput<typeof IpApiJsonSchema>
|
||||
listitems: List.Item[] = []
|
||||
|
||||
onEnterPressedOnSearchBar(): Promise<void> {
|
||||
// check if this.searchTerm is ipv4
|
||||
if (!/^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$/.test(this.searchTerm)) {
|
||||
return Promise.resolve()
|
||||
}
|
||||
return this.searchIp(this.searchTerm).then(() => {
|
||||
return ui.setSearchTerm("")
|
||||
})
|
||||
}
|
||||
|
||||
async load(): Promise<void> {
|
||||
return ui
|
||||
.setSearchBarPlaceholder("Enter an IPv4 address, and press enter to search")
|
||||
.then(() => ui.render(new List.List({ items: [], defaultAction: "Search or Copy" })))
|
||||
.then(() => this.searchIp())
|
||||
}
|
||||
|
||||
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")
|
||||
})
|
||||
}
|
||||
onActionSelected(value: string): Promise<void> {
|
||||
if (value === Actions.CopyToClipboard) {
|
||||
if (this.highlightedListItemValue) {
|
||||
return this.onListItemSelected(this.highlightedListItemValue)
|
||||
} else {
|
||||
return toast.warning("No item selected").then(() => {})
|
||||
}
|
||||
}
|
||||
return Promise.resolve()
|
||||
}
|
||||
|
||||
async searchIp(ipv4?: string) {
|
||||
const apiUrl = `http://ip-api.com/json${
|
||||
ipv4 ? "/" + ipv4 : ""
|
||||
}?fields=status,message,continent,country,countryCode,region,regionName,city,district,zip,lat,lon,timezone,offset,currency,isp,org,as,asname,reverse,proxy,hosting,query`
|
||||
|
||||
return fetch(apiUrl, {
|
||||
method: "GET"
|
||||
})
|
||||
.then((res) => res.json())
|
||||
.then((data) => {
|
||||
this.ip = parse(IpApiJsonSchema, data)
|
||||
const items: IpListItem[] = [
|
||||
{
|
||||
icon: "mdi:web",
|
||||
title: "Public IPv4",
|
||||
value: this.ip.query
|
||||
},
|
||||
{
|
||||
icon: "mdi:earth",
|
||||
title: "Continent",
|
||||
value: this.ip.continent
|
||||
},
|
||||
{
|
||||
icon: "gis:search-country",
|
||||
title: "Country",
|
||||
value: this.ip.country
|
||||
},
|
||||
{
|
||||
icon: "ic:baseline-code",
|
||||
title: "Country Code",
|
||||
value: this.ip.countryCode
|
||||
},
|
||||
{
|
||||
icon: "mdi:web",
|
||||
title: "Region",
|
||||
value: this.ip.region
|
||||
},
|
||||
{
|
||||
icon: "oui:vis-map-region",
|
||||
title: "Region Name",
|
||||
value: this.ip.regionName
|
||||
},
|
||||
{
|
||||
icon: "solar:city-bold",
|
||||
title: "City",
|
||||
value: this.ip.city
|
||||
},
|
||||
{
|
||||
icon: "oui:vis-map-region",
|
||||
title: "District",
|
||||
value: this.ip.district
|
||||
},
|
||||
{
|
||||
icon: "tabler:zip",
|
||||
title: "ZIP",
|
||||
value: this.ip.zip
|
||||
},
|
||||
{
|
||||
icon: "tabler:zip",
|
||||
title: "Geo Coordinates",
|
||||
value: `${this.ip.lat}, ${this.ip.lon}`
|
||||
},
|
||||
{
|
||||
icon: "mdi:timezone",
|
||||
title: "Timezone",
|
||||
value: this.ip.timezone
|
||||
},
|
||||
{
|
||||
icon: "mdi:clock",
|
||||
title: "Offset",
|
||||
value: `${this.ip.offset / 3600} hr`
|
||||
},
|
||||
{
|
||||
icon: "simple-icons:bitcoinsv",
|
||||
title: "Currency",
|
||||
value: this.ip.currency
|
||||
},
|
||||
{
|
||||
icon: "carbon:container-services",
|
||||
title: "ISP",
|
||||
value: this.ip.isp
|
||||
},
|
||||
{
|
||||
icon: "clarity:organization-solid",
|
||||
title: "Org",
|
||||
value: this.ip.org
|
||||
},
|
||||
{
|
||||
icon: "carbon:container-services",
|
||||
title: "AS Number",
|
||||
value: this.ip.as
|
||||
},
|
||||
{
|
||||
icon: "carbon:container-services",
|
||||
title: "AS Name",
|
||||
value: this.ip.asname
|
||||
},
|
||||
{
|
||||
icon: "material-symbols:dns",
|
||||
title: "Reverse DNS",
|
||||
value: this.ip.reverse
|
||||
},
|
||||
{
|
||||
icon: "mdi:proxy",
|
||||
title: "Proxy, VPN or Tor exit Address",
|
||||
value: this.ip.proxy.toString()
|
||||
},
|
||||
{
|
||||
icon: "clarity:host-solid",
|
||||
title: "Hosting, colocated or data center",
|
||||
value: this.ip.hosting.toString()
|
||||
}
|
||||
]
|
||||
this.listitems = items.map(mapIpInfoToListItem)
|
||||
return ui.render(
|
||||
new List.List({
|
||||
items: this.listitems,
|
||||
defaultAction: "Search or Copy"
|
||||
})
|
||||
)
|
||||
})
|
||||
.catch((error) => {
|
||||
console.error(error)
|
||||
toast.error("Failed to fetch IP info")
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
expose(new IpInfo())
|
27
tsconfig.json
Normal file
27
tsconfig.json
Normal 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": true,
|
||||
"verbatimModuleSyntax": true,
|
||||
"noEmit": true,
|
||||
|
||||
// Best practices
|
||||
"strict": true,
|
||||
"skipLibCheck": true,
|
||||
"noFallthroughCasesInSwitch": true,
|
||||
|
||||
// Some stricter flags (disabled by default)
|
||||
"noUnusedLocals": false,
|
||||
"noUnusedParameters": false,
|
||||
"noPropertyAccessFromIndexSignature": false
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user