Merge branch 'master' of ssh://git.swordlost.top:222/SikongJueluo/FPGA_WebLab

This commit is contained in:
alivender 2025-08-01 20:00:00 +08:00
commit 9fe0ee959f
20 changed files with 947 additions and 520 deletions

View File

@ -39,10 +39,10 @@
typescript-language-server typescript-language-server
]; ];
shellHook = '' shellHook = ''
export PATH=$PATH: export PATH=$PATH:/home/sikongjueluo/.dotnet/tools
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:${pkgs.zlib}/lib export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:${pkgs.zlib}/lib
export DOTNET_ROOT=${pkgs.dotnetCorePackages.sdk_9_0}/share/dotnet
''; '';
}; };
}); });
}; };

215
package-lock.json generated
View File

@ -8,9 +8,11 @@
"name": "fpga-weblab", "name": "fpga-weblab",
"version": "0.1.0", "version": "0.1.0",
"dependencies": { "dependencies": {
"@microsoft/signalr": "^9.0.6",
"@svgdotjs/svg.js": "^3.2.4", "@svgdotjs/svg.js": "^3.2.4",
"@tanstack/vue-table": "^8.21.3", "@tanstack/vue-table": "^8.21.3",
"@types/lodash": "^4.17.16", "@types/lodash": "^4.17.16",
"@types/signalr": "^2.4.3",
"@vueuse/core": "^13.5.0", "@vueuse/core": "^13.5.0",
"async-mutex": "^0.5.0", "async-mutex": "^0.5.0",
"axios": "^1.11.0", "axios": "^1.11.0",
@ -1128,6 +1130,39 @@
"@jridgewell/sourcemap-codec": "^1.4.14" "@jridgewell/sourcemap-codec": "^1.4.14"
} }
}, },
"node_modules/@microsoft/signalr": {
"version": "9.0.6",
"resolved": "https://registry.npmjs.org/@microsoft/signalr/-/signalr-9.0.6.tgz",
"integrity": "sha512-DrhgzFWI9JE4RPTsHYRxh4yr+OhnwKz8bnJe7eIi7mLLjqhJpEb62CiUy/YbFvLqLzcGzlzz1QWgVAW0zyipMQ==",
"license": "MIT",
"dependencies": {
"abort-controller": "^3.0.0",
"eventsource": "^2.0.2",
"fetch-cookie": "^2.0.3",
"node-fetch": "^2.6.7",
"ws": "^7.5.10"
}
},
"node_modules/@microsoft/signalr/node_modules/node-fetch": {
"version": "2.7.0",
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz",
"integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==",
"license": "MIT",
"dependencies": {
"whatwg-url": "^5.0.0"
},
"engines": {
"node": "4.x || >=6.0.0"
},
"peerDependencies": {
"encoding": "^0.1.0"
},
"peerDependenciesMeta": {
"encoding": {
"optional": true
}
}
},
"node_modules/@polka/url": { "node_modules/@polka/url": {
"version": "1.0.0-next.29", "version": "1.0.0-next.29",
"resolved": "https://registry.npmjs.org/@polka/url/-/url-1.0.0-next.29.tgz", "resolved": "https://registry.npmjs.org/@polka/url/-/url-1.0.0-next.29.tgz",
@ -1845,6 +1880,15 @@
"dev": true, "dev": true,
"license": "MIT" "license": "MIT"
}, },
"node_modules/@types/jquery": {
"version": "3.5.32",
"resolved": "https://registry.npmjs.org/@types/jquery/-/jquery-3.5.32.tgz",
"integrity": "sha512-b9Xbf4CkMqS02YH8zACqN1xzdxc3cO735Qe5AbSUFmyOiaWAbcpqh9Wna+Uk0vgACvoQHpWDg2rGdHkYPLmCiQ==",
"license": "MIT",
"dependencies": {
"@types/sizzle": "*"
}
},
"node_modules/@types/lodash": { "node_modules/@types/lodash": {
"version": "4.17.16", "version": "4.17.16",
"resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.17.16.tgz", "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.17.16.tgz",
@ -1861,6 +1905,21 @@
"undici-types": "~6.21.0" "undici-types": "~6.21.0"
} }
}, },
"node_modules/@types/signalr": {
"version": "2.4.3",
"resolved": "https://registry.npmjs.org/@types/signalr/-/signalr-2.4.3.tgz",
"integrity": "sha512-W6C1wMRIIhJV9nsw19yhw4h9zlkLnJzsu9dYlH35aHUQblPsDF6UpCcAVu4Ljy4RS3c3uJyV88wf2M2SOWqqZg==",
"license": "MIT",
"dependencies": {
"@types/jquery": "*"
}
},
"node_modules/@types/sizzle": {
"version": "2.3.9",
"resolved": "https://registry.npmjs.org/@types/sizzle/-/sizzle-2.3.9.tgz",
"integrity": "sha512-xzLEyKB50yqCUPUJkIsrVvoWNfFUbIZI+RspLWt8u+tIW/BetMBZtgV2LY/2o+tYH8dRvQ+eoPf3NdhQCcLE2w==",
"license": "MIT"
},
"node_modules/@types/web-bluetooth": { "node_modules/@types/web-bluetooth": {
"version": "0.0.21", "version": "0.0.21",
"resolved": "https://registry.npmjs.org/@types/web-bluetooth/-/web-bluetooth-0.0.21.tgz", "resolved": "https://registry.npmjs.org/@types/web-bluetooth/-/web-bluetooth-0.0.21.tgz",
@ -2257,6 +2316,18 @@
"url": "https://github.com/sponsors/antfu" "url": "https://github.com/sponsors/antfu"
} }
}, },
"node_modules/abort-controller": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz",
"integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==",
"license": "MIT",
"dependencies": {
"event-target-shim": "^5.0.0"
},
"engines": {
"node": ">=6.5"
}
},
"node_modules/acorn": { "node_modules/acorn": {
"version": "8.15.0", "version": "8.15.0",
"resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz",
@ -2988,6 +3059,24 @@
"integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/event-target-shim": {
"version": "5.0.1",
"resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz",
"integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==",
"license": "MIT",
"engines": {
"node": ">=6"
}
},
"node_modules/eventsource": {
"version": "2.0.2",
"resolved": "https://registry.npmjs.org/eventsource/-/eventsource-2.0.2.tgz",
"integrity": "sha512-IzUmBGPR3+oUG9dUeXynyNmf91/3zUSJg1lCktzKw47OXuhco54U3r9B7O4XX+Rb1Itm9OZ2b0RkTs10bICOxA==",
"license": "MIT",
"engines": {
"node": ">=12.0.0"
}
},
"node_modules/execa": { "node_modules/execa": {
"version": "9.5.2", "version": "9.5.2",
"resolved": "https://registry.npmjs.org/execa/-/execa-9.5.2.tgz", "resolved": "https://registry.npmjs.org/execa/-/execa-9.5.2.tgz",
@ -3061,6 +3150,16 @@
"node": "^12.20 || >= 14.13" "node": "^12.20 || >= 14.13"
} }
}, },
"node_modules/fetch-cookie": {
"version": "2.2.0",
"resolved": "https://registry.npmjs.org/fetch-cookie/-/fetch-cookie-2.2.0.tgz",
"integrity": "sha512-h9AgfjURuCgA2+2ISl8GbavpUdR+WGAM2McW/ovn4tVccegp8ZqCKWSBR8uRdM8dDNlx5WdKRWxBYUwteLDCNQ==",
"license": "Unlicense",
"dependencies": {
"set-cookie-parser": "^2.4.8",
"tough-cookie": "^4.0.0"
}
},
"node_modules/figures": { "node_modules/figures": {
"version": "6.1.0", "version": "6.1.0",
"resolved": "https://registry.npmjs.org/figures/-/figures-6.1.0.tgz", "resolved": "https://registry.npmjs.org/figures/-/figures-6.1.0.tgz",
@ -4475,6 +4574,27 @@
"integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/psl": {
"version": "1.15.0",
"resolved": "https://registry.npmjs.org/psl/-/psl-1.15.0.tgz",
"integrity": "sha512-JZd3gMVBAVQkSs6HdNZo9Sdo0LNcQeMNP3CozBJb3JYC/QUYZTnKxP+f8oWRX4rHP5EurWxqAHTSwUCjlNKa1w==",
"license": "MIT",
"dependencies": {
"punycode": "^2.3.1"
},
"funding": {
"url": "https://github.com/sponsors/lupomontero"
}
},
"node_modules/punycode": {
"version": "2.3.1",
"resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
"integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==",
"license": "MIT",
"engines": {
"node": ">=6"
}
},
"node_modules/quansync": { "node_modules/quansync": {
"version": "0.2.10", "version": "0.2.10",
"resolved": "https://registry.npmjs.org/quansync/-/quansync-0.2.10.tgz", "resolved": "https://registry.npmjs.org/quansync/-/quansync-0.2.10.tgz",
@ -4492,6 +4612,12 @@
], ],
"license": "MIT" "license": "MIT"
}, },
"node_modules/querystringify": {
"version": "2.2.0",
"resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.2.0.tgz",
"integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==",
"license": "MIT"
},
"node_modules/read-package-json-fast": { "node_modules/read-package-json-fast": {
"version": "4.0.0", "version": "4.0.0",
"resolved": "https://registry.npmjs.org/read-package-json-fast/-/read-package-json-fast-4.0.0.tgz", "resolved": "https://registry.npmjs.org/read-package-json-fast/-/read-package-json-fast-4.0.0.tgz",
@ -4577,6 +4703,12 @@
"url": "https://github.com/sponsors/antfu" "url": "https://github.com/sponsors/antfu"
} }
}, },
"node_modules/requires-port": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz",
"integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==",
"license": "MIT"
},
"node_modules/resolve-pkg-maps": { "node_modules/resolve-pkg-maps": {
"version": "1.0.0", "version": "1.0.0",
"resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz", "resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz",
@ -4662,6 +4794,12 @@
"semver": "bin/semver.js" "semver": "bin/semver.js"
} }
}, },
"node_modules/set-cookie-parser": {
"version": "2.7.1",
"resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-2.7.1.tgz",
"integrity": "sha512-IOc8uWeOZgnb3ptbCURJWNjWUPcO3ZnTTdzsurqERrP6nPyv+paC55vJM0LpOlT2ne+Ix+9+CRG1MNLlyZ4GjQ==",
"license": "MIT"
},
"node_modules/shebang-command": { "node_modules/shebang-command": {
"version": "2.0.0", "version": "2.0.0",
"resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
@ -4832,6 +4970,36 @@
"node": ">=6" "node": ">=6"
} }
}, },
"node_modules/tough-cookie": {
"version": "4.1.4",
"resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-4.1.4.tgz",
"integrity": "sha512-Loo5UUvLD9ScZ6jh8beX1T6sO1w2/MpCRpEP7V280GKMVUQ0Jzar2U3UJPsrdbziLEMMhu3Ujnq//rhiFuIeag==",
"license": "BSD-3-Clause",
"dependencies": {
"psl": "^1.1.33",
"punycode": "^2.1.1",
"universalify": "^0.2.0",
"url-parse": "^1.5.3"
},
"engines": {
"node": ">=6"
}
},
"node_modules/tough-cookie/node_modules/universalify": {
"version": "0.2.0",
"resolved": "https://registry.npmjs.org/universalify/-/universalify-0.2.0.tgz",
"integrity": "sha512-CJ1QgKmNg3CwvAv/kOFmtnEN05f0D/cn9QntgNOQlQF9dgvVTHj3t+8JPdjqawCHk7V/KA+fbUqzZ9XWhcqPUg==",
"license": "MIT",
"engines": {
"node": ">= 4.0.0"
}
},
"node_modules/tr46": {
"version": "0.0.3",
"resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz",
"integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==",
"license": "MIT"
},
"node_modules/ts-log": { "node_modules/ts-log": {
"version": "2.2.7", "version": "2.2.7",
"resolved": "https://registry.npmjs.org/ts-log/-/ts-log-2.2.7.tgz", "resolved": "https://registry.npmjs.org/ts-log/-/ts-log-2.2.7.tgz",
@ -5073,6 +5241,16 @@
"browserslist": ">= 4.21.0" "browserslist": ">= 4.21.0"
} }
}, },
"node_modules/url-parse": {
"version": "1.5.10",
"resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.5.10.tgz",
"integrity": "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==",
"license": "MIT",
"dependencies": {
"querystringify": "^2.1.1",
"requires-port": "^1.0.0"
}
},
"node_modules/v8-compile-cache-lib": { "node_modules/v8-compile-cache-lib": {
"version": "3.0.1", "version": "3.0.1",
"resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz", "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz",
@ -5392,6 +5570,12 @@
"node": ">= 8" "node": ">= 8"
} }
}, },
"node_modules/webidl-conversions": {
"version": "3.0.1",
"resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz",
"integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==",
"license": "BSD-2-Clause"
},
"node_modules/webpack-virtual-modules": { "node_modules/webpack-virtual-modules": {
"version": "0.6.2", "version": "0.6.2",
"resolved": "https://registry.npmjs.org/webpack-virtual-modules/-/webpack-virtual-modules-0.6.2.tgz", "resolved": "https://registry.npmjs.org/webpack-virtual-modules/-/webpack-virtual-modules-0.6.2.tgz",
@ -5399,6 +5583,16 @@
"dev": true, "dev": true,
"license": "MIT" "license": "MIT"
}, },
"node_modules/whatwg-url": {
"version": "5.0.0",
"resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz",
"integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==",
"license": "MIT",
"dependencies": {
"tr46": "~0.0.3",
"webidl-conversions": "^3.0.0"
}
},
"node_modules/which": { "node_modules/which": {
"version": "5.0.0", "version": "5.0.0",
"resolved": "https://registry.npmjs.org/which/-/which-5.0.0.tgz", "resolved": "https://registry.npmjs.org/which/-/which-5.0.0.tgz",
@ -5415,6 +5609,27 @@
"node": "^18.17.0 || >=20.5.0" "node": "^18.17.0 || >=20.5.0"
} }
}, },
"node_modules/ws": {
"version": "7.5.10",
"resolved": "https://registry.npmjs.org/ws/-/ws-7.5.10.tgz",
"integrity": "sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ==",
"license": "MIT",
"engines": {
"node": ">=8.3.0"
},
"peerDependencies": {
"bufferutil": "^4.0.1",
"utf-8-validate": "^5.0.2"
},
"peerDependenciesMeta": {
"bufferutil": {
"optional": true
},
"utf-8-validate": {
"optional": true
}
}
},
"node_modules/yallist": { "node_modules/yallist": {
"version": "3.1.1", "version": "3.1.1",
"resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",

View File

@ -12,9 +12,11 @@
"gen-api": "npx tsx scripts/GenerateWebAPI.ts" "gen-api": "npx tsx scripts/GenerateWebAPI.ts"
}, },
"dependencies": { "dependencies": {
"@microsoft/signalr": "^9.0.6",
"@svgdotjs/svg.js": "^3.2.4", "@svgdotjs/svg.js": "^3.2.4",
"@tanstack/vue-table": "^8.21.3", "@tanstack/vue-table": "^8.21.3",
"@types/lodash": "^4.17.16", "@types/lodash": "^4.17.16",
"@types/signalr": "^2.4.3",
"@vueuse/core": "^13.5.0", "@vueuse/core": "^13.5.0",
"async-mutex": "^0.5.0", "async-mutex": "^0.5.0",
"axios": "^1.11.0", "axios": "^1.11.0",

View File

@ -1,36 +1,42 @@
import { spawn, exec, ChildProcess } from 'child_process'; import { spawn, exec, ChildProcess } from "child_process";
import { promisify } from 'util'; import { promisify } from "util";
import fetch from 'node-fetch'; import fetch from "node-fetch";
import * as fs from 'fs'; import * as fs from "fs";
const execAsync = promisify(exec); const execAsync = promisify(exec);
// Windows 支持函数 // Windows 支持函数
function getCommand(command: string): string { function getCommand(command: string): string {
// dotnet 在 Windows 上不需要 .cmd 后缀 // dotnet 在 Windows 上不需要 .cmd 后缀
if (command === 'dotnet') { if (command === "dotnet") {
return 'dotnet'; return "dotnet";
} }
return process.platform === 'win32' ? `${command}.cmd` : command; return process.platform === "win32" ? `${command}.cmd` : command;
} }
function getSpawnOptions() { function getSpawnOptions() {
return process.platform === 'win32' ? { stdio: 'pipe', shell: true } : { stdio: 'pipe' }; return process.platform === "win32"
? { stdio: "pipe", shell: true }
: { stdio: "pipe" };
} }
async function waitForServer(url: string, maxRetries: number = 30, interval: number = 1000): Promise<boolean> { async function waitForServer(
url: string,
maxRetries: number = 30,
interval: number = 1000,
): Promise<boolean> {
for (let i = 0; i < maxRetries; i++) { for (let i = 0; i < maxRetries; i++) {
try { try {
const response = await fetch(url); const response = await fetch(url);
if (response.ok) { if (response.ok) {
console.log('✓ Server is ready'); console.log("✓ Server is ready");
return true; return true;
} }
} catch (error) { } catch (error) {
// Server not ready yet // Server not ready yet
} }
console.log(`Waiting for server... (${i + 1}/${maxRetries})`); console.log(`Waiting for server... (${i + 1}/${maxRetries})`);
await new Promise(resolve => setTimeout(resolve, interval)); await new Promise((resolve) => setTimeout(resolve, interval));
} }
return false; return false;
} }
@ -40,32 +46,39 @@ let serverProcess: ChildProcess | null = null;
let webProcess: ChildProcess | null = null; let webProcess: ChildProcess | null = null;
async function startWeb(): Promise<ChildProcess> { async function startWeb(): Promise<ChildProcess> {
console.log('Starting Vite frontend...'); console.log("Starting Vite frontend...");
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
const process = spawn(getCommand('npm'), ['run', 'dev'], getSpawnOptions() as any); const process = spawn(
getCommand("npm"),
["run", "dev"],
getSpawnOptions() as any,
);
let webStarted = false; let webStarted = false;
process.stdout?.on('data', (data) => { process.stdout?.on("data", (data) => {
const output = data.toString(); const output = data.toString();
console.log(`Web: ${output}`); console.log(`Web: ${output}`);
// 检查 Vite 是否已启动 // 检查 Vite 是否已启动
if ((output.includes('Local:') || output.includes('ready in')) && !webStarted) { if (
(output.includes("Local:") || output.includes("ready in")) &&
!webStarted
) {
webStarted = true; webStarted = true;
resolve(process); resolve(process);
} }
}); });
process.stderr?.on('data', (data) => { process.stderr?.on("data", (data) => {
console.error(`Web Error: ${data}`); console.error(`Web Error: ${data}`);
}); });
process.on('error', (error) => { process.on("error", (error) => {
reject(error); reject(error);
}); });
process.on('exit', (code, signal) => { process.on("exit", (code, signal) => {
console.log(`Web process exited with code ${code} and signal ${signal}`); console.log(`Web process exited with code ${code} and signal ${signal}`);
if (!webStarted) { if (!webStarted) {
reject(new Error(`Web process exited unexpectedly with code ${code}`)); reject(new Error(`Web process exited unexpectedly with code ${code}`));
@ -78,45 +91,53 @@ async function startWeb(): Promise<ChildProcess> {
// 超时处理 // 超时处理
setTimeout(() => { setTimeout(() => {
if (!webStarted) { if (!webStarted) {
reject(new Error('Web server failed to start within timeout')); reject(new Error("Web server failed to start within timeout"));
} }
}, 30000); // 30秒超时 }, 10000); // 10秒超时
}); });
} }
async function startServer(): Promise<ChildProcess> { async function startServer(): Promise<ChildProcess> {
console.log('Starting .NET server...'); console.log("Starting .NET server...");
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
const process = spawn(getCommand('dotnet'), ['run', '--property:Configuration=Release'], { const process = spawn(
cwd: 'server', getCommand("dotnet"),
...getSpawnOptions() ["run", "--property:Configuration=Release"],
} as any); {
cwd: "server",
...getSpawnOptions(),
} as any,
);
let serverStarted = false; let serverStarted = false;
process.stdout?.on('data', (data) => { process.stdout?.on("data", (data) => {
const output = data.toString(); const output = data.toString();
console.log(`Server: ${output}`); console.log(`Server: ${output}`);
// 检查服务器是否已启动 // 检查服务器是否已启动
if (output.includes('Now listening on:') && !serverStarted) { if (output.includes("Now listening on:") && !serverStarted) {
serverStarted = true; serverStarted = true;
resolve(process); resolve(process);
} }
}); });
process.stderr?.on('data', (data) => { process.stderr?.on("data", (data) => {
console.error(`Server Error: ${data}`); console.error(`Server Error: ${data}`);
}); });
process.on('error', (error) => { process.on("error", (error) => {
reject(error); reject(error);
}); });
process.on('exit', (code, signal) => { process.on("exit", (code, signal) => {
console.log(`Server process exited with code ${code} and signal ${signal}`); console.log(
`Server process exited with code ${code} and signal ${signal}`,
);
if (!serverStarted) { if (!serverStarted) {
reject(new Error(`Server process exited unexpectedly with code ${code}`)); reject(
new Error(`Server process exited unexpectedly with code ${code}`),
);
} }
}); });
@ -126,62 +147,53 @@ async function startServer(): Promise<ChildProcess> {
// 超时处理 // 超时处理
setTimeout(() => { setTimeout(() => {
if (!serverStarted) { if (!serverStarted) {
reject(new Error('Server failed to start within timeout')); reject(new Error("Server failed to start within timeout"));
} }
}, 30000); // 30秒超时 }, 10000); // 10秒超时
}); });
} }
async function stopServer(): Promise<void> { async function stopServer(): Promise<void> {
console.log('Stopping server...'); console.log("Stopping server...");
if (!serverProcess) { if (!serverProcess) {
console.log('No server process to stop'); console.log("No server process to stop");
return; return;
} }
try { try {
// 检查进程是否还存在 // 检查进程是否还存在
if (serverProcess.killed || serverProcess.exitCode !== null) { if (serverProcess.killed || serverProcess.exitCode !== null) {
console.log('✓ Server process already terminated'); console.log("✓ Server process already terminated");
serverProcess = null; serverProcess = null;
return; return;
} }
// 发送 SIGTERM 信号 // 发送 SIGTERM 信号
const killed = serverProcess.kill('SIGTERM'); const killed = serverProcess.kill("SIGTERM");
if (!killed) { if (!killed) {
console.warn('Failed to send SIGTERM to server process'); console.warn("Failed to send SIGTERM to server process");
return; return;
} }
// 等待进程优雅退出
const exitPromise = new Promise<void>((resolve) => {
if (serverProcess) {
serverProcess.on('exit', () => {
console.log('✓ Server stopped gracefully');
resolve();
});
} else {
resolve();
}
});
// 设置超时,如果 3 秒内没有退出则强制终止 // 设置超时,如果 3 秒内没有退出则强制终止
const timeoutPromise = new Promise<void>((resolve) => { const timeoutPromise = new Promise<void>((resolve) => {
setTimeout(() => { setTimeout(() => {
if (serverProcess && !serverProcess.killed && serverProcess.exitCode === null) { if (
console.log('Force killing server process...'); serverProcess &&
serverProcess.kill('SIGKILL'); !serverProcess.killed &&
serverProcess.exitCode === null
) {
console.log("Force killing server process...");
serverProcess.kill("SIGKILL");
} }
resolve(); resolve();
}, 3000); // 减少超时时间到3秒 }, 3000); // 减少超时时间到3秒
}); });
await Promise.race([exitPromise, timeoutPromise]); await Promise.race([timeoutPromise]);
} catch (error) { } catch (error) {
console.warn('Warning: Could not stop server process:', error); console.warn("Warning: Could not stop server process:", error);
} finally { } finally {
serverProcess = null; serverProcess = null;
@ -191,67 +203,51 @@ async function stopServer(): Promise<void> {
} }
async function stopWeb(): Promise<void> { async function stopWeb(): Promise<void> {
console.log('Stopping web server...'); console.log("Stopping web server...");
if (!webProcess) { if (!webProcess) {
console.log('No web process to stop'); console.log("No web process to stop");
return; return;
} }
try { try {
// 检查进程是否还存在 // 检查进程是否还存在
if (webProcess.killed || webProcess.exitCode !== null) { if (webProcess.killed || webProcess.exitCode !== null) {
console.log('✓ Web process already terminated'); console.log("✓ Web process already terminated");
webProcess = null; webProcess = null;
return; return;
} }
// 发送 SIGTERM 信号 // 发送 SIGTERM 信号
const killed = webProcess.kill('SIGTERM'); const killed = webProcess.kill("SIGTERM");
if (!killed) { if (!killed) {
console.warn('Failed to send SIGTERM to web process'); console.warn("Failed to send SIGTERM to web process");
return; return;
} }
// 等待进程优雅退出
const exitPromise = new Promise<void>((resolve) => {
if (webProcess) {
webProcess.on('exit', () => {
console.log('✓ Web server stopped gracefully');
resolve();
});
} else {
resolve();
}
});
// 设置超时,如果 3 秒内没有退出则强制终止 // 设置超时,如果 3 秒内没有退出则强制终止
const timeoutPromise = new Promise<void>((resolve) => { const timeoutPromise = new Promise<void>((resolve) => {
setTimeout(() => { setTimeout(() => {
if (webProcess && !webProcess.killed && webProcess.exitCode === null) { if (webProcess && !webProcess.killed && webProcess.exitCode === null) {
console.log('Force killing web process...'); console.log("Force killing web process...");
webProcess.kill('SIGKILL'); webProcess.kill("SIGKILL");
} }
resolve(); resolve();
}, 3000); // 减少超时时间到3秒 }, 3000);
}); });
await Promise.race([exitPromise, timeoutPromise]); await Promise.race([timeoutPromise]);
} catch (error) { } catch (error) {
console.warn('Warning: Could not stop web process:', error); console.warn("Warning: Could not stop web process:", error);
} finally { } finally {
webProcess = null; webProcess = null;
// 只有在进程可能没有正常退出时才执行清理
// 移除自动清理逻辑,因为正常退出时不需要
} }
} }
async function postProcessApiClient(): Promise<void> { async function postProcessApiClient(): Promise<void> {
console.log('Post-processing API client...'); console.log("Post-processing API client...");
try { try {
const filePath = 'src/APIClient.ts'; const filePath = "src/APIClient.ts";
// 检查文件是否存在 // 检查文件是否存在
if (!fs.existsSync(filePath)) { if (!fs.existsSync(filePath)) {
@ -259,37 +255,43 @@ async function postProcessApiClient(): Promise<void> {
} }
// 读取文件内容 // 读取文件内容
let content = fs.readFileSync(filePath, 'utf8'); let content = fs.readFileSync(filePath, "utf8");
// 替换 ArgumentException 中的 message 属性声明 // 替换 ArgumentException 中的 message 属性声明
content = content.replace( content = content.replace(
/(\s+)message!:\s*string;/g, /(\s+)message!:\s*string;/g,
'$1declare message: string;' "$1declare message: string;",
);
content = content.replace(
"{ AxiosError, AxiosInstance, AxiosRequestConfig, AxiosResponse, CancelToken }",
"{ AxiosError, type AxiosInstance, type AxiosRequestConfig, type AxiosResponse, type CancelToken }",
); );
// 写回文件 // 写回文件
fs.writeFileSync(filePath, content, 'utf8'); fs.writeFileSync(filePath, content, "utf8");
console.log('✓ API client post-processing completed'); console.log("✓ API client post-processing completed");
} catch (error) { } catch (error) {
throw new Error(`Failed to post-process API client: ${error}`); throw new Error(`Failed to post-process API client: ${error}`);
} }
} }
async function generateApiClient(): Promise<void> { async function generateApiClient(): Promise<void> {
console.log('Generating API client...'); console.log("Generating API client...");
try { try {
const url = 'http://127.0.0.1:5000/GetAPIClientCode'; const url = "http://127.0.0.1:5000/GetAPIClientCode";
const response = await fetch(url); const response = await fetch(url);
if (!response.ok) { if (!response.ok) {
throw new Error(`Failed to fetch API client code: ${response.status} ${response.statusText}`); throw new Error(
`Failed to fetch API client code: ${response.status} ${response.statusText}`,
);
} }
const code = await response.text(); const code = await response.text();
// 写入 APIClient.ts // 写入 APIClient.ts
const filePath = 'src/APIClient.ts'; const filePath = "src/APIClient.ts";
fs.writeFileSync(filePath, code, 'utf8'); fs.writeFileSync(filePath, code, "utf8");
console.log('✓ API client code fetched and written successfully'); console.log("✓ API client code fetched and written successfully");
// 添加后处理步骤 // 添加后处理步骤
await postProcessApiClient(); await postProcessApiClient();
@ -298,35 +300,55 @@ async function generateApiClient(): Promise<void> {
} }
} }
async function generateSignalRClient(): Promise<void> {
console.log("Generating SignalR TypeScript client...");
try {
const { stdout, stderr } = await execAsync(
"dotnet tsrts --project ./server/server.csproj --output ./src",
);
if (stdout) console.log(stdout);
if (stderr) console.error(stderr);
console.log("✓ SignalR TypeScript client generated successfully");
} catch (error) {
throw new Error(`Failed to generate SignalR client: ${error}`);
}
}
async function main(): Promise<void> { async function main(): Promise<void> {
try { try {
// Generate SignalR client
await generateSignalRClient();
console.log("✓ SignalR TypeScript client generated successfully");
// Start web frontend first // Start web frontend first
await startWeb(); await startWeb();
console.log('✓ Frontend started'); console.log("✓ Frontend started");
// Wait a bit for frontend to fully initialize // Wait a bit for frontend to fully initialize
await new Promise(resolve => setTimeout(resolve, 3000)); await new Promise((resolve) => setTimeout(resolve, 3000));
// Start server // Start server
await startServer(); await startServer();
console.log('✓ Backend started'); console.log("✓ Backend started");
// Wait for server to be ready (给服务器额外时间完全启动) // Wait for server to be ready (给服务器额外时间完全启动)
await new Promise(resolve => setTimeout(resolve, 2000)); await new Promise((resolve) => setTimeout(resolve, 2000));
// Check if swagger endpoint is available // Check if swagger endpoint is available
const serverReady = await waitForServer('http://localhost:5000/swagger/v1/swagger.json'); const serverReady = await waitForServer(
"http://localhost:5000/swagger/v1/swagger.json",
);
if (!serverReady) { if (!serverReady) {
throw new Error('Server failed to start within the expected time'); throw new Error("Server failed to start within the expected time");
} }
// Generate API client // Generate API client
await generateApiClient(); await generateApiClient();
console.log('✓ API generation completed successfully'); console.log("✓ API generation completed successfully");
} catch (error) { } catch (error) {
console.error('❌ Error:', error); console.error("❌ Error:", error);
process.exit(1); process.exit(1);
} finally { } finally {
// Always try to stop processes in order: server first, then web // Always try to stop processes in order: server first, then web
@ -340,7 +362,7 @@ let isCleaningUp = false;
const cleanup = async (signal: string) => { const cleanup = async (signal: string) => {
if (isCleaningUp) { if (isCleaningUp) {
console.log('Cleanup already in progress, ignoring signal'); console.log("Cleanup already in progress, ignoring signal");
return; return;
} }
@ -348,53 +370,44 @@ const cleanup = async (signal: string) => {
console.log(`\nReceived ${signal}, cleaning up...`); console.log(`\nReceived ${signal}, cleaning up...`);
try { try {
await Promise.all([ await Promise.all([stopServer(), stopWeb()]);
stopServer(),
stopWeb()
]);
} catch (error) { } catch (error) {
console.error('Error during cleanup:', error); console.error("Error during cleanup:", error);
} }
// 立即退出,不等待 // 立即退出,不等待
process.exit(0); process.exit(0);
}; };
process.on('SIGINT', () => cleanup('SIGINT')); process.on("SIGINT", () => cleanup("SIGINT"));
process.on('SIGTERM', () => cleanup('SIGTERM')); process.on("SIGTERM", () => cleanup("SIGTERM"));
// 处理未捕获的异常 // 处理未捕获的异常
process.on('uncaughtException', async (error) => { process.on("uncaughtException", async (error) => {
if (isCleaningUp) return; if (isCleaningUp) return;
console.error('❌ Uncaught exception:', error); console.error("❌ Uncaught exception:", error);
isCleaningUp = true; isCleaningUp = true;
try { try {
await Promise.all([ await Promise.all([stopServer(), stopWeb()]);
stopServer(),
stopWeb()
]);
} catch (cleanupError) { } catch (cleanupError) {
console.error('Error during cleanup:', cleanupError); console.error("Error during cleanup:", cleanupError);
} }
process.exit(1); process.exit(1);
}); });
process.on('unhandledRejection', async (reason, promise) => { process.on("unhandledRejection", async (reason, promise) => {
if (isCleaningUp) return; if (isCleaningUp) return;
console.error('❌ Unhandled rejection at:', promise, 'reason:', reason); console.error("❌ Unhandled rejection at:", promise, "reason:", reason);
isCleaningUp = true; isCleaningUp = true;
try { try {
await Promise.all([ await Promise.all([stopServer(), stopWeb()]);
stopServer(),
stopWeb()
]);
} catch (cleanupError) { } catch (cleanupError) {
console.error('Error during cleanup:', cleanupError); console.error("Error during cleanup:", cleanupError);
} }
process.exit(1); process.exit(1);
@ -403,16 +416,13 @@ process.on('unhandledRejection', async (reason, promise) => {
main().catch(async (error) => { main().catch(async (error) => {
if (isCleaningUp) return; if (isCleaningUp) return;
console.error('❌ Unhandled error:', error); console.error("❌ Unhandled error:", error);
isCleaningUp = true; isCleaningUp = true;
try { try {
await Promise.all([ await Promise.all([stopServer(), stopWeb()]);
stopServer(),
stopWeb()
]);
} catch (cleanupError) { } catch (cleanupError) {
console.error('Error during cleanup:', cleanupError); console.error("Error during cleanup:", cleanupError);
} }
process.exit(1); process.exit(1);

View File

@ -5,13 +5,13 @@ using Microsoft.AspNetCore.Http.Features;
using Microsoft.Extensions.FileProviders; using Microsoft.Extensions.FileProviders;
using Microsoft.IdentityModel.Tokens; using Microsoft.IdentityModel.Tokens;
using Newtonsoft.Json; using Newtonsoft.Json;
using NJsonSchema.CodeGeneration.TypeScript;
using NLog; using NLog;
using NLog.Web; using NLog.Web;
using NSwag; using NSwag;
using NSwag.CodeGeneration.TypeScript; using NSwag.CodeGeneration.TypeScript;
using NSwag.Generation.Processors.Security; using NSwag.Generation.Processors.Security;
using server.Services; using server.Services;
using TypedSignalR.Client.DevTools;
// Early init of NLog to allow startup and exception logging, before host is built // Early init of NLog to allow startup and exception logging, before host is built
var logger = NLog.LogManager.Setup() var logger = NLog.LogManager.Setup()
@ -97,6 +97,9 @@ try
); );
}); });
// Use SignalR
builder.Services.AddSignalR();
// Add Swagger // Add Swagger
builder.Services.AddSwaggerDocument(options => builder.Services.AddSwaggerDocument(options =>
{ {
@ -198,9 +201,12 @@ try
}; };
}); });
app.UseSwaggerUi(); app.UseSwaggerUi();
app.UseSignalRHubSpecification();
app.UseSignalRHubDevelopmentUI();
// Router // Router
app.MapControllers(); app.MapControllers();
app.MapHub<server.Hubs.JtagHub.JtagHub>("hubs/JtagHub").RequireCors("Users");
// Setup Program // Setup Program
MsgBus.Init(); MsgBus.Init();
@ -231,7 +237,7 @@ try
logger.Error(err); logger.Error(err);
return Results.Problem(err.ToString()); return Results.Problem(err.ToString());
} }
}); }).RequireCors("Development");
app.Run(); app.Run();
} }
@ -252,4 +258,3 @@ finally
// Close Program // Close Program
MsgBus.Exit(); MsgBus.Exit();
} }

View File

@ -32,6 +32,16 @@
<PackageReference Include="OpenCvSharp4.runtime.win" Version="4.11.0.20250507" /> <PackageReference Include="OpenCvSharp4.runtime.win" Version="4.11.0.20250507" />
<PackageReference Include="SixLabors.ImageSharp" Version="3.1.10" /> <PackageReference Include="SixLabors.ImageSharp" Version="3.1.10" />
<PackageReference Include="System.Data.SQLite.Core" Version="1.0.119" /> <PackageReference Include="System.Data.SQLite.Core" Version="1.0.119" />
<PackageReference Include="Tapper.Analyzer" Version="1.13.1">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
<PackageReference Include="TypedSignalR.Client.DevTools" Version="1.2.4" />
<PackageReference Include="TypedSignalR.Client.TypeScript.Analyzer" Version="1.15.0">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
<PackageReference Include="TypedSignalR.Client.TypeScript.Attributes" Version="1.15.0" />
</ItemGroup> </ItemGroup>
</Project> </Project>

View File

@ -687,6 +687,31 @@ public class AppDataConnection : DataConnection
return new(boards[0]); return new(boards[0]);
} }
/// <summary>
/// 根据用户名获取实验板信息
/// </summary>
/// <param name="userName">用户名</param>
/// <returns>包含实验板信息的结果,如果未找到则返回空</returns>
public Result<Optional<Board>> GetBoardByUserName(string userName)
{
var boards = this.BoardTable.Where(board => board.OccupiedUserName == userName).ToArray();
if (boards.Length > 1)
{
logger.Error($"数据库中存在多个相同用户名的实验板: {userName}");
return new(new Exception($"数据库中存在多个相同用户名的实验板: {userName}"));
}
if (boards.Length == 0)
{
logger.Info($"未找到用户名对应的实验板: {userName}");
return new(Optional<Board>.None);
}
logger.Debug($"成功获取实验板信息: {userName}");
return new(boards[0]);
}
/// <summary> /// <summary>
/// 获取所有实验板信息 /// 获取所有实验板信息
/// </summary> /// </summary>

176
server/src/Hubs/JtagHub.cs Normal file
View File

@ -0,0 +1,176 @@
using Microsoft.AspNetCore.Authorization;
using System.Security.Claims;
using Microsoft.AspNetCore.SignalR;
using DotNext;
using System.Collections.Concurrent;
using TypedSignalR.Client;
using Tapper;
namespace server.Hubs.JtagHub;
[Hub]
public interface IJtagHub
{
Task<bool> SetBoundaryScanFreq(int freq);
Task<bool> StartBoundaryScan(int freq = 100);
Task<bool> StopBoundaryScan();
}
[Receiver]
public interface IJtagReceiver
{
Task OnReceiveBoundaryScanData(Dictionary<string, bool> msg);
}
[Authorize]
public class JtagHub : Hub<IJtagReceiver>, IJtagHub
{
private static NLog.Logger logger = NLog.LogManager.GetCurrentClassLogger();
private ConcurrentDictionary<string, int> FreqTable = new();
private ConcurrentDictionary<string, CancellationTokenSource> CancellationTokenSourceTable = new();
private Optional<Peripherals.JtagClient.Jtag> GetJtagClient(string userName)
{
try
{
using var db = new Database.AppDataConnection();
var board = db.GetBoardByUserName(userName);
if (!board.IsSuccessful)
{
logger.Error($"Find Board {board.Value.Value.ID} failed because {board.Error}");
return new(null);
}
if (!board.Value.HasValue)
{
logger.Error($"Board {board.Value.Value.ID} not found");
return new(null);
}
var jtag = new Peripherals.JtagClient.Jtag(board.Value.Value.IpAddr, board.Value.Value.Port);
return new(jtag);
}
catch (Exception error)
{
logger.Error(error);
return new(null);
}
}
public async Task<bool> SetBoundaryScanFreq(int freq)
{
try
{
var userName = Context.User?.FindFirstValue(ClaimTypes.Name);
if (userName is null)
{
logger.Error("Can't get user info");
return false;
}
FreqTable.AddOrUpdate(userName, freq, (key, value) => freq);
return true;
}
catch (Exception error)
{
logger.Error(error);
return false;
}
}
public async Task<bool> StartBoundaryScan(int freq = 100)
{
try
{
var userName = Context.User?.FindFirstValue(ClaimTypes.Name);
if (userName is null)
{
logger.Error("No Such User");
return false;
}
await SetBoundaryScanFreq(freq);
var cts = CancellationTokenSource.CreateLinkedTokenSource();
CancellationTokenSourceTable.AddOrUpdate(userName, cts, (key, value) => cts);
_ = Task
.Run(
() => BoundaryScanLogicPorts(
Context.ConnectionId,
userName,
cts.Token),
cts.Token)
.ContinueWith((task) =>
{
if (!task.IsFaulted)
{
return;
}
if (task.Exception.InnerException is OperationCanceledException)
{
logger.Info($"Boundary scan operation cancelled for user {userName}");
}
else
{
logger.Error(task.Exception);
}
});
return true;
}
catch (Exception error)
{
logger.Error(error);
return false;
}
}
public async Task<bool> StopBoundaryScan()
{
var userName = Context.User?.FindFirstValue(ClaimTypes.Name);
if (userName is null)
{
logger.Error("No Such User");
return false;
}
if (!CancellationTokenSourceTable.TryGetValue(userName, out var cts))
{
return false;
}
cts.Cancel();
cts.Token.WaitHandle.WaitOne();
return true;
}
private async void BoundaryScanLogicPorts(string connectionID, string userName, CancellationToken cancellationToken)
{
var jtagCtrl = GetJtagClient(userName).OrThrow(() => new InvalidOperationException("JTAG client not found"));
var cntFail = 0;
while (true && cntFail < 5)
{
cancellationToken.ThrowIfCancellationRequested();
var ret = await jtagCtrl.BoundaryScanLogicalPorts();
if (!ret.IsSuccessful)
{
logger.Error($"User {userName} boundary scan failed for device {jtagCtrl.address}: {ret.Error}");
cntFail++;
}
await this.Clients.Client(connectionID).OnReceiveBoundaryScanData(ret.Value);
// logger.Info($"User {userName} successfully completed boundary scan for device {jtagCtrl.address}");
await Task.Delay(FreqTable.TryGetValue(userName, out var freq) ? 1000 / freq : 1000 / 100, cancellationToken);
}
if (cntFail >= 5)
{
logger.Error($"User {userName} boundary scan failed for device {jtagCtrl.address} after 5 attempts");
throw new InvalidOperationException("Boundary scan failed");
}
}
}

View File

@ -386,7 +386,10 @@ public class Jtag
readonly int timeout; readonly int timeout;
readonly int port; readonly int port;
readonly string address; /// <summary>
/// Jtag控制器IP地址
/// </summary>
public readonly string address;
private IPEndPoint ep; private IPEndPoint ep;
/// <summary> /// <summary>

View File

@ -23,6 +23,50 @@ export class Client {
} }
getSignalrDevSpec_json( cancelToken?: CancelToken): Promise<void> {
let url_ = this.baseUrl + "/signalr-dev/spec.json";
url_ = url_.replace(/[?&]$/, "");
let options_: AxiosRequestConfig = {
method: "GET",
url: url_,
headers: {
},
cancelToken
};
return this.instance.request(options_).catch((_error: any) => {
if (isAxiosError(_error) && _error.response) {
return _error.response;
} else {
throw _error;
}
}).then((_response: AxiosResponse) => {
return this.processGetSignalrDevSpec_json(_response);
});
}
protected processGetSignalrDevSpec_json(response: AxiosResponse): Promise<void> {
const status = response.status;
let _headers: any = {};
if (response.headers && typeof response.headers === "object") {
for (const k in response.headers) {
if (response.headers.hasOwnProperty(k)) {
_headers[k] = response.headers[k];
}
}
}
if (status === 200) {
const _responseText = response.data;
return Promise.resolve<void>(null as any);
} else if (status !== 200 && status !== 204) {
const _responseText = response.data;
return throwException("An unexpected server error occurred.", status, _responseText, _headers);
}
return Promise.resolve<void>(null as any);
}
getGetAPIClientCode( cancelToken?: CancelToken): Promise<void> { getGetAPIClientCode( cancelToken?: CancelToken): Promise<void> {
let url_ = this.baseUrl + "/GetAPIClientCode"; let url_ = this.baseUrl + "/GetAPIClientCode";
url_ = url_.replace(/[?&]$/, ""); url_ = url_.replace(/[?&]$/, "");

View File

@ -0,0 +1,118 @@
/* THIS (.ts) FILE IS GENERATED BY TypedSignalR.Client.TypeScript */
/* eslint-disable */
/* tslint:disable */
// @ts-nocheck
import type { HubConnection, IStreamResult, Subject } from '@microsoft/signalr';
import type { IJtagHub, IJtagReceiver } from './server.Hubs.JtagHub';
// components
export type Disposable = {
dispose(): void;
}
export type HubProxyFactory<T> = {
createHubProxy(connection: HubConnection): T;
}
export type ReceiverRegister<T> = {
register(connection: HubConnection, receiver: T): Disposable;
}
type ReceiverMethod = {
methodName: string,
method: (...args: any[]) => void
}
class ReceiverMethodSubscription implements Disposable {
public constructor(
private connection: HubConnection,
private receiverMethod: ReceiverMethod[]) {
}
public readonly dispose = () => {
for (const it of this.receiverMethod) {
this.connection.off(it.methodName, it.method);
}
}
}
// API
export type HubProxyFactoryProvider = {
(hubType: "IJtagHub"): HubProxyFactory<IJtagHub>;
}
export const getHubProxyFactory = ((hubType: string) => {
if(hubType === "IJtagHub") {
return IJtagHub_HubProxyFactory.Instance;
}
}) as HubProxyFactoryProvider;
export type ReceiverRegisterProvider = {
(receiverType: "IJtagReceiver"): ReceiverRegister<IJtagReceiver>;
}
export const getReceiverRegister = ((receiverType: string) => {
if(receiverType === "IJtagReceiver") {
return IJtagReceiver_Binder.Instance;
}
}) as ReceiverRegisterProvider;
// HubProxy
class IJtagHub_HubProxyFactory implements HubProxyFactory<IJtagHub> {
public static Instance = new IJtagHub_HubProxyFactory();
private constructor() {
}
public readonly createHubProxy = (connection: HubConnection): IJtagHub => {
return new IJtagHub_HubProxy(connection);
}
}
class IJtagHub_HubProxy implements IJtagHub {
public constructor(private connection: HubConnection) {
}
public readonly setBoundaryScanFreq = async (freq: number): Promise<boolean> => {
return await this.connection.invoke("SetBoundaryScanFreq", freq);
}
public readonly startBoundaryScan = async (freq: number): Promise<boolean> => {
return await this.connection.invoke("StartBoundaryScan", freq);
}
public readonly stopBoundaryScan = async (): Promise<boolean> => {
return await this.connection.invoke("StopBoundaryScan");
}
}
// Receiver
class IJtagReceiver_Binder implements ReceiverRegister<IJtagReceiver> {
public static Instance = new IJtagReceiver_Binder();
private constructor() {
}
public readonly register = (connection: HubConnection, receiver: IJtagReceiver): Disposable => {
const __onReceiveBoundaryScanData = (...args: [Partial<Record<string, boolean>>]) => receiver.onReceiveBoundaryScanData(...args);
connection.on("OnReceiveBoundaryScanData", __onReceiveBoundaryScanData);
const methodList: ReceiverMethod[] = [
{ methodName: "OnReceiveBoundaryScanData", method: __onReceiveBoundaryScanData }
]
return new ReceiverMethodSubscription(connection, methodList);
}
}

View File

@ -0,0 +1,31 @@
/* THIS (.ts) FILE IS GENERATED BY TypedSignalR.Client.TypeScript */
/* eslint-disable */
/* tslint:disable */
// @ts-nocheck
import type { IStreamResult, Subject } from '@microsoft/signalr';
export type IJtagHub = {
/**
* @param freq Transpiled from int
* @returns Transpiled from System.Threading.Tasks.Task<bool>
*/
setBoundaryScanFreq(freq: number): Promise<boolean>;
/**
* @param freq Transpiled from int
* @returns Transpiled from System.Threading.Tasks.Task<bool>
*/
startBoundaryScan(freq: number): Promise<boolean>;
/**
* @returns Transpiled from System.Threading.Tasks.Task<bool>
*/
stopBoundaryScan(): Promise<boolean>;
}
export type IJtagReceiver = {
/**
* @param msg Transpiled from System.Collections.Generic.Dictionary<string, bool>
* @returns Transpiled from System.Threading.Tasks.Task
*/
onReceiveBoundaryScanData(msg: Partial<Record<string, boolean>>): Promise<void>;
}

View File

@ -1,11 +0,0 @@
<template>
<button class="btn transition-transform duration-150 ease-in-out hover:scale-120">
Button A
</button>
</template>
<script lang="ts" setup></script>
<style scoped lang="postcss">
@import "@/assets/main.css";
</style>

View File

@ -1,162 +0,0 @@
<template>
<div
class="card card-dash shadow-xl h-screen"
:class="[sidebar.isClose ? 'w-31' : 'w-80']"
>
<div
class="card-body flex relative transition-all duration-500 ease-in-out"
>
<!-- Avatar and Name -->
<div class="relative" :class="sidebar.isClose ? 'h-50' : 'h-20'">
<!-- Img -->
<div
class="avatar h-10 fixed top-10"
:class="sidebar.isClose ? 'left-10' : 'left-7'"
>
<div class="rounded-full">
<img src="../assets/user.svg" alt="User" />
</div>
</div>
<!-- Text -->
<Transition>
<div v-if="!sidebar.isClose" class="mx-5 grow fixed left-20 top-11">
<label class="text-2xl">用户名</label>
</div>
</Transition>
</div>
<!-- Toggle Button -->
<button
class="btn btn-square rounded-lg p-2 m-3 fixed"
:class="sidebar.isClose ? 'left-7 top-23' : 'left-60 top-7'"
@click="sidebar.toggleSidebar"
>
<svg
t="1741694970690"
:class="sidebar.isClose ? 'rotate-0' : 'rotate-540'"
class="icon"
viewBox="0 0 1024 1024"
version="1.1"
xmlns="http://www.w3.org/2000/svg"
p-id="4546"
xmlns:xlink="http://www.w3.org/1999/xlink"
width="200"
height="200"
>
<path
d="M803.758 514.017c-0.001-0.311-0.013-0.622-0.018-0.933-0.162-23.974-9.386-47.811-27.743-65.903-0.084-0.082-0.172-0.157-0.256-0.239-0.154-0.154-0.296-0.315-0.451-0.468L417.861 94.096c-37.685-37.153-99.034-37.476-136.331-0.718-37.297 36.758-36.979 97.231 0.707 134.384l290.361 286.257-290.362 286.257c-37.685 37.153-38.004 97.625-0.707 134.383 37.297 36.758 98.646 36.435 136.331-0.718l357.43-352.378c0.155-0.153 0.297-0.314 0.451-0.468 0.084-0.082 0.172-0.157 0.256-0.239 18.354-18.089 27.578-41.922 27.743-65.892 0.004-0.315 0.017-0.631 0.018-0.947z"
:fill="theme.isLightTheme() ? '#828282' : '#C0C3C8'"
p-id="4547"
></path>
</svg>
</button>
<div class="divider"></div>
<ul class="menu h-full w-full">
<li v-for="item in props.items" class="text-xl my-1">
<a @click="router.push(item.page)">
<svg
t="1741694797806"
class="icon h-[1.5em] w-[1.5em] opacity-50 mx-1"
viewBox="0 0 1024 1024"
version="1.1"
xmlns="http://www.w3.org/2000/svg"
p-id="2622"
xmlns:xlink="http://www.w3.org/1999/xlink"
width="200"
height="200"
>
<path
d="M192 192m-64 0a64 64 0 1 0 128 0 64 64 0 1 0-128 0Z"
p-id="2623"
></path>
<path
d="M192 512m-64 0a64 64 0 1 0 128 0 64 64 0 1 0-128 0Z"
p-id="2624"
></path>
<path
d="M192 832m-64 0a64 64 0 1 0 128 0 64 64 0 1 0-128 0Z"
p-id="2625"
></path>
<path
d="M864 160H352c-17.7 0-32 14.3-32 32s14.3 32 32 32h512c17.7 0 32-14.3 32-32s-14.3-32-32-32zM864 480H352c-17.7 0-32 14.3-32 32s14.3 32 32 32h512c17.7 0 32-14.3 32-32s-14.3-32-32-32zM864 800H352c-17.7 0-32 14.3-32 32s14.3 32 32 32h512c17.7 0 32-14.3 32-32s-14.3-32-32-32z"
p-id="2626"
></path>
</svg>
<Transition>
<p class="break-keep" v-if="!sidebar.isClose">{{ item.text }}</p>
</Transition>
</a>
</li>
</ul>
<div class="divider"></div>
<ul class="menu w-full">
<li>
<a @click="theme.toggleTheme" class="text-xl">
<ThemeControlButton />
<Transition>
<p v-if="!sidebar.isClose" class="break-keep">改变主题</p>
</Transition>
<Transition>
<ThemeControlToggle v-if="!sidebar.isClose" />
</Transition>
</a>
</li>
</ul>
</div>
</div>
</template>
<script setup lang="ts">
import iconMenu from "../assets/menu.svg";
import "../router";
import { useThemeStore } from "@/stores/theme";
import { useSidebarStore } from "@/stores/sidebar";
import ThemeControlButton from "./ThemeControlButton.vue";
import ThemeControlToggle from "./ThemeControlToggle.vue";
import router from "../router";
const theme = useThemeStore();
const sidebar = useSidebarStore();
type Item = {
id: number;
icon: string;
text: string;
page: string;
};
interface Props {
items?: Array<Item>;
}
const props = withDefaults(defineProps<Props>(), {
items: () => [
{ id: 1, icon: iconMenu, text: "btn1", page: "/" },
{ id: 2, icon: iconMenu, text: "btn2", page: "/" },
{ id: 3, icon: iconMenu, text: "btn3", page: "/" },
{ id: 4, icon: iconMenu, text: "btn4", page: "/" },
],
});
</script>
<style scoped lang="postcss">
@reference "../assets/main.css";
* {
@apply transition-all duration-500 ease-in-out;
}
.v-enter-active,
.v-leave-active {
transition: opacity 0.3s ease;
}
.v-enter-from,
.v-leave-to {
opacity: 0;
}
</style>

View File

@ -8,21 +8,35 @@
<p> <p>
IDCode: 0x{{ jtagIDCode.toString(16).padStart(8, "0").toUpperCase() }} IDCode: 0x{{ jtagIDCode.toString(16).padStart(8, "0").toUpperCase() }}
</p> </p>
<button class="btn btn-circle w-6 h-6" :disabled="isGettingIDCode" :onclick="getIDCode"> <button
<RefreshCcwIcon class="icon" :class="{ 'animate-spin': isGettingIDCode }" /> class="btn btn-circle w-6 h-6"
:disabled="isGettingIDCode"
:onclick="getIDCode"
>
<RefreshCcwIcon
class="icon"
:class="{ 'animate-spin': isGettingIDCode }"
/>
</button> </button>
</div> </div>
</div> </div>
<div class="divider"></div> <div class="divider"></div>
<UploadCard class="bg-base-200" :upload-event="eqps.jtagUploadBitstream" <UploadCard
:download-event="eqps.jtagDownloadBitstream" :bitstream-file="eqps.jtagBitstream" class="bg-base-200"
:exam-id="examId" :upload-event="eqps.jtagUploadBitstream"
@update:bitstream-file="handleBitstreamChange"> :download-event="eqps.jtagDownloadBitstream"
:bitstream-file="eqps.jtagBitstream"
@update:bitstream-file="handleBitstreamChange"
>
</UploadCard> </UploadCard>
<div class="divider"></div> <div class="divider"></div>
<div class="w-full"> <div class="w-full">
<legend class="fieldset-legend text-sm mb-0.3">Jtag运行频率</legend> <legend class="fieldset-legend text-sm mb-0.3">Jtag运行频率</legend>
<select class="select w-full" @change="handleSelectJtagSpeed" :value="props.jtagFreq"> <select
class="select w-full"
@change="handleSelectJtagSpeed"
:value="props.jtagFreq"
>
<option v-for="option in selectJtagSpeedOptions" :value="option.id"> <option v-for="option in selectJtagSpeedOptions" :value="option.id">
{{ option.text }} {{ option.text }}
</option> </option>
@ -31,12 +45,23 @@
<div class="flex flex-row items-center"> <div class="flex flex-row items-center">
<fieldset class="fieldset w-70"> <fieldset class="fieldset w-70">
<legend class="fieldset-legend text-sm">边界扫描刷新率 / Hz</legend> <legend class="fieldset-legend text-sm">边界扫描刷新率 / Hz</legend>
<input type="number" class="input validator" required placeholder="Type a number between 1 to 1000" min="1" <input
max="1000" v-model="jtagBoundaryScanFreq" title="Type a number between 1 to 1000" /> type="number"
class="input validator"
required
placeholder="Type a number between 1 to 1000"
min="1"
max="1000"
v-model="jtagBoundaryScanFreq"
title="Type a number between 1 to 1000"
/>
<p class="validator-hint">输入一个1 ~ 1000的数</p> <p class="validator-hint">输入一个1 ~ 1000的数</p>
</fieldset> </fieldset>
<button class="btn btn-primary grow mx-4" :class="eqps.enableJtagBoundaryScan ? '' : 'btn-soft'" <button
:onclick="toggleJtagBoundaryScan"> class="btn btn-primary grow mx-4"
:class="eqps.enableJtagBoundaryScan ? '' : 'btn-soft'"
:onclick="toggleJtagBoundaryScan"
>
{{ eqps.enableJtagBoundaryScan ? "关闭边界扫描" : "启动边界扫描" }} {{ eqps.enableJtagBoundaryScan ? "关闭边界扫描" : "启动边界扫描" }}
</button> </button>
</div> </div>
@ -44,8 +69,12 @@
<h1 class="font-bold text-center text-2xl">外设</h1> <h1 class="font-bold text-center text-2xl">外设</h1>
<div class="flex flex-row justify-center"> <div class="flex flex-row justify-center">
<div class="flex flex-row"> <div class="flex flex-row">
<input type="checkbox" class="checkbox" :checked="eqps.enableMatrixKey" <input
@change="handleMatrixkeyCheckboxChange" /> type="checkbox"
class="checkbox"
:checked="eqps.enableMatrixKey"
@change="handleMatrixkeyCheckboxChange"
/>
<p class="mx-2">启用矩阵键盘</p> <p class="mx-2">启用矩阵键盘</p>
</div> </div>
</div> </div>
@ -119,7 +148,7 @@ async function handleMatrixkeyCheckboxChange(event: Event) {
} }
async function toggleJtagBoundaryScan() { async function toggleJtagBoundaryScan() {
eqps.enableJtagBoundaryScan = !eqps.enableJtagBoundaryScan; eqps.jtagBoundaryScanSetOnOff(!eqps.enableJtagBoundaryScan);
} }
const isGettingIDCode = ref(false); const isGettingIDCode = ref(false);

View File

@ -1,26 +1,25 @@
import { ref, computed, reactive } from 'vue' import { ref, computed, reactive } from "vue";
import { defineStore } from 'pinia' import { defineStore } from "pinia";
import { isBoolean } from 'lodash'; import { isBoolean } from "lodash";
// 约束电平状态类型 // 约束电平状态类型
export type ConstraintLevel = 'high' | 'low' | 'undefined'; export type ConstraintLevel = "high" | "low" | "undefined";
export const useConstraintsStore = defineStore('constraints', () => {
export const useConstraintsStore = defineStore("constraints", () => {
// 约束状态存储 // 约束状态存储
const constraintStates = reactive<Record<string, ConstraintLevel>>({}); const constraintStates = reactive<Record<string, ConstraintLevel>>({});
// 约束颜色映射 // 约束颜色映射
const constraintColors = { const constraintColors = {
high: '#ff3333', // 高电平为红色 high: "#ff3333", // 高电平为红色
low: '#3333ff', // 低电平为蓝色 low: "#3333ff", // 低电平为蓝色
undefined: '#999999' // 未定义为灰色 undefined: "#999999", // 未定义为灰色
}; };
// 获取约束状态 // 获取约束状态
function getConstraintState(constraint: string): ConstraintLevel { function getConstraintState(constraint: string): ConstraintLevel {
if (!constraint) return 'undefined'; if (!constraint) return "undefined";
return constraintStates[constraint] || 'undefined'; return constraintStates[constraint] || "undefined";
} }
// 设置约束状态 // 设置约束状态
@ -30,7 +29,9 @@ export const useConstraintsStore = defineStore('constraints', () => {
} }
// 批量设置约束状态 // 批量设置约束状态
function batchSetConstraintStates(states: Record<string, ConstraintLevel> | Record<string, boolean>) { function batchSetConstraintStates(
states: Record<string, ConstraintLevel> | Partial<Record<string, boolean>>,
) {
// 收集发生变化的约束 // 收集发生变化的约束
const changedConstraints: [string, ConstraintLevel][] = []; const changedConstraints: [string, ConstraintLevel][] = [];
@ -38,6 +39,8 @@ export const useConstraintsStore = defineStore('constraints', () => {
Object.entries(states).forEach(([constraint, level]) => { Object.entries(states).forEach(([constraint, level]) => {
if (isBoolean(level)) { if (isBoolean(level)) {
level = level ? "high" : "low"; level = level ? "high" : "low";
} else {
level = "low";
} }
if (constraintStates[constraint] !== level) { if (constraintStates[constraint] !== level) {
@ -48,7 +51,7 @@ export const useConstraintsStore = defineStore('constraints', () => {
// 通知所有变化 // 通知所有变化
changedConstraints.forEach(([constraint, level]) => { changedConstraints.forEach(([constraint, level]) => {
stateChangeCallbacks.forEach(callback => callback(constraint, level)); stateChangeCallbacks.forEach((callback) => callback(constraint, level));
}); });
} }
@ -60,7 +63,7 @@ export const useConstraintsStore = defineStore('constraints', () => {
// 清除所有约束状态 // 清除所有约束状态
function clearAllConstraintStates() { function clearAllConstraintStates() {
Object.keys(constraintStates).forEach(key => { Object.keys(constraintStates).forEach((key) => {
delete constraintStates[key]; delete constraintStates[key];
}); });
} }
@ -71,9 +74,14 @@ export const useConstraintsStore = defineStore('constraints', () => {
} }
// 注册约束状态变化回调 // 注册约束状态变化回调
const stateChangeCallbacks: ((constraint: string, level: ConstraintLevel) => void)[] = []; const stateChangeCallbacks: ((
constraint: string,
level: ConstraintLevel,
) => void)[] = [];
function onConstraintStateChange(callback: (constraint: string, level: ConstraintLevel) => void) { function onConstraintStateChange(
callback: (constraint: string, level: ConstraintLevel) => void,
) {
stateChangeCallbacks.push(callback); stateChangeCallbacks.push(callback);
return () => { return () => {
const index = stateChangeCallbacks.indexOf(callback); const index = stateChangeCallbacks.indexOf(callback);
@ -86,7 +94,7 @@ export const useConstraintsStore = defineStore('constraints', () => {
// 触发约束变化 // 触发约束变化
function notifyConstraintChange(constraint: string, level: ConstraintLevel) { function notifyConstraintChange(constraint: string, level: ConstraintLevel) {
setConstraintState(constraint, level); setConstraintState(constraint, level);
stateChangeCallbacks.forEach(callback => callback(constraint, level)); stateChangeCallbacks.forEach((callback) => callback(constraint, level));
} }
return { return {
@ -98,6 +106,5 @@ export const useConstraintsStore = defineStore('constraints', () => {
getAllConstraintStates, getAllConstraintStates,
onConstraintStateChange, onConstraintStateChange,
notifyConstraintChange, notifyConstraintChange,
} };
}) });

View File

@ -1,15 +1,16 @@
import { ref, reactive, watchPostEffect } from "vue"; import { ref, reactive, watchPostEffect, onMounted, onUnmounted } from "vue";
import { defineStore } from "pinia"; import { defineStore } from "pinia";
import { useLocalStorage } from "@vueuse/core"; import { useLocalStorage } from "@vueuse/core";
import { isString, toNumber } from "lodash"; import { isString, toNumber, type Dictionary } from "lodash";
import z from "zod"; import z from "zod";
import { isNumber } from "mathjs"; import { isNumber } from "mathjs";
import { JtagClient, MatrixKeyClient, PowerClient } from "@/APIClient";
import { Mutex, withTimeout } from "async-mutex"; import { Mutex, withTimeout } from "async-mutex";
import { useConstraintsStore } from "@/stores/constraints"; import { useConstraintsStore } from "@/stores/constraints";
import { useDialogStore } from "./dialog"; import { useDialogStore } from "./dialog";
import { toFileParameterOrUndefined } from "@/utils/Common"; import { toFileParameterOrUndefined } from "@/utils/Common";
import { AuthManager } from "@/utils/AuthManager"; import { AuthManager } from "@/utils/AuthManager";
import { HubConnectionBuilder } from "@microsoft/signalr";
import { getHubProxyFactory, getReceiverRegister } from "@/TypedSignalR.Client";
export const useEquipments = defineStore("equipments", () => { export const useEquipments = defineStore("equipments", () => {
// Global Stores // Global Stores
@ -22,13 +23,28 @@ export const useEquipments = defineStore("equipments", () => {
// Jtag // Jtag
const jtagBitstream = ref<File>(); const jtagBitstream = ref<File>();
const jtagBoundaryScanFreq = ref(100); const jtagBoundaryScanFreq = ref(100);
const jtagBoundaryScanErrorCount = ref(0); // 边界扫描连续错误计数
const maxJtagBoundaryScanErrors = 5; // 最大允许连续错误次数
const jtagClientMutex = withTimeout( const jtagClientMutex = withTimeout(
new Mutex(), new Mutex(),
1000, 1000,
new Error("JtagClient Mutex Timeout!"), new Error("JtagClient Mutex Timeout!"),
); );
const jtagHubConnection = new HubConnectionBuilder()
.withUrl("/hubs/JtagHub")
.withAutomaticReconnect()
.build();
const jtagHubProxy =
getHubProxyFactory("IJtagHub").createHubProxy(jtagHubConnection);
const jtagHubSubscription = getReceiverRegister("IJtagReceiver").register(
jtagHubConnection,
{
onReceiveBoundaryScanData: async (msg) => {
constrainsts.batchSetConstraintStates(msg);
},
},
);
onMounted(() => {
jtagHubConnection.start();
});
// Matrix Key // Matrix Key
const matrixKeyStates = reactive(new Array<boolean>(16).fill(false)); const matrixKeyStates = reactive(new Array<boolean>(16).fill(false));
@ -50,41 +66,6 @@ export const useEquipments = defineStore("equipments", () => {
const enableMatrixKey = ref(false); const enableMatrixKey = ref(false);
const enablePower = ref(false); const enablePower = ref(false);
// Watch
watchPostEffect(async () => {
if (true === enableJtagBoundaryScan.value) {
// 重新启用时重置错误计数器
jtagBoundaryScanErrorCount.value = 0;
jtagBoundaryScan();
}
});
// Parse and Set
function setAddr(address: string | undefined): boolean {
if (isString(address) && z.string().ip("4").safeParse(address).success) {
boardAddr.value = address;
return true;
}
return false;
}
function setPort(port: string | number | undefined): boolean {
if (isString(port) && port.length != 0) {
const portNumber = toNumber(port);
if (z.number().nonnegative().max(65535).safeParse(portNumber).success) {
boardPort.value = portNumber;
return true;
}
} else if (isNumber(port)) {
if (z.number().nonnegative().max(65535).safeParse(port).success) {
boardPort.value = port;
return true;
}
}
return false;
}
function setMatrixKey( function setMatrixKey(
keyNum: number | string | undefined, keyNum: number | string | undefined,
keyValue: boolean, keyValue: boolean,
@ -105,38 +86,12 @@ export const useEquipments = defineStore("equipments", () => {
return false; return false;
} }
async function jtagBoundaryScan() { async function jtagBoundaryScanSetOnOff(enable: boolean) {
const release = await jtagClientMutex.acquire(); enableJtagBoundaryScan.value = enable;
try { if (enable) {
// 自动开启电源 jtagHubProxy.startBoundaryScan(jtagBoundaryScanFreq.value);
await powerSetOnOff(true); } else {
jtagHubProxy.stopBoundaryScan();
const jtagClient = AuthManager.createAuthenticatedJtagClient();
const portStates = await jtagClient.boundaryScanLogicalPorts(
boardAddr.value,
boardPort.value,
);
constrainsts.batchSetConstraintStates(portStates);
// 扫描成功,重置错误计数器
jtagBoundaryScanErrorCount.value = 0;
} catch (error) {
jtagBoundaryScanErrorCount.value++;
console.error(`边界扫描错误 (${jtagBoundaryScanErrorCount.value}/${maxJtagBoundaryScanErrors}):`, error);
// 如果错误次数超过最大允许次数,才停止扫描并显示错误
if (jtagBoundaryScanErrorCount.value >= maxJtagBoundaryScanErrors) {
dialog.error("边界扫描发生连续错误,已自动停止");
enableJtagBoundaryScan.value = false;
jtagBoundaryScanErrorCount.value = 0; // 重置错误计数器
}
} finally {
release();
if (enableJtagBoundaryScan.value)
setTimeout(jtagBoundaryScan, 1000 / jtagBoundaryScanFreq.value);
} }
} }
@ -224,7 +179,8 @@ export const useEquipments = defineStore("equipments", () => {
const release = await matrixKeypadClientMutex.acquire(); const release = await matrixKeypadClientMutex.acquire();
console.log("set Key !!!!!!!!!!!!"); console.log("set Key !!!!!!!!!!!!");
try { try {
const matrixKeypadClient = AuthManager.createAuthenticatedMatrixKeyClient(); const matrixKeypadClient =
AuthManager.createAuthenticatedMatrixKeyClient();
const resp = await matrixKeypadClient.setMatrixKeyStatus( const resp = await matrixKeypadClient.setMatrixKeyStatus(
boardAddr.value, boardAddr.value,
boardPort.value, boardPort.value,
@ -243,7 +199,8 @@ export const useEquipments = defineStore("equipments", () => {
const release = await matrixKeypadClientMutex.acquire(); const release = await matrixKeypadClientMutex.acquire();
try { try {
if (enable) { if (enable) {
const matrixKeypadClient = AuthManager.createAuthenticatedMatrixKeyClient(); const matrixKeypadClient =
AuthManager.createAuthenticatedMatrixKeyClient();
const resp = await matrixKeypadClient.enabelMatrixKey( const resp = await matrixKeypadClient.enabelMatrixKey(
boardAddr.value, boardAddr.value,
boardPort.value, boardPort.value,
@ -251,7 +208,8 @@ export const useEquipments = defineStore("equipments", () => {
enableMatrixKey.value = resp; enableMatrixKey.value = resp;
return resp; return resp;
} else { } else {
const matrixKeypadClient = AuthManager.createAuthenticatedMatrixKeyClient(); const matrixKeypadClient =
AuthManager.createAuthenticatedMatrixKeyClient();
const resp = await matrixKeypadClient.disableMatrixKey( const resp = await matrixKeypadClient.disableMatrixKey(
boardAddr.value, boardAddr.value,
boardPort.value, boardPort.value,
@ -290,16 +248,13 @@ export const useEquipments = defineStore("equipments", () => {
return { return {
boardAddr, boardAddr,
boardPort, boardPort,
setAddr,
setPort,
setMatrixKey, setMatrixKey,
// Jtag // Jtag
enableJtagBoundaryScan, enableJtagBoundaryScan,
jtagBoundaryScanSetOnOff,
jtagBitstream, jtagBitstream,
jtagBoundaryScanFreq, jtagBoundaryScanFreq,
jtagBoundaryScanErrorCount,
jtagClientMutex,
jtagUploadBitstream, jtagUploadBitstream,
jtagDownloadBitstream, jtagDownloadBitstream,
jtagGetIDCode, jtagGetIDCode,

View File

@ -1,29 +0,0 @@
import { ref, computed } from 'vue'
import { defineStore } from 'pinia'
export const useSidebarStore = defineStore('sidebar', () => {
const isClose = ref(false);
function closeSidebar() {
isClose.value = true;
console.info("Close sidebar");
}
function openSidebar() {
isClose.value = false;
console.info("Open sidebar");
}
function toggleSidebar() {
if (isClose.value) {
openSidebar();
// themeSidebar.value = "card-dash sidebar-base sidebar-open"
} else {
closeSidebar();
// themeSidebar.value = "card-dash sidebar-base sidebar-close"
}
}
return { isClose, closeSidebar, openSidebar, toggleSidebar }
})

View File

@ -1,13 +1,13 @@
import { fileURLToPath, URL } from 'node:url' import { fileURLToPath, URL } from "node:url";
import { defineConfig } from 'vite' import { defineConfig } from "vite";
import vue from '@vitejs/plugin-vue' import vue from "@vitejs/plugin-vue";
import vueJsx from '@vitejs/plugin-vue-jsx' import vueJsx from "@vitejs/plugin-vue-jsx";
import vueDevTools from 'vite-plugin-vue-devtools' import vueDevTools from "vite-plugin-vue-devtools";
import tailwindcss from '@tailwindcss/postcss' import tailwindcss from "@tailwindcss/postcss";
import autoprefixer from 'autoprefixer' import autoprefixer from "autoprefixer";
import Components from 'unplugin-vue-components/vite' import Components from "unplugin-vue-components/vite";
import RekaResolver from 'reka-ui/resolver' import RekaResolver from "reka-ui/resolver";
// https://vite.dev/config/ // https://vite.dev/config/
export default defineConfig({ export default defineConfig({
@ -16,49 +16,48 @@ export default defineConfig({
template: { template: {
compilerOptions: { compilerOptions: {
// 将所有 wokwi- 开头的标签视为自定义元素 // 将所有 wokwi- 开头的标签视为自定义元素
isCustomElement: (tag) => tag.startsWith('wokwi-') isCustomElement: (tag) => tag.startsWith("wokwi-"),
} },
} },
}), }),
vueJsx(), vueJsx(),
vueDevTools(), vueDevTools(),
Components( Components({
{
dts: true, dts: true,
resolvers: [ resolvers: [
RekaResolver() RekaResolver(),
// RekaResolver({ // RekaResolver({
// prefix: '' // use the prefix option to add Prefix to the imported components // prefix: '' // use the prefix option to add Prefix to the imported components
// }) // })
], ],
} }),
)
], ],
resolve: { resolve: {
alias: { alias: {
'@': fileURLToPath(new URL('./src', import.meta.url)) "@": fileURLToPath(new URL("./src", import.meta.url)),
}, },
}, },
css: { css: {
postcss: { postcss: {
plugins: [ plugins: [tailwindcss(), autoprefixer()],
tailwindcss(), },
autoprefixer()
]
}
}, },
build: { build: {
outDir: 'wwwroot', outDir: "wwwroot",
emptyOutDir: true, // also necessary emptyOutDir: true, // also necessary
}, },
server: { server: {
proxy: { proxy: {
"/swagger": { "/swagger": {
target: 'http://localhost:5000', target: "http://localhost:5000",
changeOrigin: true changeOrigin: true,
} },
"/hubs": {
target: "http://localhost:5000",
changeOrigin: true,
},
}, },
port: 5173, port: 5173,
} },
}) });