Merge branch 'master' of ssh://git.swordlost.top:222/SikongJueluo/FPGA_WebLab
This commit is contained in:
		@@ -7,7 +7,7 @@
 | 
			
		||||
    let
 | 
			
		||||
      supportedSystems = [ "x86_64-linux" "aarch64-linux" "x86_64-darwin" "aarch64-darwin" ];
 | 
			
		||||
      forEachSupportedSystem = f: nixpkgs.lib.genAttrs supportedSystems (system: f {
 | 
			
		||||
        pkgs = import nixpkgs { 
 | 
			
		||||
        pkgs = import nixpkgs {
 | 
			
		||||
          inherit system;
 | 
			
		||||
          config.permittedInsecurePackages = ["dotnet-sdk-6.0.428"];
 | 
			
		||||
        };
 | 
			
		||||
@@ -16,7 +16,7 @@
 | 
			
		||||
    {
 | 
			
		||||
      devShells = forEachSupportedSystem ({ pkgs }: {
 | 
			
		||||
        default = pkgs.mkShell {
 | 
			
		||||
          packages = with pkgs; [ 
 | 
			
		||||
          packages = with pkgs; [
 | 
			
		||||
            # Frontend
 | 
			
		||||
            nodejs
 | 
			
		||||
            sqlite
 | 
			
		||||
@@ -39,10 +39,10 @@
 | 
			
		||||
            typescript-language-server
 | 
			
		||||
          ];
 | 
			
		||||
          shellHook = ''
 | 
			
		||||
            export PATH=$PATH:
 | 
			
		||||
            export PATH=$PATH:/home/sikongjueluo/.dotnet/tools
 | 
			
		||||
            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
									
									
									
								
							
							
						
						
									
										215
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							@@ -8,9 +8,11 @@
 | 
			
		||||
      "name": "fpga-weblab",
 | 
			
		||||
      "version": "0.1.0",
 | 
			
		||||
      "dependencies": {
 | 
			
		||||
        "@microsoft/signalr": "^9.0.6",
 | 
			
		||||
        "@svgdotjs/svg.js": "^3.2.4",
 | 
			
		||||
        "@tanstack/vue-table": "^8.21.3",
 | 
			
		||||
        "@types/lodash": "^4.17.16",
 | 
			
		||||
        "@types/signalr": "^2.4.3",
 | 
			
		||||
        "@vueuse/core": "^13.5.0",
 | 
			
		||||
        "async-mutex": "^0.5.0",
 | 
			
		||||
        "axios": "^1.11.0",
 | 
			
		||||
@@ -1128,6 +1130,39 @@
 | 
			
		||||
        "@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": {
 | 
			
		||||
      "version": "1.0.0-next.29",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/@polka/url/-/url-1.0.0-next.29.tgz",
 | 
			
		||||
@@ -1845,6 +1880,15 @@
 | 
			
		||||
      "dev": true,
 | 
			
		||||
      "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": {
 | 
			
		||||
      "version": "4.17.16",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.17.16.tgz",
 | 
			
		||||
@@ -1861,6 +1905,21 @@
 | 
			
		||||
        "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": {
 | 
			
		||||
      "version": "0.0.21",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/@types/web-bluetooth/-/web-bluetooth-0.0.21.tgz",
 | 
			
		||||
@@ -2257,6 +2316,18 @@
 | 
			
		||||
        "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": {
 | 
			
		||||
      "version": "8.15.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz",
 | 
			
		||||
@@ -2988,6 +3059,24 @@
 | 
			
		||||
      "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==",
 | 
			
		||||
      "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": {
 | 
			
		||||
      "version": "9.5.2",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/execa/-/execa-9.5.2.tgz",
 | 
			
		||||
@@ -3061,6 +3150,16 @@
 | 
			
		||||
        "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": {
 | 
			
		||||
      "version": "6.1.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/figures/-/figures-6.1.0.tgz",
 | 
			
		||||
@@ -4475,6 +4574,27 @@
 | 
			
		||||
      "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==",
 | 
			
		||||
      "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": {
 | 
			
		||||
      "version": "0.2.10",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/quansync/-/quansync-0.2.10.tgz",
 | 
			
		||||
@@ -4492,6 +4612,12 @@
 | 
			
		||||
      ],
 | 
			
		||||
      "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": {
 | 
			
		||||
      "version": "4.0.0",
 | 
			
		||||
      "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"
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "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": {
 | 
			
		||||
      "version": "1.0.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz",
 | 
			
		||||
@@ -4662,6 +4794,12 @@
 | 
			
		||||
        "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": {
 | 
			
		||||
      "version": "2.0.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
 | 
			
		||||
@@ -4832,6 +4970,36 @@
 | 
			
		||||
        "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": {
 | 
			
		||||
      "version": "2.2.7",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/ts-log/-/ts-log-2.2.7.tgz",
 | 
			
		||||
@@ -5073,6 +5241,16 @@
 | 
			
		||||
        "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": {
 | 
			
		||||
      "version": "3.0.1",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz",
 | 
			
		||||
@@ -5392,6 +5570,12 @@
 | 
			
		||||
        "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": {
 | 
			
		||||
      "version": "0.6.2",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/webpack-virtual-modules/-/webpack-virtual-modules-0.6.2.tgz",
 | 
			
		||||
@@ -5399,6 +5583,16 @@
 | 
			
		||||
      "dev": true,
 | 
			
		||||
      "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": {
 | 
			
		||||
      "version": "5.0.0",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/which/-/which-5.0.0.tgz",
 | 
			
		||||
@@ -5415,6 +5609,27 @@
 | 
			
		||||
        "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": {
 | 
			
		||||
      "version": "3.1.1",
 | 
			
		||||
      "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
 | 
			
		||||
 
 | 
			
		||||
@@ -12,9 +12,11 @@
 | 
			
		||||
    "gen-api": "npx tsx scripts/GenerateWebAPI.ts"
 | 
			
		||||
  },
 | 
			
		||||
  "dependencies": {
 | 
			
		||||
    "@microsoft/signalr": "^9.0.6",
 | 
			
		||||
    "@svgdotjs/svg.js": "^3.2.4",
 | 
			
		||||
    "@tanstack/vue-table": "^8.21.3",
 | 
			
		||||
    "@types/lodash": "^4.17.16",
 | 
			
		||||
    "@types/signalr": "^2.4.3",
 | 
			
		||||
    "@vueuse/core": "^13.5.0",
 | 
			
		||||
    "async-mutex": "^0.5.0",
 | 
			
		||||
    "axios": "^1.11.0",
 | 
			
		||||
 
 | 
			
		||||
@@ -1,36 +1,42 @@
 | 
			
		||||
import { spawn, exec, ChildProcess } from 'child_process';
 | 
			
		||||
import { promisify } from 'util';
 | 
			
		||||
import fetch from 'node-fetch';
 | 
			
		||||
import * as fs from 'fs';
 | 
			
		||||
import { spawn, exec, ChildProcess } from "child_process";
 | 
			
		||||
import { promisify } from "util";
 | 
			
		||||
import fetch from "node-fetch";
 | 
			
		||||
import * as fs from "fs";
 | 
			
		||||
 | 
			
		||||
const execAsync = promisify(exec);
 | 
			
		||||
 | 
			
		||||
// Windows 支持函数
 | 
			
		||||
function getCommand(command: string): string {
 | 
			
		||||
  // dotnet 在 Windows 上不需要 .cmd 后缀
 | 
			
		||||
  if (command === 'dotnet') {
 | 
			
		||||
    return 'dotnet';
 | 
			
		||||
  if (command === "dotnet") {
 | 
			
		||||
    return "dotnet";
 | 
			
		||||
  }
 | 
			
		||||
  return process.platform === 'win32' ? `${command}.cmd` : command;
 | 
			
		||||
  return process.platform === "win32" ? `${command}.cmd` : command;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
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++) {
 | 
			
		||||
    try {
 | 
			
		||||
      const response = await fetch(url);
 | 
			
		||||
      if (response.ok) {
 | 
			
		||||
        console.log('✓ Server is ready');
 | 
			
		||||
        console.log("✓ Server is ready");
 | 
			
		||||
        return true;
 | 
			
		||||
      }
 | 
			
		||||
    } catch (error) {
 | 
			
		||||
      // Server not ready yet
 | 
			
		||||
    }
 | 
			
		||||
    console.log(`Waiting for server... (${i + 1}/${maxRetries})`);
 | 
			
		||||
    await new Promise(resolve => setTimeout(resolve, interval));
 | 
			
		||||
    await new Promise((resolve) => setTimeout(resolve, interval));
 | 
			
		||||
  }
 | 
			
		||||
  return false;
 | 
			
		||||
}
 | 
			
		||||
@@ -40,32 +46,39 @@ let serverProcess: ChildProcess | null = null;
 | 
			
		||||
let webProcess: ChildProcess | null = null;
 | 
			
		||||
 | 
			
		||||
async function startWeb(): Promise<ChildProcess> {
 | 
			
		||||
  console.log('Starting Vite frontend...');
 | 
			
		||||
  console.log("Starting Vite frontend...");
 | 
			
		||||
  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;
 | 
			
		||||
 | 
			
		||||
    process.stdout?.on('data', (data) => {
 | 
			
		||||
    process.stdout?.on("data", (data) => {
 | 
			
		||||
      const output = data.toString();
 | 
			
		||||
      console.log(`Web: ${output}`);
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // 检查 Vite 是否已启动
 | 
			
		||||
      if ((output.includes('Local:') || output.includes('ready in')) && !webStarted) {
 | 
			
		||||
      if (
 | 
			
		||||
        (output.includes("Local:") || output.includes("ready in")) &&
 | 
			
		||||
        !webStarted
 | 
			
		||||
      ) {
 | 
			
		||||
        webStarted = true;
 | 
			
		||||
        resolve(process);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    process.stderr?.on('data', (data) => {
 | 
			
		||||
    process.stderr?.on("data", (data) => {
 | 
			
		||||
      console.error(`Web Error: ${data}`);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    process.on('error', (error) => {
 | 
			
		||||
    process.on("error", (error) => {
 | 
			
		||||
      reject(error);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    process.on('exit', (code, signal) => {
 | 
			
		||||
    process.on("exit", (code, signal) => {
 | 
			
		||||
      console.log(`Web process exited with code ${code} and signal ${signal}`);
 | 
			
		||||
      if (!webStarted) {
 | 
			
		||||
        reject(new Error(`Web process exited unexpectedly with code ${code}`));
 | 
			
		||||
@@ -74,222 +87,211 @@ async function startWeb(): Promise<ChildProcess> {
 | 
			
		||||
 | 
			
		||||
    // 存储进程引用
 | 
			
		||||
    webProcess = process;
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // 超时处理
 | 
			
		||||
    setTimeout(() => {
 | 
			
		||||
      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> {
 | 
			
		||||
  console.log('Starting .NET server...');
 | 
			
		||||
  console.log("Starting .NET server...");
 | 
			
		||||
  return new Promise((resolve, reject) => {
 | 
			
		||||
    const process = spawn(getCommand('dotnet'), ['run', '--property:Configuration=Release'], {
 | 
			
		||||
      cwd: 'server',
 | 
			
		||||
      ...getSpawnOptions()
 | 
			
		||||
    } as any);
 | 
			
		||||
    const process = spawn(
 | 
			
		||||
      getCommand("dotnet"),
 | 
			
		||||
      ["run", "--property:Configuration=Release"],
 | 
			
		||||
      {
 | 
			
		||||
        cwd: "server",
 | 
			
		||||
        ...getSpawnOptions(),
 | 
			
		||||
      } as any,
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    let serverStarted = false;
 | 
			
		||||
 | 
			
		||||
    process.stdout?.on('data', (data) => {
 | 
			
		||||
    process.stdout?.on("data", (data) => {
 | 
			
		||||
      const output = data.toString();
 | 
			
		||||
      console.log(`Server: ${output}`);
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      // 检查服务器是否已启动
 | 
			
		||||
      if (output.includes('Now listening on:') && !serverStarted) {
 | 
			
		||||
      if (output.includes("Now listening on:") && !serverStarted) {
 | 
			
		||||
        serverStarted = true;
 | 
			
		||||
        resolve(process);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    process.stderr?.on('data', (data) => {
 | 
			
		||||
    process.stderr?.on("data", (data) => {
 | 
			
		||||
      console.error(`Server Error: ${data}`);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    process.on('error', (error) => {
 | 
			
		||||
    process.on("error", (error) => {
 | 
			
		||||
      reject(error);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    process.on('exit', (code, signal) => {
 | 
			
		||||
      console.log(`Server process exited with code ${code} and signal ${signal}`);
 | 
			
		||||
    process.on("exit", (code, signal) => {
 | 
			
		||||
      console.log(
 | 
			
		||||
        `Server process exited with code ${code} and signal ${signal}`,
 | 
			
		||||
      );
 | 
			
		||||
      if (!serverStarted) {
 | 
			
		||||
        reject(new Error(`Server process exited unexpectedly with code ${code}`));
 | 
			
		||||
        reject(
 | 
			
		||||
          new Error(`Server process exited unexpectedly with code ${code}`),
 | 
			
		||||
        );
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    // 存储进程引用
 | 
			
		||||
    serverProcess = process;
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // 超时处理
 | 
			
		||||
    setTimeout(() => {
 | 
			
		||||
      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> {
 | 
			
		||||
  console.log('Stopping server...');
 | 
			
		||||
  
 | 
			
		||||
  console.log("Stopping server...");
 | 
			
		||||
 | 
			
		||||
  if (!serverProcess) {
 | 
			
		||||
    console.log('No server process to stop');
 | 
			
		||||
    console.log("No server process to stop");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  try {
 | 
			
		||||
    // 检查进程是否还存在
 | 
			
		||||
    if (serverProcess.killed || serverProcess.exitCode !== null) {
 | 
			
		||||
      console.log('✓ Server process already terminated');
 | 
			
		||||
      console.log("✓ Server process already terminated");
 | 
			
		||||
      serverProcess = null;
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 发送 SIGTERM 信号
 | 
			
		||||
    const killed = serverProcess.kill('SIGTERM');
 | 
			
		||||
    const killed = serverProcess.kill("SIGTERM");
 | 
			
		||||
    if (!killed) {
 | 
			
		||||
      console.warn('Failed to send SIGTERM to server process');
 | 
			
		||||
      console.warn("Failed to send SIGTERM to server process");
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 等待进程优雅退出
 | 
			
		||||
    const exitPromise = new Promise<void>((resolve) => {
 | 
			
		||||
      if (serverProcess) {
 | 
			
		||||
        serverProcess.on('exit', () => {
 | 
			
		||||
          console.log('✓ Server stopped gracefully');
 | 
			
		||||
          resolve();
 | 
			
		||||
        });
 | 
			
		||||
      } else {
 | 
			
		||||
        resolve();
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    // 设置超时,如果 3 秒内没有退出则强制终止
 | 
			
		||||
    const timeoutPromise = new Promise<void>((resolve) => {
 | 
			
		||||
      setTimeout(() => {
 | 
			
		||||
        if (serverProcess && !serverProcess.killed && serverProcess.exitCode === null) {
 | 
			
		||||
          console.log('Force killing server process...');
 | 
			
		||||
          serverProcess.kill('SIGKILL');
 | 
			
		||||
        if (
 | 
			
		||||
          serverProcess &&
 | 
			
		||||
          !serverProcess.killed &&
 | 
			
		||||
          serverProcess.exitCode === null
 | 
			
		||||
        ) {
 | 
			
		||||
          console.log("Force killing server process...");
 | 
			
		||||
          serverProcess.kill("SIGKILL");
 | 
			
		||||
        }
 | 
			
		||||
        resolve();
 | 
			
		||||
      }, 3000); // 减少超时时间到3秒
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    await Promise.race([exitPromise, timeoutPromise]);
 | 
			
		||||
    
 | 
			
		||||
    await Promise.race([timeoutPromise]);
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    console.warn('Warning: Could not stop server process:', error);
 | 
			
		||||
    console.warn("Warning: Could not stop server process:", error);
 | 
			
		||||
  } finally {
 | 
			
		||||
    serverProcess = null;
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // 只有在进程可能没有正常退出时才执行清理
 | 
			
		||||
    // 移除自动清理逻辑,因为正常退出时不需要
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
async function stopWeb(): Promise<void> {
 | 
			
		||||
  console.log('Stopping web server...');
 | 
			
		||||
  
 | 
			
		||||
  console.log("Stopping web server...");
 | 
			
		||||
 | 
			
		||||
  if (!webProcess) {
 | 
			
		||||
    console.log('No web process to stop');
 | 
			
		||||
    console.log("No web process to stop");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  try {
 | 
			
		||||
    // 检查进程是否还存在
 | 
			
		||||
    if (webProcess.killed || webProcess.exitCode !== null) {
 | 
			
		||||
      console.log('✓ Web process already terminated');
 | 
			
		||||
      console.log("✓ Web process already terminated");
 | 
			
		||||
      webProcess = null;
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 发送 SIGTERM 信号
 | 
			
		||||
    const killed = webProcess.kill('SIGTERM');
 | 
			
		||||
    const killed = webProcess.kill("SIGTERM");
 | 
			
		||||
    if (!killed) {
 | 
			
		||||
      console.warn('Failed to send SIGTERM to web process');
 | 
			
		||||
      console.warn("Failed to send SIGTERM to web process");
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 等待进程优雅退出
 | 
			
		||||
    const exitPromise = new Promise<void>((resolve) => {
 | 
			
		||||
      if (webProcess) {
 | 
			
		||||
        webProcess.on('exit', () => {
 | 
			
		||||
          console.log('✓ Web server stopped gracefully');
 | 
			
		||||
          resolve();
 | 
			
		||||
        });
 | 
			
		||||
      } else {
 | 
			
		||||
        resolve();
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    // 设置超时,如果 3 秒内没有退出则强制终止
 | 
			
		||||
    const timeoutPromise = new Promise<void>((resolve) => {
 | 
			
		||||
      setTimeout(() => {
 | 
			
		||||
        if (webProcess && !webProcess.killed && webProcess.exitCode === null) {
 | 
			
		||||
          console.log('Force killing web process...');
 | 
			
		||||
          webProcess.kill('SIGKILL');
 | 
			
		||||
          console.log("Force killing web process...");
 | 
			
		||||
          webProcess.kill("SIGKILL");
 | 
			
		||||
        }
 | 
			
		||||
        resolve();
 | 
			
		||||
      }, 3000); // 减少超时时间到3秒
 | 
			
		||||
      }, 3000);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    await Promise.race([exitPromise, timeoutPromise]);
 | 
			
		||||
    
 | 
			
		||||
    await Promise.race([timeoutPromise]);
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    console.warn('Warning: Could not stop web process:', error);
 | 
			
		||||
    console.warn("Warning: Could not stop web process:", error);
 | 
			
		||||
  } finally {
 | 
			
		||||
    webProcess = null;
 | 
			
		||||
    
 | 
			
		||||
    // 只有在进程可能没有正常退出时才执行清理
 | 
			
		||||
    // 移除自动清理逻辑,因为正常退出时不需要
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
async function postProcessApiClient(): Promise<void> {
 | 
			
		||||
  console.log('Post-processing API client...');
 | 
			
		||||
  console.log("Post-processing API client...");
 | 
			
		||||
  try {
 | 
			
		||||
    const filePath = 'src/APIClient.ts';
 | 
			
		||||
    
 | 
			
		||||
    const filePath = "src/APIClient.ts";
 | 
			
		||||
 | 
			
		||||
    // 检查文件是否存在
 | 
			
		||||
    if (!fs.existsSync(filePath)) {
 | 
			
		||||
      throw new Error(`API client file not found: ${filePath}`);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
    // 读取文件内容
 | 
			
		||||
    let content = fs.readFileSync(filePath, 'utf8');
 | 
			
		||||
    
 | 
			
		||||
    let content = fs.readFileSync(filePath, "utf8");
 | 
			
		||||
 | 
			
		||||
    // 替换 ArgumentException 中的 message 属性声明
 | 
			
		||||
    content = content.replace(
 | 
			
		||||
      /(\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');
 | 
			
		||||
    
 | 
			
		||||
    console.log('✓ API client post-processing completed');
 | 
			
		||||
    fs.writeFileSync(filePath, content, "utf8");
 | 
			
		||||
 | 
			
		||||
    console.log("✓ API client post-processing completed");
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    throw new Error(`Failed to post-process API client: ${error}`);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
async function generateApiClient(): Promise<void> {
 | 
			
		||||
  console.log('Generating API client...');
 | 
			
		||||
  console.log("Generating API client...");
 | 
			
		||||
  try {
 | 
			
		||||
    const url = 'http://127.0.0.1:5000/GetAPIClientCode';
 | 
			
		||||
    const url = "http://127.0.0.1:5000/GetAPIClientCode";
 | 
			
		||||
    const response = await fetch(url);
 | 
			
		||||
    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();
 | 
			
		||||
 | 
			
		||||
    // 写入 APIClient.ts
 | 
			
		||||
    const filePath = 'src/APIClient.ts';
 | 
			
		||||
    fs.writeFileSync(filePath, code, 'utf8');
 | 
			
		||||
    console.log('✓ API client code fetched and written successfully');
 | 
			
		||||
    const filePath = "src/APIClient.ts";
 | 
			
		||||
    fs.writeFileSync(filePath, code, "utf8");
 | 
			
		||||
    console.log("✓ API client code fetched and written successfully");
 | 
			
		||||
 | 
			
		||||
    // 添加后处理步骤
 | 
			
		||||
    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> {
 | 
			
		||||
  try {
 | 
			
		||||
    // Generate SignalR client
 | 
			
		||||
    await generateSignalRClient();
 | 
			
		||||
    console.log("✓ SignalR TypeScript client generated successfully");
 | 
			
		||||
 | 
			
		||||
    // Start web frontend first
 | 
			
		||||
    await startWeb();
 | 
			
		||||
    console.log('✓ Frontend started');
 | 
			
		||||
    
 | 
			
		||||
    console.log("✓ Frontend started");
 | 
			
		||||
 | 
			
		||||
    // Wait a bit for frontend to fully initialize
 | 
			
		||||
    await new Promise(resolve => setTimeout(resolve, 3000));
 | 
			
		||||
    
 | 
			
		||||
    await new Promise((resolve) => setTimeout(resolve, 3000));
 | 
			
		||||
 | 
			
		||||
    // Start server
 | 
			
		||||
    await startServer();
 | 
			
		||||
    console.log('✓ Backend started');
 | 
			
		||||
    
 | 
			
		||||
    console.log("✓ Backend started");
 | 
			
		||||
 | 
			
		||||
    // 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
 | 
			
		||||
    const serverReady = await waitForServer('http://localhost:5000/swagger/v1/swagger.json');
 | 
			
		||||
    
 | 
			
		||||
    const serverReady = await waitForServer(
 | 
			
		||||
      "http://localhost:5000/swagger/v1/swagger.json",
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    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
 | 
			
		||||
    await generateApiClient();
 | 
			
		||||
    
 | 
			
		||||
    console.log('✓ API generation completed successfully');
 | 
			
		||||
 | 
			
		||||
    console.log("✓ API generation completed successfully");
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    console.error('❌ Error:', error);
 | 
			
		||||
    console.error("❌ Error:", error);
 | 
			
		||||
    process.exit(1);
 | 
			
		||||
  } finally {
 | 
			
		||||
    // Always try to stop processes in order: server first, then web
 | 
			
		||||
@@ -340,80 +362,68 @@ let isCleaningUp = false;
 | 
			
		||||
 | 
			
		||||
const cleanup = async (signal: string) => {
 | 
			
		||||
  if (isCleaningUp) {
 | 
			
		||||
    console.log('Cleanup already in progress, ignoring signal');
 | 
			
		||||
    console.log("Cleanup already in progress, ignoring signal");
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  isCleaningUp = true;
 | 
			
		||||
  console.log(`\nReceived ${signal}, cleaning up...`);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  try {
 | 
			
		||||
    await Promise.all([
 | 
			
		||||
      stopServer(),
 | 
			
		||||
      stopWeb()
 | 
			
		||||
    ]);
 | 
			
		||||
    await Promise.all([stopServer(), stopWeb()]);
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    console.error('Error during cleanup:', error);
 | 
			
		||||
    console.error("Error during cleanup:", error);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  // 立即退出,不等待
 | 
			
		||||
  process.exit(0);
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
process.on('SIGINT', () => cleanup('SIGINT'));
 | 
			
		||||
process.on('SIGTERM', () => cleanup('SIGTERM'));
 | 
			
		||||
process.on("SIGINT", () => cleanup("SIGINT"));
 | 
			
		||||
process.on("SIGTERM", () => cleanup("SIGTERM"));
 | 
			
		||||
 | 
			
		||||
// 处理未捕获的异常
 | 
			
		||||
process.on('uncaughtException', async (error) => {
 | 
			
		||||
process.on("uncaughtException", async (error) => {
 | 
			
		||||
  if (isCleaningUp) return;
 | 
			
		||||
  
 | 
			
		||||
  console.error('❌ Uncaught exception:', error);
 | 
			
		||||
 | 
			
		||||
  console.error("❌ Uncaught exception:", error);
 | 
			
		||||
  isCleaningUp = true;
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  try {
 | 
			
		||||
    await Promise.all([
 | 
			
		||||
      stopServer(),
 | 
			
		||||
      stopWeb()
 | 
			
		||||
    ]);
 | 
			
		||||
    await Promise.all([stopServer(), stopWeb()]);
 | 
			
		||||
  } catch (cleanupError) {
 | 
			
		||||
    console.error('Error during cleanup:', cleanupError);
 | 
			
		||||
    console.error("Error during cleanup:", cleanupError);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  process.exit(1);
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
process.on('unhandledRejection', async (reason, promise) => {
 | 
			
		||||
process.on("unhandledRejection", async (reason, promise) => {
 | 
			
		||||
  if (isCleaningUp) return;
 | 
			
		||||
  
 | 
			
		||||
  console.error('❌ Unhandled rejection at:', promise, 'reason:', reason);
 | 
			
		||||
 | 
			
		||||
  console.error("❌ Unhandled rejection at:", promise, "reason:", reason);
 | 
			
		||||
  isCleaningUp = true;
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  try {
 | 
			
		||||
    await Promise.all([
 | 
			
		||||
      stopServer(),
 | 
			
		||||
      stopWeb()
 | 
			
		||||
    ]);
 | 
			
		||||
    await Promise.all([stopServer(), stopWeb()]);
 | 
			
		||||
  } catch (cleanupError) {
 | 
			
		||||
    console.error('Error during cleanup:', cleanupError);
 | 
			
		||||
    console.error("Error during cleanup:", cleanupError);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  process.exit(1);
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
main().catch(async (error) => {
 | 
			
		||||
  if (isCleaningUp) return;
 | 
			
		||||
  
 | 
			
		||||
  console.error('❌ Unhandled error:', error);
 | 
			
		||||
 | 
			
		||||
  console.error("❌ Unhandled error:", error);
 | 
			
		||||
  isCleaningUp = true;
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  try {
 | 
			
		||||
    await Promise.all([
 | 
			
		||||
      stopServer(),
 | 
			
		||||
      stopWeb()
 | 
			
		||||
    ]);
 | 
			
		||||
    await Promise.all([stopServer(), stopWeb()]);
 | 
			
		||||
  } catch (cleanupError) {
 | 
			
		||||
    console.error('Error during cleanup:', cleanupError);
 | 
			
		||||
    console.error("Error during cleanup:", cleanupError);
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  process.exit(1);
 | 
			
		||||
});
 | 
			
		||||
});
 | 
			
		||||
 
 | 
			
		||||
@@ -5,13 +5,13 @@ using Microsoft.AspNetCore.Http.Features;
 | 
			
		||||
using Microsoft.Extensions.FileProviders;
 | 
			
		||||
using Microsoft.IdentityModel.Tokens;
 | 
			
		||||
using Newtonsoft.Json;
 | 
			
		||||
using NJsonSchema.CodeGeneration.TypeScript;
 | 
			
		||||
using NLog;
 | 
			
		||||
using NLog.Web;
 | 
			
		||||
using NSwag;
 | 
			
		||||
using NSwag.CodeGeneration.TypeScript;
 | 
			
		||||
using NSwag.Generation.Processors.Security;
 | 
			
		||||
using server.Services;
 | 
			
		||||
using TypedSignalR.Client.DevTools;
 | 
			
		||||
 | 
			
		||||
// Early init of NLog to allow startup and exception logging, before host is built
 | 
			
		||||
var logger = NLog.LogManager.Setup()
 | 
			
		||||
@@ -97,6 +97,9 @@ try
 | 
			
		||||
        );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    // Use SignalR
 | 
			
		||||
    builder.Services.AddSignalR();
 | 
			
		||||
 | 
			
		||||
    // Add Swagger
 | 
			
		||||
    builder.Services.AddSwaggerDocument(options =>
 | 
			
		||||
    {
 | 
			
		||||
@@ -198,9 +201,12 @@ try
 | 
			
		||||
        };
 | 
			
		||||
    });
 | 
			
		||||
    app.UseSwaggerUi();
 | 
			
		||||
    app.UseSignalRHubSpecification();
 | 
			
		||||
    app.UseSignalRHubDevelopmentUI();
 | 
			
		||||
 | 
			
		||||
    // Router
 | 
			
		||||
    app.MapControllers();
 | 
			
		||||
    app.MapHub<server.Hubs.JtagHub.JtagHub>("hubs/JtagHub").RequireCors("Users");
 | 
			
		||||
 | 
			
		||||
    // Setup Program
 | 
			
		||||
    MsgBus.Init();
 | 
			
		||||
@@ -231,7 +237,7 @@ try
 | 
			
		||||
            logger.Error(err);
 | 
			
		||||
            return Results.Problem(err.ToString());
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
    }).RequireCors("Development");
 | 
			
		||||
 | 
			
		||||
    app.Run();
 | 
			
		||||
}
 | 
			
		||||
@@ -252,4 +258,3 @@ finally
 | 
			
		||||
    // Close Program
 | 
			
		||||
    MsgBus.Exit();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -32,6 +32,16 @@
 | 
			
		||||
    <PackageReference Include="OpenCvSharp4.runtime.win" Version="4.11.0.20250507" />
 | 
			
		||||
    <PackageReference Include="SixLabors.ImageSharp" Version="3.1.10" />
 | 
			
		||||
    <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>
 | 
			
		||||
 | 
			
		||||
</Project>
 | 
			
		||||
 
 | 
			
		||||
@@ -133,7 +133,7 @@ public class Board
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public enum BoardStatus
 | 
			
		||||
    {
 | 
			
		||||
        /// <summary> 
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 未启用状态,无法被使用
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        Disabled,
 | 
			
		||||
@@ -687,6 +687,31 @@ public class AppDataConnection : DataConnection
 | 
			
		||||
        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>
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										176
									
								
								server/src/Hubs/JtagHub.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								server/src/Hubs/JtagHub.cs
									
									
									
									
									
										Normal 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");
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -386,7 +386,10 @@ public class Jtag
 | 
			
		||||
    readonly int timeout;
 | 
			
		||||
 | 
			
		||||
    readonly int port;
 | 
			
		||||
    readonly string address;
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Jtag控制器IP地址
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public readonly string address;
 | 
			
		||||
    private IPEndPoint ep;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
 
 | 
			
		||||
@@ -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> {
 | 
			
		||||
        let url_ = this.baseUrl + "/GetAPIClientCode";
 | 
			
		||||
        url_ = url_.replace(/[?&]$/, "");
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										118
									
								
								src/TypedSignalR.Client/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								src/TypedSignalR.Client/index.ts
									
									
									
									
									
										Normal 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);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										31
									
								
								src/TypedSignalR.Client/server.Hubs.JtagHub.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								src/TypedSignalR.Client/server.Hubs.JtagHub.ts
									
									
									
									
									
										Normal 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>;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -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>
 | 
			
		||||
@@ -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>
 | 
			
		||||
@@ -8,21 +8,35 @@
 | 
			
		||||
        <p>
 | 
			
		||||
          IDCode: 0x{{ jtagIDCode.toString(16).padStart(8, "0").toUpperCase() }}
 | 
			
		||||
        </p>
 | 
			
		||||
        <button class="btn btn-circle w-6 h-6" :disabled="isGettingIDCode" :onclick="getIDCode">
 | 
			
		||||
          <RefreshCcwIcon class="icon" :class="{ 'animate-spin': isGettingIDCode }" />
 | 
			
		||||
        <button
 | 
			
		||||
          class="btn btn-circle w-6 h-6"
 | 
			
		||||
          :disabled="isGettingIDCode"
 | 
			
		||||
          :onclick="getIDCode"
 | 
			
		||||
        >
 | 
			
		||||
          <RefreshCcwIcon
 | 
			
		||||
            class="icon"
 | 
			
		||||
            :class="{ 'animate-spin': isGettingIDCode }"
 | 
			
		||||
          />
 | 
			
		||||
        </button>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="divider"></div>
 | 
			
		||||
    <UploadCard class="bg-base-200" :upload-event="eqps.jtagUploadBitstream"
 | 
			
		||||
      :download-event="eqps.jtagDownloadBitstream" :bitstream-file="eqps.jtagBitstream"
 | 
			
		||||
      :exam-id="examId"
 | 
			
		||||
      @update:bitstream-file="handleBitstreamChange">
 | 
			
		||||
    <UploadCard
 | 
			
		||||
      class="bg-base-200"
 | 
			
		||||
      :upload-event="eqps.jtagUploadBitstream"
 | 
			
		||||
      :download-event="eqps.jtagDownloadBitstream"
 | 
			
		||||
      :bitstream-file="eqps.jtagBitstream"
 | 
			
		||||
      @update:bitstream-file="handleBitstreamChange"
 | 
			
		||||
    >
 | 
			
		||||
    </UploadCard>
 | 
			
		||||
    <div class="divider"></div>
 | 
			
		||||
    <div class="w-full">
 | 
			
		||||
      <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.text }}
 | 
			
		||||
        </option>
 | 
			
		||||
@@ -31,12 +45,23 @@
 | 
			
		||||
    <div class="flex flex-row items-center">
 | 
			
		||||
      <fieldset class="fieldset w-70">
 | 
			
		||||
        <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"
 | 
			
		||||
          max="1000" v-model="jtagBoundaryScanFreq" title="Type a number between 1 to 1000" />
 | 
			
		||||
        <input
 | 
			
		||||
          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>
 | 
			
		||||
      </fieldset>
 | 
			
		||||
      <button class="btn btn-primary grow mx-4" :class="eqps.enableJtagBoundaryScan ? '' : 'btn-soft'"
 | 
			
		||||
        :onclick="toggleJtagBoundaryScan">
 | 
			
		||||
      <button
 | 
			
		||||
        class="btn btn-primary grow mx-4"
 | 
			
		||||
        :class="eqps.enableJtagBoundaryScan ? '' : 'btn-soft'"
 | 
			
		||||
        :onclick="toggleJtagBoundaryScan"
 | 
			
		||||
      >
 | 
			
		||||
        {{ eqps.enableJtagBoundaryScan ? "关闭边界扫描" : "启动边界扫描" }}
 | 
			
		||||
      </button>
 | 
			
		||||
    </div>
 | 
			
		||||
@@ -44,8 +69,12 @@
 | 
			
		||||
    <h1 class="font-bold text-center text-2xl">外设</h1>
 | 
			
		||||
    <div class="flex flex-row justify-center">
 | 
			
		||||
      <div class="flex flex-row">
 | 
			
		||||
        <input type="checkbox" class="checkbox" :checked="eqps.enableMatrixKey"
 | 
			
		||||
          @change="handleMatrixkeyCheckboxChange" />
 | 
			
		||||
        <input
 | 
			
		||||
          type="checkbox"
 | 
			
		||||
          class="checkbox"
 | 
			
		||||
          :checked="eqps.enableMatrixKey"
 | 
			
		||||
          @change="handleMatrixkeyCheckboxChange"
 | 
			
		||||
        />
 | 
			
		||||
        <p class="mx-2">启用矩阵键盘</p>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
@@ -119,7 +148,7 @@ async function handleMatrixkeyCheckboxChange(event: Event) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
async function toggleJtagBoundaryScan() {
 | 
			
		||||
  eqps.enableJtagBoundaryScan = !eqps.enableJtagBoundaryScan;
 | 
			
		||||
  eqps.jtagBoundaryScanSetOnOff(!eqps.enableJtagBoundaryScan);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const isGettingIDCode = ref(false);
 | 
			
		||||
 
 | 
			
		||||
@@ -1,26 +1,25 @@
 | 
			
		||||
import { ref, computed, reactive } from 'vue'
 | 
			
		||||
import { defineStore } from 'pinia'
 | 
			
		||||
import { isBoolean } from 'lodash';
 | 
			
		||||
import { ref, computed, reactive } from "vue";
 | 
			
		||||
import { defineStore } from "pinia";
 | 
			
		||||
import { isBoolean } from "lodash";
 | 
			
		||||
 | 
			
		||||
// 约束电平状态类型
 | 
			
		||||
export type ConstraintLevel = 'high' | 'low' | 'undefined';
 | 
			
		||||
 | 
			
		||||
export const useConstraintsStore = defineStore('constraints', () => {
 | 
			
		||||
export type ConstraintLevel = "high" | "low" | "undefined";
 | 
			
		||||
 | 
			
		||||
export const useConstraintsStore = defineStore("constraints", () => {
 | 
			
		||||
  // 约束状态存储
 | 
			
		||||
  const constraintStates = reactive<Record<string, ConstraintLevel>>({});
 | 
			
		||||
 | 
			
		||||
  // 约束颜色映射
 | 
			
		||||
  const constraintColors = {
 | 
			
		||||
    high: '#ff3333', // 高电平为红色
 | 
			
		||||
    low: '#3333ff',  // 低电平为蓝色
 | 
			
		||||
    undefined: '#999999' // 未定义为灰色
 | 
			
		||||
    high: "#ff3333", // 高电平为红色
 | 
			
		||||
    low: "#3333ff", // 低电平为蓝色
 | 
			
		||||
    undefined: "#999999", // 未定义为灰色
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  // 获取约束状态
 | 
			
		||||
  function getConstraintState(constraint: string): ConstraintLevel {
 | 
			
		||||
    if (!constraint) return 'undefined';
 | 
			
		||||
    return constraintStates[constraint] || 'undefined';
 | 
			
		||||
    if (!constraint) return "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][] = [];
 | 
			
		||||
 | 
			
		||||
@@ -38,6 +39,8 @@ export const useConstraintsStore = defineStore('constraints', () => {
 | 
			
		||||
    Object.entries(states).forEach(([constraint, level]) => {
 | 
			
		||||
      if (isBoolean(level)) {
 | 
			
		||||
        level = level ? "high" : "low";
 | 
			
		||||
      } else {
 | 
			
		||||
        level = "low";
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (constraintStates[constraint] !== level) {
 | 
			
		||||
@@ -48,7 +51,7 @@ export const useConstraintsStore = defineStore('constraints', () => {
 | 
			
		||||
 | 
			
		||||
    // 通知所有变化
 | 
			
		||||
    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() {
 | 
			
		||||
    Object.keys(constraintStates).forEach(key => {
 | 
			
		||||
    Object.keys(constraintStates).forEach((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);
 | 
			
		||||
    return () => {
 | 
			
		||||
      const index = stateChangeCallbacks.indexOf(callback);
 | 
			
		||||
@@ -86,7 +94,7 @@ export const useConstraintsStore = defineStore('constraints', () => {
 | 
			
		||||
  // 触发约束变化
 | 
			
		||||
  function notifyConstraintChange(constraint: string, level: ConstraintLevel) {
 | 
			
		||||
    setConstraintState(constraint, level);
 | 
			
		||||
    stateChangeCallbacks.forEach(callback => callback(constraint, level));
 | 
			
		||||
    stateChangeCallbacks.forEach((callback) => callback(constraint, level));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
@@ -98,6 +106,5 @@ export const useConstraintsStore = defineStore('constraints', () => {
 | 
			
		||||
    getAllConstraintStates,
 | 
			
		||||
    onConstraintStateChange,
 | 
			
		||||
    notifyConstraintChange,
 | 
			
		||||
  }
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
  };
 | 
			
		||||
});
 | 
			
		||||
 
 | 
			
		||||
@@ -1,15 +1,16 @@
 | 
			
		||||
import { ref, reactive, watchPostEffect } from "vue";
 | 
			
		||||
import { ref, reactive, watchPostEffect, onMounted, onUnmounted } from "vue";
 | 
			
		||||
import { defineStore } from "pinia";
 | 
			
		||||
import { useLocalStorage } from "@vueuse/core";
 | 
			
		||||
import { isString, toNumber } from "lodash";
 | 
			
		||||
import { isString, toNumber, type Dictionary } from "lodash";
 | 
			
		||||
import z from "zod";
 | 
			
		||||
import { isNumber } from "mathjs";
 | 
			
		||||
import { JtagClient, MatrixKeyClient, PowerClient } from "@/APIClient";
 | 
			
		||||
import { Mutex, withTimeout } from "async-mutex";
 | 
			
		||||
import { useConstraintsStore } from "@/stores/constraints";
 | 
			
		||||
import { useDialogStore } from "./dialog";
 | 
			
		||||
import { toFileParameterOrUndefined } from "@/utils/Common";
 | 
			
		||||
import { AuthManager } from "@/utils/AuthManager";
 | 
			
		||||
import { HubConnectionBuilder } from "@microsoft/signalr";
 | 
			
		||||
import { getHubProxyFactory, getReceiverRegister } from "@/TypedSignalR.Client";
 | 
			
		||||
 | 
			
		||||
export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
  // Global Stores
 | 
			
		||||
@@ -22,13 +23,28 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
  // Jtag
 | 
			
		||||
  const jtagBitstream = ref<File>();
 | 
			
		||||
  const jtagBoundaryScanFreq = ref(100);
 | 
			
		||||
  const jtagBoundaryScanErrorCount = ref(0); // 边界扫描连续错误计数
 | 
			
		||||
  const maxJtagBoundaryScanErrors = 5; // 最大允许连续错误次数
 | 
			
		||||
  const jtagClientMutex = withTimeout(
 | 
			
		||||
    new Mutex(),
 | 
			
		||||
    1000,
 | 
			
		||||
    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
 | 
			
		||||
  const matrixKeyStates = reactive(new Array<boolean>(16).fill(false));
 | 
			
		||||
@@ -50,41 +66,6 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
  const enableMatrixKey = 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(
 | 
			
		||||
    keyNum: number | string | undefined,
 | 
			
		||||
    keyValue: boolean,
 | 
			
		||||
@@ -105,38 +86,12 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
    return false;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  async function jtagBoundaryScan() {
 | 
			
		||||
    const release = await jtagClientMutex.acquire();
 | 
			
		||||
    try {
 | 
			
		||||
      // 自动开启电源
 | 
			
		||||
      await powerSetOnOff(true);
 | 
			
		||||
      
 | 
			
		||||
      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);
 | 
			
		||||
  async function jtagBoundaryScanSetOnOff(enable: boolean) {
 | 
			
		||||
    enableJtagBoundaryScan.value = enable;
 | 
			
		||||
    if (enable) {
 | 
			
		||||
      jtagHubProxy.startBoundaryScan(jtagBoundaryScanFreq.value);
 | 
			
		||||
    } else {
 | 
			
		||||
      jtagHubProxy.stopBoundaryScan();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -144,7 +99,7 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
    try {
 | 
			
		||||
      // 自动开启电源
 | 
			
		||||
      await powerSetOnOff(true);
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      const jtagClient = AuthManager.createAuthenticatedJtagClient();
 | 
			
		||||
      const resp = await jtagClient.uploadBitstream(
 | 
			
		||||
        boardAddr.value,
 | 
			
		||||
@@ -163,7 +118,7 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
    try {
 | 
			
		||||
      // 自动开启电源
 | 
			
		||||
      await powerSetOnOff(true);
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      const jtagClient = AuthManager.createAuthenticatedJtagClient();
 | 
			
		||||
      const resp = await jtagClient.downloadBitstream(
 | 
			
		||||
        boardAddr.value,
 | 
			
		||||
@@ -184,7 +139,7 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
    try {
 | 
			
		||||
      // 自动开启电源
 | 
			
		||||
      await powerSetOnOff(true);
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      const jtagClient = AuthManager.createAuthenticatedJtagClient();
 | 
			
		||||
      const resp = await jtagClient.getDeviceIDCode(
 | 
			
		||||
        boardAddr.value,
 | 
			
		||||
@@ -204,7 +159,7 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
    try {
 | 
			
		||||
      // 自动开启电源
 | 
			
		||||
      await powerSetOnOff(true);
 | 
			
		||||
      
 | 
			
		||||
 | 
			
		||||
      const jtagClient = AuthManager.createAuthenticatedJtagClient();
 | 
			
		||||
      const resp = await jtagClient.setSpeed(
 | 
			
		||||
        boardAddr.value,
 | 
			
		||||
@@ -224,7 +179,8 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
    const release = await matrixKeypadClientMutex.acquire();
 | 
			
		||||
    console.log("set Key !!!!!!!!!!!!");
 | 
			
		||||
    try {
 | 
			
		||||
      const matrixKeypadClient = AuthManager.createAuthenticatedMatrixKeyClient();
 | 
			
		||||
      const matrixKeypadClient =
 | 
			
		||||
        AuthManager.createAuthenticatedMatrixKeyClient();
 | 
			
		||||
      const resp = await matrixKeypadClient.setMatrixKeyStatus(
 | 
			
		||||
        boardAddr.value,
 | 
			
		||||
        boardPort.value,
 | 
			
		||||
@@ -243,7 +199,8 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
    const release = await matrixKeypadClientMutex.acquire();
 | 
			
		||||
    try {
 | 
			
		||||
      if (enable) {
 | 
			
		||||
        const matrixKeypadClient = AuthManager.createAuthenticatedMatrixKeyClient();
 | 
			
		||||
        const matrixKeypadClient =
 | 
			
		||||
          AuthManager.createAuthenticatedMatrixKeyClient();
 | 
			
		||||
        const resp = await matrixKeypadClient.enabelMatrixKey(
 | 
			
		||||
          boardAddr.value,
 | 
			
		||||
          boardPort.value,
 | 
			
		||||
@@ -251,7 +208,8 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
        enableMatrixKey.value = resp;
 | 
			
		||||
        return resp;
 | 
			
		||||
      } else {
 | 
			
		||||
        const matrixKeypadClient = AuthManager.createAuthenticatedMatrixKeyClient();
 | 
			
		||||
        const matrixKeypadClient =
 | 
			
		||||
          AuthManager.createAuthenticatedMatrixKeyClient();
 | 
			
		||||
        const resp = await matrixKeypadClient.disableMatrixKey(
 | 
			
		||||
          boardAddr.value,
 | 
			
		||||
          boardPort.value,
 | 
			
		||||
@@ -290,16 +248,13 @@ export const useEquipments = defineStore("equipments", () => {
 | 
			
		||||
  return {
 | 
			
		||||
    boardAddr,
 | 
			
		||||
    boardPort,
 | 
			
		||||
    setAddr,
 | 
			
		||||
    setPort,
 | 
			
		||||
    setMatrixKey,
 | 
			
		||||
 | 
			
		||||
    // Jtag
 | 
			
		||||
    enableJtagBoundaryScan,
 | 
			
		||||
    jtagBoundaryScanSetOnOff,
 | 
			
		||||
    jtagBitstream,
 | 
			
		||||
    jtagBoundaryScanFreq,
 | 
			
		||||
    jtagBoundaryScanErrorCount,
 | 
			
		||||
    jtagClientMutex,
 | 
			
		||||
    jtagUploadBitstream,
 | 
			
		||||
    jtagDownloadBitstream,
 | 
			
		||||
    jtagGetIDCode,
 | 
			
		||||
 
 | 
			
		||||
@@ -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 }
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
@@ -1,13 +1,13 @@
 | 
			
		||||
import { fileURLToPath, URL } from 'node:url'
 | 
			
		||||
import { fileURLToPath, URL } from "node:url";
 | 
			
		||||
 | 
			
		||||
import { defineConfig } from 'vite'
 | 
			
		||||
import vue from '@vitejs/plugin-vue'
 | 
			
		||||
import vueJsx from '@vitejs/plugin-vue-jsx'
 | 
			
		||||
import vueDevTools from 'vite-plugin-vue-devtools'
 | 
			
		||||
import tailwindcss from '@tailwindcss/postcss'
 | 
			
		||||
import autoprefixer from 'autoprefixer'
 | 
			
		||||
import Components from 'unplugin-vue-components/vite'
 | 
			
		||||
import RekaResolver from 'reka-ui/resolver'
 | 
			
		||||
import { defineConfig } from "vite";
 | 
			
		||||
import vue from "@vitejs/plugin-vue";
 | 
			
		||||
import vueJsx from "@vitejs/plugin-vue-jsx";
 | 
			
		||||
import vueDevTools from "vite-plugin-vue-devtools";
 | 
			
		||||
import tailwindcss from "@tailwindcss/postcss";
 | 
			
		||||
import autoprefixer from "autoprefixer";
 | 
			
		||||
import Components from "unplugin-vue-components/vite";
 | 
			
		||||
import RekaResolver from "reka-ui/resolver";
 | 
			
		||||
 | 
			
		||||
// https://vite.dev/config/
 | 
			
		||||
export default defineConfig({
 | 
			
		||||
@@ -16,49 +16,48 @@ export default defineConfig({
 | 
			
		||||
      template: {
 | 
			
		||||
        compilerOptions: {
 | 
			
		||||
          // 将所有 wokwi- 开头的标签视为自定义元素
 | 
			
		||||
          isCustomElement: (tag) => tag.startsWith('wokwi-')
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
          isCustomElement: (tag) => tag.startsWith("wokwi-"),
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    }),
 | 
			
		||||
    vueJsx(),
 | 
			
		||||
    vueDevTools(),
 | 
			
		||||
    Components(
 | 
			
		||||
      {
 | 
			
		||||
    Components({
 | 
			
		||||
      dts: true,
 | 
			
		||||
      resolvers: [
 | 
			
		||||
        RekaResolver()
 | 
			
		||||
        RekaResolver(),
 | 
			
		||||
 | 
			
		||||
        // RekaResolver({
 | 
			
		||||
        //   prefix: '' // use the prefix option to add Prefix to the imported components
 | 
			
		||||
        // })
 | 
			
		||||
      ],
 | 
			
		||||
    }
 | 
			
		||||
    )
 | 
			
		||||
    }),
 | 
			
		||||
  ],
 | 
			
		||||
  resolve: {
 | 
			
		||||
    alias: {
 | 
			
		||||
      '@': fileURLToPath(new URL('./src', import.meta.url))
 | 
			
		||||
      "@": fileURLToPath(new URL("./src", import.meta.url)),
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
  css: {
 | 
			
		||||
    postcss: {
 | 
			
		||||
      plugins: [
 | 
			
		||||
        tailwindcss(),
 | 
			
		||||
        autoprefixer()
 | 
			
		||||
      ]
 | 
			
		||||
    }
 | 
			
		||||
      plugins: [tailwindcss(), autoprefixer()],
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
  build: {
 | 
			
		||||
    outDir: 'wwwroot',
 | 
			
		||||
    outDir: "wwwroot",
 | 
			
		||||
    emptyOutDir: true, // also necessary
 | 
			
		||||
  },
 | 
			
		||||
  server: {
 | 
			
		||||
    proxy: {
 | 
			
		||||
      "/swagger": {
 | 
			
		||||
        target: 'http://localhost:5000',
 | 
			
		||||
        changeOrigin: true
 | 
			
		||||
      }
 | 
			
		||||
        target: "http://localhost:5000",
 | 
			
		||||
        changeOrigin: true,
 | 
			
		||||
      },
 | 
			
		||||
      "/hubs": {
 | 
			
		||||
        target: "http://localhost:5000",
 | 
			
		||||
        changeOrigin: true,
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    port: 5173,
 | 
			
		||||
  }
 | 
			
		||||
})
 | 
			
		||||
  },
 | 
			
		||||
});
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user