mirror of
https://github.com/kunkunsh/kunkun-ext-disk-speed.git
synced 2025-04-03 18:56:44 +00:00
init
This commit is contained in:
commit
a3c05ef042
48
.github/workflows/npm-publish.yml
vendored
Normal file
48
.github/workflows/npm-publish.yml
vendored
Normal file
@ -0,0 +1,48 @@
|
||||
# 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:
|
||||
push:
|
||||
branches: [main]
|
||||
release:
|
||||
types: [created]
|
||||
workflow_dispatch:
|
||||
|
||||
jobs:
|
||||
publish-npm:
|
||||
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/
|
||||
|
||||
- uses: pnpm/action-setup@v2
|
||||
with:
|
||||
version: latest
|
||||
- uses: oven-sh/setup-bun@v2
|
||||
- run: pnpm install
|
||||
- run: pnpm build
|
||||
- run: |
|
||||
PACKAGE_NAME=$(jq -r '.name' package.json)
|
||||
PACKAGE_VERSION=$(jq -r '.version' package.json)
|
||||
|
||||
# Get the version from npm registry
|
||||
REGISTRY_VERSION=$(npm show "$PACKAGE_NAME" version)
|
||||
|
||||
# Compare versions
|
||||
if [ "$PACKAGE_VERSION" == "$REGISTRY_VERSION" ]; then
|
||||
echo "Version $PACKAGE_VERSION already exists in the npm registry."
|
||||
exit 0
|
||||
else
|
||||
echo "Version $PACKAGE_VERSION does not exist in the npm registry. Proceeding..."
|
||||
npm publish --provenance --access public
|
||||
fi
|
||||
env:
|
||||
NODE_AUTH_TOKEN: ${{secrets.NPM_TOKEN}}
|
25
.gitignore
vendored
Normal file
25
.gitignore
vendored
Normal file
@ -0,0 +1,25 @@
|
||||
node_modules
|
||||
|
||||
# Output
|
||||
.output
|
||||
.vercel
|
||||
/.svelte-kit
|
||||
/build
|
||||
|
||||
# OS
|
||||
.DS_Store
|
||||
Thumbs.db
|
||||
|
||||
# Env
|
||||
.env
|
||||
.env.*
|
||||
!.env.example
|
||||
!.env.test
|
||||
|
||||
# Vite
|
||||
vite.config.js.timestamp-*
|
||||
vite.config.ts.timestamp-*
|
||||
extensions_support/
|
||||
|
||||
.pnpm-store
|
||||
dist/
|
47
README.md
Normal file
47
README.md
Normal file
@ -0,0 +1,47 @@
|
||||
# Svelte + TS + Vite
|
||||
|
||||
This template should help get you started developing with Svelte and TypeScript in Vite.
|
||||
|
||||
## Recommended IDE Setup
|
||||
|
||||
[VS Code](https://code.visualstudio.com/) + [Svelte](https://marketplace.visualstudio.com/items?itemName=svelte.svelte-vscode).
|
||||
|
||||
## Need an official Svelte framework?
|
||||
|
||||
Check out [SvelteKit](https://github.com/sveltejs/kit#readme), which is also powered by Vite. Deploy anywhere with its serverless-first approach and adapt to various platforms, with out of the box support for TypeScript, SCSS, and Less, and easily-added support for mdsvex, GraphQL, PostCSS, Tailwind CSS, and more.
|
||||
|
||||
## Technical considerations
|
||||
|
||||
**Why use this over SvelteKit?**
|
||||
|
||||
- It brings its own routing solution which might not be preferable for some users.
|
||||
- It is first and foremost a framework that just happens to use Vite under the hood, not a Vite app.
|
||||
|
||||
This template contains as little as possible to get started with Vite + TypeScript + Svelte, while taking into account the developer experience with regards to HMR and intellisense. It demonstrates capabilities on par with the other `create-vite` templates and is a good starting point for beginners dipping their toes into a Vite + Svelte project.
|
||||
|
||||
Should you later need the extended capabilities and extensibility provided by SvelteKit, the template has been structured similarly to SvelteKit so that it is easy to migrate.
|
||||
|
||||
**Why `global.d.ts` instead of `compilerOptions.types` inside `jsconfig.json` or `tsconfig.json`?**
|
||||
|
||||
Setting `compilerOptions.types` shuts out all other types not explicitly listed in the configuration. Using triple-slash references keeps the default TypeScript setting of accepting type information from the entire workspace, while also adding `svelte` and `vite/client` type information.
|
||||
|
||||
**Why include `.vscode/extensions.json`?**
|
||||
|
||||
Other templates indirectly recommend extensions via the README, but this file allows VS Code to prompt the user to install the recommended extension upon opening the project.
|
||||
|
||||
**Why enable `allowJs` in the TS template?**
|
||||
|
||||
While `allowJs: false` would indeed prevent the use of `.js` files in the project, it does not prevent the use of JavaScript syntax in `.svelte` files. In addition, it would force `checkJs: false`, bringing the worst of both worlds: not being able to guarantee the entire codebase is TypeScript, and also having worse typechecking for the existing JavaScript. In addition, there are valid use cases in which a mixed codebase may be relevant.
|
||||
|
||||
**Why is HMR not preserving my local component state?**
|
||||
|
||||
HMR state preservation comes with a number of gotchas! It has been disabled by default in both `svelte-hmr` and `@sveltejs/vite-plugin-svelte` due to its often surprising behavior. You can read the details [here](https://github.com/rixo/svelte-hmr#svelte-hmr).
|
||||
|
||||
If you have state that's important to retain within a component, consider creating an external store which would not be replaced by HMR.
|
||||
|
||||
```ts
|
||||
// store.ts
|
||||
// An extremely simple external store
|
||||
import { writable } from 'svelte/store'
|
||||
export default writable(0)
|
||||
```
|
14
components.json
Normal file
14
components.json
Normal file
@ -0,0 +1,14 @@
|
||||
{
|
||||
"$schema": "https://shadcn-svelte.com/schema.json",
|
||||
"style": "new-york",
|
||||
"tailwind": {
|
||||
"config": "tailwind.config.ts",
|
||||
"css": "src/app.css",
|
||||
"baseColor": "neutral"
|
||||
},
|
||||
"aliases": {
|
||||
"components": "$lib/components",
|
||||
"utils": "$lib/utils"
|
||||
},
|
||||
"typescript": true
|
||||
}
|
8
deno-src/deno.json
Normal file
8
deno-src/deno.json
Normal file
@ -0,0 +1,8 @@
|
||||
{
|
||||
"tasks": {
|
||||
"dev": "deno run --watch main.ts"
|
||||
},
|
||||
"imports": {
|
||||
"@kunkun/api": "jsr:@kunkun/api@^0.0.52"
|
||||
}
|
||||
}
|
45
deno-src/deno.lock
generated
Normal file
45
deno-src/deno.lock
generated
Normal file
@ -0,0 +1,45 @@
|
||||
{
|
||||
"version": "4",
|
||||
"specifiers": {
|
||||
"jsr:@kunkun/api@^0.0.39": "0.0.39",
|
||||
"npm:@types/node@*": "22.5.4",
|
||||
"npm:kkrpc@^0.0.10": "0.0.10_typescript@5.6.3"
|
||||
},
|
||||
"jsr": {
|
||||
"@kunkun/api@0.0.39": {
|
||||
"integrity": "af1f0728083a6553279a4a7ce12ca83a6affe7dcda09b041376934e6c26e979e",
|
||||
"dependencies": [
|
||||
"npm:kkrpc"
|
||||
]
|
||||
}
|
||||
},
|
||||
"npm": {
|
||||
"@types/node@22.5.4": {
|
||||
"integrity": "sha512-FDuKUJQm/ju9fT/SeX/6+gBzoPzlVCzfzmGkwKvRHQVxi4BntVbyIwf6a4Xn62mrvndLiml6z/UBXIdEVjQLXg==",
|
||||
"dependencies": [
|
||||
"undici-types"
|
||||
]
|
||||
},
|
||||
"kkrpc@0.0.10_typescript@5.6.3": {
|
||||
"integrity": "sha512-lkQKVnN9f6JrS4ybKbGkV4mtuGhWYLTnaWx60ysytEap+sP5jcTbAuJlSrY6JqlwaohiS0X3ZbvJ2rAXYRdTng==",
|
||||
"dependencies": [
|
||||
"typescript",
|
||||
"ws"
|
||||
]
|
||||
},
|
||||
"typescript@5.6.3": {
|
||||
"integrity": "sha512-hjcS1mhfuyi4WW8IWtjP7brDrG2cuDZukyrYrSauoXGNgx0S7zceP07adYkJycEr56BOUTNPzbInooiN3fn1qw=="
|
||||
},
|
||||
"undici-types@6.19.8": {
|
||||
"integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw=="
|
||||
},
|
||||
"ws@8.18.0": {
|
||||
"integrity": "sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw=="
|
||||
}
|
||||
},
|
||||
"workspace": {
|
||||
"dependencies": [
|
||||
"jsr:@kunkun/api@^0.0.39"
|
||||
]
|
||||
}
|
||||
}
|
0
deno-src/dev.ts
Normal file
0
deno-src/dev.ts
Normal file
89
deno-src/lib.ts
Normal file
89
deno-src/lib.ts
Normal file
@ -0,0 +1,89 @@
|
||||
import { expose } from "@kunkun/api/runtime/deno"
|
||||
import { API, Progress } from "../src/types.ts"
|
||||
|
||||
const oneMB = 1024 * 1024
|
||||
|
||||
export async function sequentialWriteTest(
|
||||
options: {
|
||||
filePath: string
|
||||
sizeInMB: number
|
||||
rounds: number
|
||||
bufferSizeMB: number
|
||||
keepTheFile?: boolean
|
||||
},
|
||||
callback?: (progress: Progress) => void
|
||||
): Promise<Progress> {
|
||||
const { filePath, sizeInMB, rounds, bufferSizeMB } = options
|
||||
const data = new Uint8Array(bufferSizeMB * oneMB) // 1MB buffer
|
||||
let start = performance.now()
|
||||
let totalMB = 0
|
||||
let totalDuration = 0
|
||||
for (let round = 0; round < rounds; round++) {
|
||||
const file = await Deno.open(filePath, { write: true, create: true })
|
||||
const writer = file.writable.getWriter()
|
||||
|
||||
start = performance.now()
|
||||
for (let i = 0; i < Math.floor(sizeInMB / bufferSizeMB); i++) {
|
||||
await writer.write(data)
|
||||
totalMB += bufferSizeMB
|
||||
}
|
||||
const roundEnd = performance.now()
|
||||
totalDuration += (roundEnd - start) / 1000
|
||||
callback?.({ totalMB, totalDuration })
|
||||
await writer.close()
|
||||
// if keepTheFile, do not remove the file in the last round
|
||||
const isLastRound = round === rounds - 1
|
||||
if (!isLastRound && !options.keepTheFile) {
|
||||
Deno.removeSync(filePath)
|
||||
}
|
||||
}
|
||||
|
||||
return { totalDuration, totalMB }
|
||||
}
|
||||
|
||||
export async function createEmptyFile(filePath: string, sizeInMB: number): Promise<void> {
|
||||
if (await fileExists(filePath)) {
|
||||
await Deno.remove(filePath)
|
||||
}
|
||||
const file = await Deno.open(filePath, { write: true, create: true })
|
||||
const writer = file.writable.getWriter()
|
||||
for (let i = 0; i < sizeInMB; i++) {
|
||||
await writer.write(new Uint8Array(oneMB))
|
||||
}
|
||||
await writer.close()
|
||||
}
|
||||
|
||||
// Sequential Read
|
||||
export async function sequentialReadTest(
|
||||
filePath: string,
|
||||
options: { deleteAfter: boolean } = { deleteAfter: true }
|
||||
): Promise<Progress> {
|
||||
const file = await Deno.open(filePath, { read: true })
|
||||
const buffer = new Uint8Array(oneMB) // 1MB buffer
|
||||
const start = performance.now()
|
||||
let totalMB = 0
|
||||
while ((await file.read(buffer)) !== null) {
|
||||
totalMB += 1
|
||||
}
|
||||
const totalDuration = (performance.now() - start) / 1000
|
||||
file.close()
|
||||
if (options.deleteAfter) {
|
||||
Deno.removeSync(filePath)
|
||||
}
|
||||
return { totalMB, totalDuration }
|
||||
}
|
||||
|
||||
export function fileExists(filePath: string): boolean {
|
||||
try {
|
||||
Deno.statSync(filePath)
|
||||
return true
|
||||
} catch (_error) {
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
expose({
|
||||
sequentialWriteTest,
|
||||
sequentialReadTest,
|
||||
createEmptyFile
|
||||
} satisfies API)
|
97
deno-src/random.ts
Normal file
97
deno-src/random.ts
Normal file
@ -0,0 +1,97 @@
|
||||
import { parseArgs } from "jsr:@std/cli/parse-args"
|
||||
import { DiskSpeedTestInput, DiskSpeedTestOutput } from "../src/model.ts"
|
||||
|
||||
const args = parseArgs(Deno.args)
|
||||
|
||||
if (args._.length !== 1) {
|
||||
console.error("Missing Arguments")
|
||||
Deno.exit(1)
|
||||
}
|
||||
const encodedArgs = args._[0]
|
||||
const base64Decoded = atob(encodedArgs as string)
|
||||
|
||||
const decodedJsonArgs: DiskSpeedTestInput = JSON.parse(base64Decoded)
|
||||
|
||||
// Pre-fill file with zeros to a given size (in MB)
|
||||
async function initializeFile(filePath: string, sizeInMB: number) {
|
||||
const file = await Deno.open(filePath, { write: true, create: true })
|
||||
const data = new Uint8Array(1024 * 1024) // 1MB buffer filled with zeros
|
||||
const start = performance.now()
|
||||
const writer = file.writable.getWriter()
|
||||
for (let i = 0; i < sizeInMB; i++) {
|
||||
await writer.write(data)
|
||||
}
|
||||
|
||||
file.close()
|
||||
return (performance.now() - start) / 1000
|
||||
// console.log(`File Initialization: ${sizeInMB}MB took ${duration.toFixed(3)} seconds`)
|
||||
}
|
||||
|
||||
// Random Write
|
||||
async function randomWrite(filePath: string, iterations: number, blockSize: number) {
|
||||
const file = await Deno.open(filePath, { write: true, create: true })
|
||||
const fileSize = (await Deno.stat(filePath)).size
|
||||
const data = new Uint8Array(blockSize)
|
||||
const start = performance.now()
|
||||
const writer = file.writable.getWriter()
|
||||
const totalDataMB = (iterations * blockSize) / (1024 * 1024) // Total data in MB
|
||||
|
||||
for (let i = 0; i < iterations; i++) {
|
||||
const offset = Math.floor(Math.random() * (fileSize - blockSize))
|
||||
await file.seek(offset, Deno.SeekMode.Start)
|
||||
await writer.write(data)
|
||||
}
|
||||
|
||||
file.close()
|
||||
const duration = (performance.now() - start) / 1000
|
||||
return totalDataMB / duration
|
||||
|
||||
// const speed = totalDataMB / duration // Speed in MB/s
|
||||
// console.log(
|
||||
// `Random Write: ${iterations} iterations (${totalDataMB.toFixed(
|
||||
// 3
|
||||
// )}MB) took ${duration.toFixed(3)} seconds`
|
||||
// )
|
||||
// console.log(`Write Speed: ${speed.toFixed(3)} MB/s`)
|
||||
}
|
||||
|
||||
// Random Read
|
||||
async function randomRead(filePath: string, iterations: number, blockSize: number) {
|
||||
const file = await Deno.open(filePath, { read: true })
|
||||
const fileSize = (await Deno.stat(filePath)).size
|
||||
const buffer = new Uint8Array(blockSize)
|
||||
const start = performance.now()
|
||||
const totalDataMB = (iterations * blockSize) / (1024 * 1024) // Total data in MB
|
||||
|
||||
for (let i = 0; i < iterations; i++) {
|
||||
const offset = Math.floor(Math.random() * (fileSize - blockSize))
|
||||
await file.seek(offset, Deno.SeekMode.Start)
|
||||
await file.read(buffer)
|
||||
}
|
||||
|
||||
file.close()
|
||||
const duration = (performance.now() - start) / 1000
|
||||
const speed = totalDataMB / duration
|
||||
return speed
|
||||
// console.log(
|
||||
// `Random Read: ${iterations} iterations (${totalDataMB.toFixed(
|
||||
// 3
|
||||
// )}MB) took ${duration.toFixed(3)} seconds`
|
||||
// )
|
||||
// console.log(`Read Speed: ${speed.toFixed(3)} MB/s`)
|
||||
}
|
||||
|
||||
// Example Usage
|
||||
await initializeFile("./testfile.dat", 1000) // Pre-fill the file with 100MB of data
|
||||
const writeSpeed = await randomWrite("./testfile.dat", 10000, 4096) // Perform 1000 random writes (4KB blocks)
|
||||
const readSpeed = await randomRead("./testfile.dat", 10000, 4096) // Perform 1000 random reads (4KB blocks)
|
||||
|
||||
const output: DiskSpeedTestOutput = {
|
||||
writeSpeedMBps: writeSpeed,
|
||||
readSpeedMBps: readSpeed
|
||||
}
|
||||
|
||||
console.log(JSON.stringify(output))
|
||||
|
||||
// remove the file
|
||||
await Deno.remove("./testfile.dat")
|
9
deno.json
Normal file
9
deno.json
Normal file
@ -0,0 +1,9 @@
|
||||
{
|
||||
"tasks": {
|
||||
"dev": "deno run --watch main.ts"
|
||||
},
|
||||
"imports": {
|
||||
"@std/assert": "jsr:@std/assert@1",
|
||||
"@valibot/valibot": "jsr:@valibot/valibot@^0.42.1"
|
||||
}
|
||||
}
|
31
dev.ts
Normal file
31
dev.ts
Normal file
@ -0,0 +1,31 @@
|
||||
import { $ } from "bun"
|
||||
import { DiskSpeedTestInput } from "./src/model.ts"
|
||||
|
||||
const input: DiskSpeedTestInput = {
|
||||
targetPath: "./testfile.dat",
|
||||
sequential: {
|
||||
stressFileSizeMB: 2000
|
||||
},
|
||||
random: {
|
||||
stressFileSizeMB: 1000,
|
||||
iterations: 1000,
|
||||
blockSize: 4096
|
||||
}
|
||||
}
|
||||
const encoded = btoa(JSON.stringify(input))
|
||||
// sequential
|
||||
;(async () => {
|
||||
const res =
|
||||
await $`deno run --allow-read --allow-write deno-scripts/sequential.ts ${encoded}`.quiet()
|
||||
const stdoutSplit = res.stdout.toString("utf-8").split("\n")
|
||||
console.log(JSON.parse(stdoutSplit[stdoutSplit.length - 2]))
|
||||
})()
|
||||
|
||||
// random
|
||||
;(async () => {
|
||||
const res = await $`deno run --allow-read --allow-write deno-scripts/random.ts ${encoded}`.quiet()
|
||||
console.log("stdout", res.stdout.toString("utf-8"))
|
||||
|
||||
const stdoutSplit = res.stdout.toString("utf-8").split("\n")
|
||||
console.log(JSON.parse(stdoutSplit[stdoutSplit.length - 2]))
|
||||
})()
|
13
index.html
Normal file
13
index.html
Normal file
@ -0,0 +1,13 @@
|
||||
<!doctype html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>Vite + Svelte + TS</title>
|
||||
</head>
|
||||
<body>
|
||||
<div id="app"></div>
|
||||
<script type="module" src="/src/main.ts"></script>
|
||||
</body>
|
||||
</html>
|
95
package.json
Normal file
95
package.json
Normal file
@ -0,0 +1,95 @@
|
||||
{
|
||||
"$schema": "https://schema.kunkun.sh",
|
||||
"name": "kunkun-ext-disk-speed",
|
||||
"license": "MIT",
|
||||
"repository": "https://github.com/kunkunsh/kunkun-ext-disk-speed",
|
||||
"version": "0.0.3",
|
||||
"type": "module",
|
||||
"kunkun": {
|
||||
"name": "Disk Speed",
|
||||
"shortDescription": "Test the speed of your disk",
|
||||
"longDescription": "Test the speed of your disk",
|
||||
"identifier": "disk-speed",
|
||||
"icon": {
|
||||
"type": "iconify",
|
||||
"value": "carbon:meter"
|
||||
},
|
||||
"demoImages": [],
|
||||
"permissions": [
|
||||
"dialog:all",
|
||||
{
|
||||
"permission": "open:folder",
|
||||
"allow": [
|
||||
{
|
||||
"path": "**"
|
||||
}
|
||||
]
|
||||
},
|
||||
{
|
||||
"permission": "shell:deno:spawn",
|
||||
"allow": [
|
||||
{
|
||||
"path": "$EXTENSION/deno-src/lib.ts",
|
||||
"read": "*",
|
||||
"write": "*"
|
||||
}
|
||||
]
|
||||
},
|
||||
"shell:stdin-write",
|
||||
"shell:kill"
|
||||
],
|
||||
"customUiCmds": [
|
||||
{
|
||||
"main": "/",
|
||||
"dist": "dist",
|
||||
"devMain": "http://localhost:5173",
|
||||
"name": "Disk Speed",
|
||||
"window": {
|
||||
"hiddenTitle": true,
|
||||
"titleBarStyle": "overlay"
|
||||
},
|
||||
"cmds": []
|
||||
}
|
||||
],
|
||||
"templateUiCmds": []
|
||||
},
|
||||
"scripts": {
|
||||
"dev": "vite",
|
||||
"build": "vite build",
|
||||
"preview": "vite preview",
|
||||
"check": "svelte-check --tsconfig ./tsconfig.json && tsc -p tsconfig.node.json"
|
||||
},
|
||||
"dependencies": {
|
||||
"@iconify/svelte": "^4.0.2",
|
||||
"@kksh/api": "^0.0.52",
|
||||
"@kksh/svelte": "0.1.7",
|
||||
"bits-ui": "^0.21.16",
|
||||
"clsx": "^2.1.1",
|
||||
"echarts": "^5.5.1",
|
||||
"lucide-svelte": "^0.416.0",
|
||||
"tailwind-merge": "^2.4.0",
|
||||
"tailwind-variants": "^0.2.1",
|
||||
"valibot": "0.40.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@sveltejs/vite-plugin-svelte": "^4.0.0",
|
||||
"@tsconfig/svelte": "^5.0.4",
|
||||
"@types/bun": "^1.1.10",
|
||||
"autoprefixer": "^10.4.19",
|
||||
"postcss": "^8.4.38",
|
||||
"svelte": "^5.0.3",
|
||||
"svelte-check": "^4.0.5",
|
||||
"tailwindcss": "^3.4.4",
|
||||
"tslib": "^2.8.0",
|
||||
"typescript": "~5.6.2",
|
||||
"vite": "^5.4.9"
|
||||
},
|
||||
"files": [
|
||||
"dist",
|
||||
".gitignore",
|
||||
"deno-src",
|
||||
"deno.json",
|
||||
"deno.lock"
|
||||
],
|
||||
"packageManager": "pnpm@9.15.3"
|
||||
}
|
4123
pnpm-lock.yaml
generated
Normal file
4123
pnpm-lock.yaml
generated
Normal file
File diff suppressed because it is too large
Load Diff
6
postcss.config.js
Normal file
6
postcss.config.js
Normal file
@ -0,0 +1,6 @@
|
||||
export default {
|
||||
plugins: {
|
||||
tailwindcss: {},
|
||||
autoprefixer: {}
|
||||
}
|
||||
};
|
1
public/vite.svg
Normal file
1
public/vite.svg
Normal file
@ -0,0 +1 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="31.88" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 257"><defs><linearGradient id="IconifyId1813088fe1fbc01fb466" x1="-.828%" x2="57.636%" y1="7.652%" y2="78.411%"><stop offset="0%" stop-color="#41D1FF"></stop><stop offset="100%" stop-color="#BD34FE"></stop></linearGradient><linearGradient id="IconifyId1813088fe1fbc01fb467" x1="43.376%" x2="50.316%" y1="2.242%" y2="89.03%"><stop offset="0%" stop-color="#FFEA83"></stop><stop offset="8.333%" stop-color="#FFDD35"></stop><stop offset="100%" stop-color="#FFA800"></stop></linearGradient></defs><path fill="url(#IconifyId1813088fe1fbc01fb466)" d="M255.153 37.938L134.897 252.976c-2.483 4.44-8.862 4.466-11.382.048L.875 37.958c-2.746-4.814 1.371-10.646 6.827-9.67l120.385 21.517a6.537 6.537 0 0 0 2.322-.004l117.867-21.483c5.438-.991 9.574 4.796 6.877 9.62Z"></path><path fill="url(#IconifyId1813088fe1fbc01fb467)" d="M185.432.063L96.44 17.501a3.268 3.268 0 0 0-2.634 3.014l-5.474 92.456a3.268 3.268 0 0 0 3.997 3.378l24.777-5.718c2.318-.535 4.413 1.507 3.936 3.838l-7.361 36.047c-.495 2.426 1.782 4.5 4.151 3.78l15.304-4.649c2.372-.72 4.652 1.36 4.15 3.788l-11.698 56.621c-.732 3.542 3.979 5.473 5.943 2.437l1.313-2.028l72.516-144.72c1.215-2.423-.88-5.186-3.54-4.672l-25.505 4.922c-2.396.462-4.435-1.77-3.759-4.114l16.646-57.705c.677-2.35-1.37-4.583-3.769-4.113Z"></path></svg>
|
After Width: | Height: | Size: 1.5 KiB |
133
src/App.svelte
Normal file
133
src/App.svelte
Normal file
@ -0,0 +1,133 @@
|
||||
<script lang="ts">
|
||||
import { open, path, shell, toast, ui } from "@kksh/api/ui/iframe"
|
||||
import { Button, ModeWatcher, ThemeWrapper, updateTheme } from "@kksh/svelte"
|
||||
import SpeedGauge from "$lib/components/SpeedGauge.svelte"
|
||||
import StressSelect from "$lib/components/StressSelect.svelte"
|
||||
import TargetDirSelect from "$lib/components/TargetDirSelect.svelte"
|
||||
import { stress, targetDir } from "$lib/store"
|
||||
import { onMount } from "svelte"
|
||||
import { get } from "svelte/store"
|
||||
import { type API } from "./types"
|
||||
|
||||
let readSpeedMBps = $state(0)
|
||||
let writeSpeedMBps = $state(0)
|
||||
let running = $state(false)
|
||||
|
||||
onMount(() => {
|
||||
ui.registerDragRegion()
|
||||
ui.showBackButton({ right: 0.5, bottom: 0.5 })
|
||||
|
||||
updateTheme({
|
||||
theme: "neutral",
|
||||
radius: 0.5,
|
||||
lightMode: "dark"
|
||||
})
|
||||
})
|
||||
|
||||
async function startSpeedTest() {
|
||||
running = true
|
||||
const _targetDir = get(targetDir)
|
||||
if (!_targetDir) {
|
||||
toast.error("Target directory is not set")
|
||||
return
|
||||
}
|
||||
|
||||
const { rpcChannel, process } = await shell.createDenoRpcChannel<{}, API>(
|
||||
"$EXTENSION/deno-src/lib.ts",
|
||||
[],
|
||||
{
|
||||
allowAllRead: true,
|
||||
allowAllWrite: true
|
||||
},
|
||||
{}
|
||||
)
|
||||
|
||||
const api = rpcChannel.getAPI()
|
||||
const testFileName = "kk-disk-speed-test"
|
||||
|
||||
const testFilePath = await path.join(_targetDir, testFileName)
|
||||
|
||||
const writeResult = await api.sequentialWriteTest(
|
||||
{
|
||||
filePath: testFilePath,
|
||||
sizeInMB: get(stress) * 1024,
|
||||
rounds: 3,
|
||||
bufferSizeMB: 1,
|
||||
keepTheFile: true
|
||||
},
|
||||
({ totalMB, totalDuration }) => {
|
||||
writeSpeedMBps = totalMB / totalDuration
|
||||
}
|
||||
)
|
||||
const readResult = await api.sequentialReadTest(testFilePath, {
|
||||
deleteAfter: true
|
||||
})
|
||||
writeSpeedMBps = writeResult.totalMB / writeResult.totalDuration
|
||||
console.log("writeDuration", writeResult)
|
||||
|
||||
readSpeedMBps = readResult.totalMB / readResult.totalDuration
|
||||
console.log("readDuration", readResult)
|
||||
|
||||
process
|
||||
.kill()
|
||||
.then(() => {
|
||||
console.log("process killed")
|
||||
})
|
||||
.catch((err) => {
|
||||
console.error("error killing process", err)
|
||||
toast.error(`Error killing process ${process.pid}`)
|
||||
})
|
||||
.finally(() => {
|
||||
running = false
|
||||
})
|
||||
}
|
||||
</script>
|
||||
|
||||
<svelte:window
|
||||
on:keydown={(e) => {
|
||||
if (e.key === "Escape") {
|
||||
ui.goBack()
|
||||
}
|
||||
}}
|
||||
/>
|
||||
<ModeWatcher />
|
||||
|
||||
<ThemeWrapper>
|
||||
<main class="container flex flex-col gap-4 pt-10">
|
||||
<div class="absolute left-0 top-0 h-10 w-screen" data-kunkun-drag-region></div>
|
||||
<div class="flex items-center gap-4">
|
||||
<strong>Stress</strong>
|
||||
<StressSelect />
|
||||
</div>
|
||||
<div class="flex items-center gap-4">
|
||||
<strong>Target Directory</strong>
|
||||
<TargetDirSelect />
|
||||
{#if $targetDir}
|
||||
<button
|
||||
onclick={() => {
|
||||
if ($targetDir) {
|
||||
open.folder($targetDir)
|
||||
}
|
||||
}}
|
||||
>
|
||||
<pre>{$targetDir}</pre>
|
||||
</button>
|
||||
{:else}
|
||||
<pre class="text-red-500">Pick a target directory to test</pre>
|
||||
{/if}
|
||||
</div>
|
||||
<Button disabled={!$targetDir || running} on:click={startSpeedTest}>Start Speed Test</Button>
|
||||
<!-- <div class="flex items-center gap-4">
|
||||
<strong>Write Speed</strong>
|
||||
<pre>{writeSpeedMBps} MB/s</pre>
|
||||
</div>
|
||||
<div class="flex items-center gap-4">
|
||||
<strong>Read Speed</strong>
|
||||
<pre>{readSpeedMBps} MB/s</pre>
|
||||
</div> -->
|
||||
<div class="grid h-96 w-full grid-cols-2">
|
||||
<SpeedGauge speedInMBps={writeSpeedMBps} title="Write Speed" class="h-full w-full" />
|
||||
<SpeedGauge speedInMBps={readSpeedMBps} title="Read Speed" class="h-full w-full" />
|
||||
</div>
|
||||
</main>
|
||||
</ThemeWrapper>
|
81
src/app.css
Normal file
81
src/app.css
Normal file
@ -0,0 +1,81 @@
|
||||
@import url("@kksh/svelte/themes");
|
||||
@tailwind base;
|
||||
@tailwind components;
|
||||
@tailwind utilities;
|
||||
|
||||
@layer base {
|
||||
:root {
|
||||
--background: 0 0% 100%;
|
||||
--foreground: 0 0% 3.9%;
|
||||
|
||||
--muted: 0 0% 96.1%;
|
||||
--muted-foreground: 0 0% 45.1%;
|
||||
|
||||
--popover: 0 0% 100%;
|
||||
--popover-foreground: 0 0% 3.9%;
|
||||
|
||||
--card: 0 0% 100%;
|
||||
--card-foreground: 0 0% 3.9%;
|
||||
|
||||
--border: 0 0% 89.8%;
|
||||
--input: 0 0% 89.8%;
|
||||
|
||||
--primary: 0 0% 9%;
|
||||
--primary-foreground: 0 0% 98%;
|
||||
|
||||
--secondary: 0 0% 96.1%;
|
||||
--secondary-foreground: 0 0% 9%;
|
||||
|
||||
--accent: 0 0% 96.1%;
|
||||
--accent-foreground: 0 0% 9%;
|
||||
|
||||
--destructive: 0 72.2% 50.6%;
|
||||
--destructive-foreground: 0 0% 98%;
|
||||
|
||||
--ring: 0 0% 3.9%;
|
||||
|
||||
--radius: 0.5rem;
|
||||
}
|
||||
|
||||
.dark {
|
||||
--background: 0 0% 3.9%;
|
||||
--foreground: 0 0% 98%;
|
||||
|
||||
--muted: 0 0% 14.9%;
|
||||
--muted-foreground: 0 0% 63.9%;
|
||||
|
||||
--popover: 0 0% 3.9%;
|
||||
--popover-foreground: 0 0% 98%;
|
||||
|
||||
--card: 0 0% 3.9%;
|
||||
--card-foreground: 0 0% 98%;
|
||||
|
||||
--border: 0 0% 14.9%;
|
||||
--input: 0 0% 14.9%;
|
||||
|
||||
--primary: 0 0% 98%;
|
||||
--primary-foreground: 0 0% 9%;
|
||||
|
||||
--secondary: 0 0% 14.9%;
|
||||
--secondary-foreground: 0 0% 98%;
|
||||
|
||||
--accent: 0 0% 14.9%;
|
||||
--accent-foreground: 0 0% 98%;
|
||||
|
||||
--destructive: 0 62.8% 30.6%;
|
||||
--destructive-foreground: 0 0% 98%;
|
||||
|
||||
--ring: 0 0% 83.1%;
|
||||
}
|
||||
}
|
||||
|
||||
@layer base {
|
||||
* {
|
||||
@apply border-border;
|
||||
}
|
||||
|
||||
body {
|
||||
@apply bg-background text-foreground;
|
||||
@apply overflow-x-hidden;
|
||||
}
|
||||
}
|
1
src/assets/svelte.svg
Normal file
1
src/assets/svelte.svg
Normal file
@ -0,0 +1 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="26.6" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 308"><path fill="#FF3E00" d="M239.682 40.707C211.113-.182 154.69-12.301 113.895 13.69L42.247 59.356a82.198 82.198 0 0 0-37.135 55.056a86.566 86.566 0 0 0 8.536 55.576a82.425 82.425 0 0 0-12.296 30.719a87.596 87.596 0 0 0 14.964 66.244c28.574 40.893 84.997 53.007 125.787 27.016l71.648-45.664a82.182 82.182 0 0 0 37.135-55.057a86.601 86.601 0 0 0-8.53-55.577a82.409 82.409 0 0 0 12.29-30.718a87.573 87.573 0 0 0-14.963-66.244"></path><path fill="#FFF" d="M106.889 270.841c-23.102 6.007-47.497-3.036-61.103-22.648a52.685 52.685 0 0 1-9.003-39.85a49.978 49.978 0 0 1 1.713-6.693l1.35-4.115l3.671 2.697a92.447 92.447 0 0 0 28.036 14.007l2.663.808l-.245 2.659a16.067 16.067 0 0 0 2.89 10.656a17.143 17.143 0 0 0 18.397 6.828a15.786 15.786 0 0 0 4.403-1.935l71.67-45.672a14.922 14.922 0 0 0 6.734-9.977a15.923 15.923 0 0 0-2.713-12.011a17.156 17.156 0 0 0-18.404-6.832a15.78 15.78 0 0 0-4.396 1.933l-27.35 17.434a52.298 52.298 0 0 1-14.553 6.391c-23.101 6.007-47.497-3.036-61.101-22.649a52.681 52.681 0 0 1-9.004-39.849a49.428 49.428 0 0 1 22.34-33.114l71.664-45.677a52.218 52.218 0 0 1 14.563-6.398c23.101-6.007 47.497 3.036 61.101 22.648a52.685 52.685 0 0 1 9.004 39.85a50.559 50.559 0 0 1-1.713 6.692l-1.35 4.116l-3.67-2.693a92.373 92.373 0 0 0-28.037-14.013l-2.664-.809l.246-2.658a16.099 16.099 0 0 0-2.89-10.656a17.143 17.143 0 0 0-18.398-6.828a15.786 15.786 0 0 0-4.402 1.935l-71.67 45.674a14.898 14.898 0 0 0-6.73 9.975a15.9 15.9 0 0 0 2.709 12.012a17.156 17.156 0 0 0 18.404 6.832a15.841 15.841 0 0 0 4.402-1.935l27.345-17.427a52.147 52.147 0 0 1 14.552-6.397c23.101-6.006 47.497 3.037 61.102 22.65a52.681 52.681 0 0 1 9.003 39.848a49.453 49.453 0 0 1-22.34 33.12l-71.664 45.673a52.218 52.218 0 0 1-14.563 6.398"></path></svg>
|
After Width: | Height: | Size: 1.9 KiB |
10
src/lib/Counter.svelte
Normal file
10
src/lib/Counter.svelte
Normal file
@ -0,0 +1,10 @@
|
||||
<script lang="ts">
|
||||
let count: number = 0
|
||||
const increment = () => {
|
||||
count += 1
|
||||
}
|
||||
</script>
|
||||
|
||||
<button on:click={increment}>
|
||||
count is {count}
|
||||
</button>
|
96
src/lib/components/SpeedGauge.svelte
Normal file
96
src/lib/components/SpeedGauge.svelte
Normal file
@ -0,0 +1,96 @@
|
||||
<script lang="ts">
|
||||
import { echarts } from "$lib/echarts.action.svelte"
|
||||
import { cn } from "$lib/utils"
|
||||
// import * as echarts from "echarts"
|
||||
import { onMount } from "svelte"
|
||||
|
||||
onMount(() => {})
|
||||
|
||||
const {
|
||||
speedInMBps,
|
||||
title,
|
||||
class: className
|
||||
}: { speedInMBps: number; title: string; class?: string } = $props()
|
||||
const maxSpeed = $derived.by(() => {
|
||||
if (speedInMBps <= 100) return 100
|
||||
if (speedInMBps <= 500) return 500
|
||||
if (speedInMBps <= 1_000) return 1_000
|
||||
if (speedInMBps <= 2_000) return 2_000
|
||||
if (speedInMBps <= 5_000) return 5_000
|
||||
if (speedInMBps <= 10_000) return 10_000
|
||||
return 100_000
|
||||
})
|
||||
const option = $derived({
|
||||
tooltip: {
|
||||
formatter: "{a} {b} : {c}%"
|
||||
},
|
||||
series: [
|
||||
{
|
||||
name: "Speed",
|
||||
type: "gauge",
|
||||
min: 0,
|
||||
max: maxSpeed,
|
||||
splitNumber: 5,
|
||||
// progress: {
|
||||
// show: true
|
||||
// },
|
||||
pointer: {
|
||||
itemStyle: {
|
||||
color: "auto"
|
||||
}
|
||||
},
|
||||
axisTick: {
|
||||
// distance: -15,
|
||||
// length: 8,
|
||||
// lineStyle: {
|
||||
// color: "#fff",
|
||||
// width: 2
|
||||
// }
|
||||
},
|
||||
// splitLine: {
|
||||
// distance: 0,
|
||||
// // length: 30,
|
||||
// lineStyle: {
|
||||
// color: "#fff",
|
||||
// width: 4
|
||||
// }
|
||||
// },
|
||||
axisLabel: {
|
||||
color: "inherit",
|
||||
distance: 20,
|
||||
fontSize: 15
|
||||
},
|
||||
axisLine: {
|
||||
lineStyle: {
|
||||
width: 10,
|
||||
color: [
|
||||
[0.3, "#67e0e3"],
|
||||
[0.7, "#37a2da"],
|
||||
[1, "#fd666d"]
|
||||
]
|
||||
}
|
||||
},
|
||||
detail: {
|
||||
fontSize: 20,
|
||||
valueAnimation: true,
|
||||
formatter: "{value}MB/s",
|
||||
color: "inherit"
|
||||
},
|
||||
title: {
|
||||
fontSize: 20
|
||||
},
|
||||
data: [
|
||||
{
|
||||
value: Math.round(speedInMBps),
|
||||
name: title
|
||||
}
|
||||
]
|
||||
}
|
||||
]
|
||||
})
|
||||
</script>
|
||||
|
||||
<div
|
||||
class={cn("flex min-h-96 min-w-96 items-center justify-center", className)}
|
||||
use:echarts={option}
|
||||
></div>
|
36
src/lib/components/StressSelect.svelte
Normal file
36
src/lib/components/StressSelect.svelte
Normal file
@ -0,0 +1,36 @@
|
||||
<script lang="ts">
|
||||
import { Select } from "@kksh/svelte"
|
||||
import { stress } from "$lib/store"
|
||||
import { type Selected } from "bits-ui"
|
||||
|
||||
const options = [
|
||||
{ value: 1, label: "1GB" },
|
||||
{ value: 2, label: "2GB" },
|
||||
{ value: 3, label: "3GB" },
|
||||
{ value: 4, label: "4GB" },
|
||||
{ value: 5, label: "5GB" }
|
||||
]
|
||||
|
||||
let selected: Selected<number> = $state(options[0])
|
||||
|
||||
$effect(() => {
|
||||
if (selected) {
|
||||
stress.set(selected.value)
|
||||
}
|
||||
})
|
||||
</script>
|
||||
|
||||
<Select.Root bind:selected>
|
||||
<Select.Trigger class="w-56">
|
||||
<Select.Value placeholder="Pick a Stress Level" />
|
||||
</Select.Trigger>
|
||||
<Select.Content>
|
||||
<Select.Group>
|
||||
<Select.Label>Stress</Select.Label>
|
||||
{#each options as option}
|
||||
<Select.Item value={option.value} label={option.label}>{option.label}</Select.Item>
|
||||
{/each}
|
||||
</Select.Group>
|
||||
</Select.Content>
|
||||
<Select.Input name="stress-level" />
|
||||
</Select.Root>
|
18
src/lib/components/TargetDirSelect.svelte
Normal file
18
src/lib/components/TargetDirSelect.svelte
Normal file
@ -0,0 +1,18 @@
|
||||
<script lang="ts">
|
||||
import Icon from "@iconify/svelte"
|
||||
import { clipboard, dialog, notification, toast } from "@kksh/api/ui/iframe"
|
||||
import { Button } from "@kksh/svelte"
|
||||
import { targetDir } from "$lib/store"
|
||||
|
||||
async function chooseDirectory() {
|
||||
const result = await dialog.open({
|
||||
directory: true
|
||||
})
|
||||
if (!result) return toast.warning("No directory selected")
|
||||
targetDir.set(result)
|
||||
}
|
||||
</script>
|
||||
|
||||
<Button variant="outline" size="icon" on:click={chooseDirectory}>
|
||||
<Icon icon="material-symbols:folder-outline" class="h-4 w-4" />
|
||||
</Button>
|
20
src/lib/components/ThemeCustomizer.svelte
Normal file
20
src/lib/components/ThemeCustomizer.svelte
Normal file
@ -0,0 +1,20 @@
|
||||
<script lang="ts">
|
||||
import { ui } from "@kksh/api/ui/iframe"
|
||||
import { ThemeCustomizerButton, updateTheme, type ThemeConfig } from "@kksh/svelte"
|
||||
import { onMount } from "svelte"
|
||||
|
||||
let config: ThemeConfig = {
|
||||
radius: 0.5,
|
||||
theme: "zinc",
|
||||
lightMode: "auto"
|
||||
}
|
||||
onMount(() => {
|
||||
ui.getTheme().then((theme) => {
|
||||
config = theme
|
||||
})
|
||||
})
|
||||
|
||||
$: updateTheme(config)
|
||||
</script>
|
||||
|
||||
<ThemeCustomizerButton bind:config />
|
20
src/lib/echarts.action.svelte.ts
Normal file
20
src/lib/echarts.action.svelte.ts
Normal file
@ -0,0 +1,20 @@
|
||||
/// <reference lib="dom" />
|
||||
import * as charts from "echarts"
|
||||
import { onMount } from "svelte"
|
||||
|
||||
export function echarts(node: HTMLElement, option: Record<string, any>) {
|
||||
let chart: charts.ECharts
|
||||
chart = charts.init(node)
|
||||
chart.setOption(option)
|
||||
setTimeout(() => {}, 500)
|
||||
|
||||
return {
|
||||
update(newOption: Record<string, any>) {
|
||||
// option = newOption
|
||||
chart.setOption(newOption)
|
||||
},
|
||||
destroy() {
|
||||
chart.dispose() // Clean up when component is destroyed
|
||||
}
|
||||
}
|
||||
}
|
4
src/lib/store.ts
Normal file
4
src/lib/store.ts
Normal file
@ -0,0 +1,4 @@
|
||||
import { writable } from "svelte/store"
|
||||
|
||||
export const stress = writable(1)
|
||||
export const targetDir = writable<string | undefined>(undefined)
|
62
src/lib/utils.ts
Normal file
62
src/lib/utils.ts
Normal file
@ -0,0 +1,62 @@
|
||||
import { type ClassValue, clsx } from "clsx";
|
||||
import { twMerge } from "tailwind-merge";
|
||||
import { cubicOut } from "svelte/easing";
|
||||
import type { TransitionConfig } from "svelte/transition";
|
||||
|
||||
export function cn(...inputs: ClassValue[]) {
|
||||
return twMerge(clsx(inputs));
|
||||
}
|
||||
|
||||
type FlyAndScaleParams = {
|
||||
y?: number;
|
||||
x?: number;
|
||||
start?: number;
|
||||
duration?: number;
|
||||
};
|
||||
|
||||
export const flyAndScale = (
|
||||
node: Element,
|
||||
params: FlyAndScaleParams = { y: -8, x: 0, start: 0.95, duration: 150 }
|
||||
): TransitionConfig => {
|
||||
const style = getComputedStyle(node);
|
||||
const transform = style.transform === "none" ? "" : style.transform;
|
||||
|
||||
const scaleConversion = (
|
||||
valueA: number,
|
||||
scaleA: [number, number],
|
||||
scaleB: [number, number]
|
||||
) => {
|
||||
const [minA, maxA] = scaleA;
|
||||
const [minB, maxB] = scaleB;
|
||||
|
||||
const percentage = (valueA - minA) / (maxA - minA);
|
||||
const valueB = percentage * (maxB - minB) + minB;
|
||||
|
||||
return valueB;
|
||||
};
|
||||
|
||||
const styleToString = (
|
||||
style: Record<string, number | string | undefined>
|
||||
): string => {
|
||||
return Object.keys(style).reduce((str, key) => {
|
||||
if (style[key] === undefined) return str;
|
||||
return str + `${key}:${style[key]};`;
|
||||
}, "");
|
||||
};
|
||||
|
||||
return {
|
||||
duration: params.duration ?? 200,
|
||||
delay: 0,
|
||||
css: (t) => {
|
||||
const y = scaleConversion(t, [0, 1], [params.y ?? 5, 0]);
|
||||
const x = scaleConversion(t, [0, 1], [params.x ?? 0, 0]);
|
||||
const scale = scaleConversion(t, [0, 1], [params.start ?? 0.95, 1]);
|
||||
|
||||
return styleToString({
|
||||
transform: `${transform} translate3d(${x}px, ${y}px, 0) scale(${scale})`,
|
||||
opacity: t
|
||||
});
|
||||
},
|
||||
easing: cubicOut
|
||||
};
|
||||
};
|
9
src/main.ts
Normal file
9
src/main.ts
Normal file
@ -0,0 +1,9 @@
|
||||
import { mount } from "svelte"
|
||||
import "./app.css"
|
||||
import App from "./App.svelte"
|
||||
|
||||
const app = mount(App, {
|
||||
target: document.getElementById("app")!
|
||||
})
|
||||
|
||||
export default app
|
16
src/model.ts
Normal file
16
src/model.ts
Normal file
@ -0,0 +1,16 @@
|
||||
export interface DiskSpeedTestInput {
|
||||
targetPath: string
|
||||
sequential: {
|
||||
stressFileSizeMB: number
|
||||
}
|
||||
random: {
|
||||
stressFileSizeMB: number
|
||||
iterations: number
|
||||
blockSize: number
|
||||
}
|
||||
}
|
||||
|
||||
export interface DiskSpeedTestOutput {
|
||||
writeSpeedMBps: number
|
||||
readSpeedMBps: number
|
||||
}
|
15
src/types.ts
Normal file
15
src/types.ts
Normal file
@ -0,0 +1,15 @@
|
||||
export type Progress = { totalMB: number; totalDuration: number }
|
||||
export interface API {
|
||||
sequentialWriteTest: (
|
||||
options: {
|
||||
filePath: string
|
||||
sizeInMB: number
|
||||
rounds: number
|
||||
bufferSizeMB: number
|
||||
keepTheFile?: boolean
|
||||
},
|
||||
callback?: (progress: Progress) => void
|
||||
) => Promise<Progress>
|
||||
sequentialReadTest: (filePath: string, options: { deleteAfter: boolean }) => Promise<Progress>
|
||||
createEmptyFile: (filePath: string, sizeInMB: number) => Promise<void>
|
||||
}
|
2
src/vite-env.d.ts
vendored
Normal file
2
src/vite-env.d.ts
vendored
Normal file
@ -0,0 +1,2 @@
|
||||
/// <reference types="svelte" />
|
||||
/// <reference types="vite/client" />
|
7
svelte.config.js
Normal file
7
svelte.config.js
Normal file
@ -0,0 +1,7 @@
|
||||
import { vitePreprocess } from "@sveltejs/vite-plugin-svelte"
|
||||
|
||||
export default {
|
||||
// Consult https://svelte.dev/docs#compile-time-svelte-preprocess
|
||||
// for more information about preprocessors
|
||||
preprocess: vitePreprocess()
|
||||
}
|
67
tailwind.config.ts
Normal file
67
tailwind.config.ts
Normal file
@ -0,0 +1,67 @@
|
||||
import type { Config } from "tailwindcss"
|
||||
import { fontFamily } from "tailwindcss/defaultTheme"
|
||||
|
||||
const config: Config = {
|
||||
darkMode: ["class"],
|
||||
content: [
|
||||
"./src/**/*.{html,js,svelte,ts}",
|
||||
"./node_modules/@kksh/svelte/dist/**/*.{html,js,svelte,ts}"
|
||||
],
|
||||
safelist: ["dark"],
|
||||
theme: {
|
||||
container: {
|
||||
center: true,
|
||||
padding: "2rem",
|
||||
screens: {
|
||||
"2xl": "1400px"
|
||||
}
|
||||
},
|
||||
extend: {
|
||||
colors: {
|
||||
border: "hsl(var(--border) / <alpha-value>)",
|
||||
input: "hsl(var(--input) / <alpha-value>)",
|
||||
ring: "hsl(var(--ring) / <alpha-value>)",
|
||||
background: "hsl(var(--background) / <alpha-value>)",
|
||||
foreground: "hsl(var(--foreground) / <alpha-value>)",
|
||||
primary: {
|
||||
DEFAULT: "hsl(var(--primary) / <alpha-value>)",
|
||||
foreground: "hsl(var(--primary-foreground) / <alpha-value>)"
|
||||
},
|
||||
secondary: {
|
||||
DEFAULT: "hsl(var(--secondary) / <alpha-value>)",
|
||||
foreground: "hsl(var(--secondary-foreground) / <alpha-value>)"
|
||||
},
|
||||
destructive: {
|
||||
DEFAULT: "hsl(var(--destructive) / <alpha-value>)",
|
||||
foreground: "hsl(var(--destructive-foreground) / <alpha-value>)"
|
||||
},
|
||||
muted: {
|
||||
DEFAULT: "hsl(var(--muted) / <alpha-value>)",
|
||||
foreground: "hsl(var(--muted-foreground) / <alpha-value>)"
|
||||
},
|
||||
accent: {
|
||||
DEFAULT: "hsl(var(--accent) / <alpha-value>)",
|
||||
foreground: "hsl(var(--accent-foreground) / <alpha-value>)"
|
||||
},
|
||||
popover: {
|
||||
DEFAULT: "hsl(var(--popover) / <alpha-value>)",
|
||||
foreground: "hsl(var(--popover-foreground) / <alpha-value>)"
|
||||
},
|
||||
card: {
|
||||
DEFAULT: "hsl(var(--card) / <alpha-value>)",
|
||||
foreground: "hsl(var(--card-foreground) / <alpha-value>)"
|
||||
}
|
||||
},
|
||||
borderRadius: {
|
||||
lg: "var(--radius)",
|
||||
md: "calc(var(--radius) - 2px)",
|
||||
sm: "calc(var(--radius) - 4px)"
|
||||
},
|
||||
fontFamily: {
|
||||
sans: [...fontFamily.sans]
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export default config
|
38
tsconfig.json
Normal file
38
tsconfig.json
Normal file
@ -0,0 +1,38 @@
|
||||
{
|
||||
"extends": "@tsconfig/svelte/tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"target": "ESNext",
|
||||
"useDefineForClassFields": true,
|
||||
"module": "ESNext",
|
||||
"resolveJsonModule": true,
|
||||
/**
|
||||
* Typecheck JS in `.svelte` and `.js` files by default.
|
||||
* Disable checkJs if you'd like to use dynamic types in JS.
|
||||
* Note that setting allowJs false does not prevent the use
|
||||
* of JS in `.svelte` files.
|
||||
*/
|
||||
"allowJs": true,
|
||||
"checkJs": true,
|
||||
"isolatedModules": true,
|
||||
"moduleDetection": "force",
|
||||
"baseUrl": ".",
|
||||
"paths": {
|
||||
"$lib": [
|
||||
"./src/lib"
|
||||
],
|
||||
"$lib/*": [
|
||||
"./src/lib/*"
|
||||
]
|
||||
}
|
||||
},
|
||||
"include": [
|
||||
"src/**/*.ts",
|
||||
"src/**/*.js",
|
||||
"src/**/*.svelte"
|
||||
],
|
||||
"references": [
|
||||
{
|
||||
"path": "./tsconfig.node.json"
|
||||
}
|
||||
]
|
||||
}
|
12
tsconfig.node.json
Normal file
12
tsconfig.node.json
Normal file
@ -0,0 +1,12 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"composite": true,
|
||||
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo",
|
||||
"skipLibCheck": true,
|
||||
"module": "ESNext",
|
||||
"moduleResolution": "bundler",
|
||||
"strict": true,
|
||||
"noEmit": true
|
||||
},
|
||||
"include": ["vite.config.ts"]
|
||||
}
|
13
vite.config.ts
Normal file
13
vite.config.ts
Normal file
@ -0,0 +1,13 @@
|
||||
import path from "path"
|
||||
import { svelte } from "@sveltejs/vite-plugin-svelte"
|
||||
import { defineConfig } from "vite"
|
||||
|
||||
// https://vitejs.dev/config/
|
||||
export default defineConfig({
|
||||
plugins: [svelte()],
|
||||
resolve: {
|
||||
alias: {
|
||||
$lib: path.resolve("./src/lib")
|
||||
}
|
||||
}
|
||||
})
|
Loading…
x
Reference in New Issue
Block a user