[add] init

This commit is contained in:
建喵 2024-09-10 10:50:39 +08:00
commit b437e53fe5
13 changed files with 2676 additions and 0 deletions

5
.env.dev Normal file
View File

@ -0,0 +1,5 @@
URLPATH = uploads
PORT = 3101
prikeyPath = ./certificate/RSA-privkey.pem
certPath = ./certificate/RSA-cert.pem
cafilePath = ./certificate/RSA-chain.pem

5
.env.prod Normal file
View File

@ -0,0 +1,5 @@
URLPATH = uploads
PORT = 3100
prikeyPath = /certificate/RSA-privkey.pem
certPath = /certificate/RSA-cert.pem
cafilePath = /certificate/RSA-chain.pem

367
.eslintrc.json Normal file
View File

@ -0,0 +1,367 @@
{
"env": {
"browser": true,
"es2021": true,
"jest": true
},
"extends": [
"eslint:recommended",
"plugin:react/recommended",
"plugin:@typescript-eslint/recommended"
],
"overrides": [],
"parser": "@typescript-eslint/parser",
"parserOptions": {
"ecmaFeatures": {
"jsx": true
},
"ecmaVersion": "latest",
"sourceType": "module"
},
"plugins": [
"react",
"@typescript-eslint",
"react-hooks"
],
"rules": {
"no-alert": 0, //使alert confirm prompt
"no-bitwise": 0, //使
"no-console": "off", //使console
"no-continue": 0, //使continue
"no-debugger": 2, //使debugger
"no-delete-var": 2, //var使delete
"no-div-regex": 1, //使/=foo/
"no-dupe-args": 2, //
"no-duplicate-case": 2, //switchcase
"no-else-return": "off", //ifreturn,else
"no-empty-label": "off", //使label
"no-eq-null": "off", //null使==!=
"no-extend-native": "off", //native
"no-extra-parens": "off", //
"no-extra-semi": 2, //
"no-floating-decimal": 2, //0 .5 3.
"no-implicit-coercion": "off", //
"no-inline-comments": 0, //
"no-invalid-this": 2, //this
"no-iterator": 2, //使__iterator__
"no-lonely-if": "off", //elseif
"no-mixed-requires": [
0,
false
], //
"no-mixed-spaces-and-tabs": "off", //tab
"no-multiple-empty-lines": [
1,
{
"max": 2
}
], //2
"no-nested-ternary": 0, //使
"no-new": "off", //使new
"no-new-require": 2, //使new require
"no-param-reassign": "off", //
"no-path-concat": 0, //node使__dirname__filename
"no-plusplus": 0, //使++--
"no-process-env": 0, //使process.env
"no-process-exit": 0, //使process.exit()
"no-redeclare": "off", //
"no-restricted-modules": 0, //使
"no-return-assign": "off", //return
"no-self-compare": 2, //
"no-sequences": 0, //使
"no-shadow": "off", //
"no-sync": 0, //nodejs
"no-ternary": 0, //使
"no-this-before-super": 0, //super()使thissuper
"no-throw-literal": 2, // throw "error";
"no-undef": "off", //
"no-undef-init": "off", //undefined
"no-undefined": "off", //使undefined
"no-unexpected-multiline": 2, //
"no-underscore-dangle": "off", //_
"no-unneeded-ternary": 2, // var isYes = answer === 1 ? true : false;
"no-unused-expressions": "off", //
"no-unused-vars": "off", //使
"no-use-before-define": "off", //使
"no-useless-call": "off", //callapply
"no-void": "off", //void
"no-var": 0, //varletconst
"no-warning-comments": "off", //
"no-array-constructor": "error", // 使
"no-caller": "error", // 使arguments.callerarguments.callee
"no-catch-shadow": "error", // catch
"no-class-assign": "error", //
"no-cond-assign": [
"error",
"except-parens"
], // 使
"no-constant-condition": "error", // 使 if(true) if(1)
"no-control-regex": "error", // 使
"no-dupe-keys": "error", // {a: 1, a: 1}
"no-empty": "error", //
"no-empty-character-class": "error", // []
"no-eval": "error", // 使eval
"no-ex-assign": "error", // catch
"no-extra-bind": "error", //
"no-extra-boolean-cast": "off", // bool
"no-fallthrough": "error", // switch穿
"no-func-assign": "error", //
"no-implied-eval": "error", // 使eval
"no-inner-declarations": "off", // 使
"no-invalid-regexp": "error", //
"no-irregular-whitespace": "error", //
"no-label-var": "error", // labelvar
"no-labels": "error", //
"no-lone-blocks": "error", //
"no-loop-func": "error", // 使
"no-multi-spaces": "error", //
"no-multi-str": "error", // \
"no-native-reassign": "error", // native
"no-negated-in-lhs": "error", // in !
"no-new-func": "error", // 使new Function
"no-new-object": "error", // 使new Object()
"no-new-wrappers": "error", // 使newnew String new Boolean new Number
"no-obj-calls": "error", // Math() JSON()
"no-octal": "error", // 使(0)
"no-octal-escape": "error", // 使
"no-proto": "error", // 使__proto__(__proto__)
"no-prototype-builtins": "off",
"no-regex-spaces": "error", // 使 /foo bar/
"no-script-url": "off", // 使javascript:void(0)
"no-shadow-restricted-names": "error", // 使
"no-spaced-func": "error", // ()
"no-sparse-arrays": "error", // [1,,2]
"no-trailing-spaces": [
"error",
{
"skipBlankLines": true
}
], // ( )
"no-unreachable": "error", //
"no-const-assign": "error", // const
"no-with": "error", // with
"comma-dangle": "off", //
"comma-spacing": "error", // 西
"curly": [
"error",
"multi-line"
], // 使 {}
"eqeqeq": "off", // 使
"indent": [
"off",
"tab",
{
"SwitchCase": 1
}
], // tab
"key-spacing": [
"error",
{
"beforeColon": false,
"afterColon": true
}
], //
"keyword-spacing": "off", //
"new-parens": "error", // new const person = new Person();
"quotes": [
"error",
"double",
{
"allowTemplateLiterals": true
}
], // ''
"semi": [
"error",
"always"
], //
"semi-spacing": [
0,
{
"before": false,
"after": true
}
], // 西
"space-before-blocks": [
"error",
"always"
], // {
// "space-before-function-paren": ["error", "never"], //
"space-infix-ops": "error", // a + b
"space-unary-ops": [
"error",
{
"words": true,
"nonwords": false
}
], // / new Foo 1++
// "spaced-comment": ["error", "always", { "markers": ["*!"] }], //
"strict": [
"error",
"global"
], // 使
"use-isnan": "error", // 使NaNisNaN()
"arrow-parens": 0, //
"arrow-spacing": 0, //=>/
"accessor-pairs": 0, //使getter/setter
"block-scoped-var": 0, //使var
"brace-style": "off", //
"callback-return": "off", //
"comma-style": [
"error",
"last"
], //
"complexity": [
0,
11
], //
"computed-property-spacing": [
0,
"never"
], //
"consistent-return": 0, //return
"consistent-this": "off", //this
"constructor-super": 0, //supersuper
"default-case": "off", //switchdefault
"dot-location": 0, //访
"dot-notation": [
0,
{
"allowKeywords": true
}
], //
"eol-last": 0, //
"func-names": 0, //
"func-style": [
0,
"declaration"
], //使/
"generator-star-spacing": 0, //*
"guard-for-in": 0, //for inif
"handle-callback-err": 0, //nodejs
"id-length": 0, //
"init-declarations": 0, //
"lines-around-comment": 0, ///
"max-depth": [
0,
4
], //
"max-len": [
0,
80,
4
], //
"max-nested-callbacks": [
0,
2
], //
"max-params": [
0,
3
], //3
"max-statements": [
0,
10
], //
"new-cap": "off", //使newnew
"newline-after-var": "off", //
"object-shorthand": 0, //
"one-var": "off", //
"operator-assignment": [
0,
"always"
], // += -=
"operator-linebreak": "off", //
"padded-blocks": 0, //
"prefer-spread": 0, //
"prefer-reflect": 0, //Reflect
"quote-props": "off", //
"radix": "off", //parseInt
"id-match": 0, //
"sort-vars": 0, //
"space-after-keywords": [
0,
"always"
], //
"space-before-function-paren": [
0,
"always"
], //
"space-in-parens": [
0,
"never"
], //
"space-return-throw-case": "off", //return throw case
"spaced-comment": 0, //
"valid-jsdoc": 0, //jsdoc
"valid-typeof": "error", //使typeof
"vars-on-top": "error", //var
"wrap-iife": [
"error",
"inside"
], //
"wrap-regex": 0, //
"yoda": [
"error",
"never"
], //
"linebreak-style": [
0,
"windows"
], //
"array-bracket-spacing": [
2,
"never"
], //
"react/react-in-jsx-scope": "off",
"camelcase": "off",
"block-spacing": "error",
"no-duplicate-imports": "error",
"require-yield": "off",
"prefer-const": "off",
"object-curly-spacing": [
"error",
"always"
],
"react/jsx-curly-spacing": [
"error",
"never"
],
"@typescript-eslint/no-namespace": "off",
"@typescript-eslint/no-inferrable-types": "off",
"@typescript-eslint/no-explicit-any": "off",
"@typescript-eslint/no-empty-function": "off",
"@typescript-eslint/ban-ts-comment": "off",
"@typescript-eslint/prefer-namespace-keyword": "off",
"@typescript-eslint/ban-types": "off",
"@typescript-eslint/no-this-alias": "off",
"@typescript-eslint/no-empty-interface": "off",
"@typescript-eslint/no-unused-vars": "off",
"@typescript-eslint/explicit-function-return-type": [
"off",
{
"allowExpressions": true
}
],
"@typescript-eslint/typedef": [
"warn",
{
"arrayDestructuring": false,
"arrowParameter": false,
"objectDestructuring": false,
"memberVariableDeclaration": true,
"parameter": true,
"propertyDeclaration": true,
"variableDeclaration": false,
"variableDeclarationIgnoreFunction": true
}
]
},
"settings": {
"import/resolver": {
"typescript": {}
},
"react": {
"version": "detect"
}
}
}

1
.gitignore vendored Normal file
View File

@ -0,0 +1 @@
/node_modules

43
Dockerfile Normal file
View File

@ -0,0 +1,43 @@
# sudo docker build -t linebotts .
# 将 Docker 镜像保存为文件
# sudo docker save -o linebotts.tar linebotts
# 使用 scp 传输文件到目标机器
# scp -P 20022 linebotts.tar jianmiau@192.168.0.77:~/Nas/docker
# 在目标机器上加载镜像
# sudo docker load -i /path/to/destination/linebotts.tar
# sudo docker run -v /volume1/homes/JianMiau/www/certificate:/certificate -e TZ=Asia/Taipei --name=linebotts --restart always --net=host linebotts
# 後續查看容器
# docker ps
# sudo docker exec -it [Container ID] /bin/bash
# 選擇node
FROM node:19.4.0
# 指定NODE_ENV為production
ENV NODE_ENV=production
# 指定預設/工作資料夾
WORKDIR /app
VOLUME ["/certificate"]
# 只copy package.json檔案
COPY ["package.json", "./"]
# 安裝dependencies
# If you are building your code for production
# RUN npm ci --only=production
RUN npm install
# copy其餘目錄及檔案
COPY . .
# 指定啟動container後執行命令
CMD [ "npm", "start" ]

29
certificate/RSA-cert.pem Normal file
View File

@ -0,0 +1,29 @@
-----BEGIN CERTIFICATE-----
MIIE5jCCA86gAwIBAgISAwEzwhx7A+2CU+spgQkLFmSdMA0GCSqGSIb3DQEBCwUA
MDMxCzAJBgNVBAYTAlVTMRYwFAYDVQQKEw1MZXQncyBFbmNyeXB0MQwwCgYDVQQD
EwNSMTEwHhcNMjQwNzI4MDI1NjU4WhcNMjQxMDI2MDI1NjU3WjAWMRQwEgYDVQQD
EwtqaWFubWlhdS50azCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBALR8
YvOMS1AhJlXyU/pHvXsNt1lTaWNqrR2pcY3fIbnlcHdnFrZ3g1VeVv0jsahgh6OE
FHflnMKecGoGys9rPIZ86FeEGnaBiaKgyPA5eyjiYLu7uGfNBJIiA0zjnpTU3AtG
QZUeAbGEQVruG382BD9kGMMHpwkVrnOP6I6XOZzzJ54lltJ94jYYT7A161NHgudv
aivco0VPqG+Wm6tn/6dyIMbNrbHM3R1SriWPiFQxzdF65KAAT9ery5X95hs6oMsx
LAV49HGyzShbNfbDrlyLZ/T24CJf+JSJL871pFMIQkQRHhfeJspvzuVGjrvg98PT
kALGi6wc5S0hBZPQok8CAwEAAaOCAg8wggILMA4GA1UdDwEB/wQEAwIFoDAdBgNV
HSUEFjAUBggrBgEFBQcDAQYIKwYBBQUHAwIwDAYDVR0TAQH/BAIwADAdBgNVHQ4E
FgQUsWmH6zp7bsGhno9zzcfUNy138l4wHwYDVR0jBBgwFoAUxc9GpOr0w8B6bJXE
LbBeki8m47kwVwYIKwYBBQUHAQEESzBJMCIGCCsGAQUFBzABhhZodHRwOi8vcjEx
Lm8ubGVuY3Iub3JnMCMGCCsGAQUFBzAChhdodHRwOi8vcjExLmkubGVuY3Iub3Jn
LzAWBgNVHREEDzANggtqaWFubWlhdS50azATBgNVHSAEDDAKMAgGBmeBDAECATCC
AQQGCisGAQQB1nkCBAIEgfUEgfIA8AB2AEiw42vapkc0D+VqAvqdMOscUgHLVt0s
gdm7v6s52IRzAAABkPd8BnQAAAQDAEcwRQIhAPc55DDfzwfV+d4Aj4K72kVrjHur
VgppzCix06gTip8DAiBVUW4+7HjY9dpxpyab4NmGfoAcGBJvt4FgkDJsL6K1XQB2
AD8XS0/XIkdYlB1lHIS+DRLtkDd/H4Vq68G/KIXs+GRuAAABkPd8Bn0AAAQDAEcw
RQIhAPkbBXjJOkDQ9Jls+RbLVyHdpVwbVxwqjdGKuBraVDYiAiBcbEpr45ikUDYj
P/DyjrGiB60WsMXO4NGiQK0TjInv6jANBgkqhkiG9w0BAQsFAAOCAQEAoH2tgoVJ
dfM0ViCs90N533xOJgS5CjC80io+ablm6NuHrgAwtcbej6QfsaH9od6PGMlnmlq+
x61vE2MZjX5jakC3DBOwVt+TRJWYyj+rsCJunuWqt2R3vGNUj2UR6dQpQmhx6Ekl
7NJdEqEwh/GfbiD3TfutBMAlgaAoCBq4wpoqOFUanO9OUUWFznkOy2TEcGu98jnO
hMUz5XH40XqKo1g6aEhsU5B0Xh/bJ8Bmt27H1B3AqZAufEgWEZki+JV9lM6v6BQl
qOJ89L5VwYh6CpB/rg4KqyChRTqmAQDZEzXJGLqnzwllPqF/g/TyHw3TWRokmoWJ
AUs6GBTSVcAk8w==
-----END CERTIFICATE-----

29
certificate/RSA-chain.pem Normal file
View File

@ -0,0 +1,29 @@
-----BEGIN CERTIFICATE-----
MIIFBjCCAu6gAwIBAgIRAIp9PhPWLzDvI4a9KQdrNPgwDQYJKoZIhvcNAQELBQAw
TzELMAkGA1UEBhMCVVMxKTAnBgNVBAoTIEludGVybmV0IFNlY3VyaXR5IFJlc2Vh
cmNoIEdyb3VwMRUwEwYDVQQDEwxJU1JHIFJvb3QgWDEwHhcNMjQwMzEzMDAwMDAw
WhcNMjcwMzEyMjM1OTU5WjAzMQswCQYDVQQGEwJVUzEWMBQGA1UEChMNTGV0J3Mg
RW5jcnlwdDEMMAoGA1UEAxMDUjExMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIB
CgKCAQEAuoe8XBsAOcvKCs3UZxD5ATylTqVhyybKUvsVAbe5KPUoHu0nsyQYOWcJ
DAjs4DqwO3cOvfPlOVRBDE6uQdaZdN5R2+97/1i9qLcT9t4x1fJyyXJqC4N0lZxG
AGQUmfOx2SLZzaiSqhwmej/+71gFewiVgdtxD4774zEJuwm+UE1fj5F2PVqdnoPy
6cRms+EGZkNIGIBloDcYmpuEMpexsr3E+BUAnSeI++JjF5ZsmydnS8TbKF5pwnnw
SVzgJFDhxLyhBax7QG0AtMJBP6dYuC/FXJuluwme8f7rsIU5/agK70XEeOtlKsLP
Xzze41xNG/cLJyuqC0J3U095ah2H2QIDAQABo4H4MIH1MA4GA1UdDwEB/wQEAwIB
hjAdBgNVHSUEFjAUBggrBgEFBQcDAgYIKwYBBQUHAwEwEgYDVR0TAQH/BAgwBgEB
/wIBADAdBgNVHQ4EFgQUxc9GpOr0w8B6bJXELbBeki8m47kwHwYDVR0jBBgwFoAU
ebRZ5nu25eQBc4AIiMgaWPbpm24wMgYIKwYBBQUHAQEEJjAkMCIGCCsGAQUFBzAC
hhZodHRwOi8veDEuaS5sZW5jci5vcmcvMBMGA1UdIAQMMAowCAYGZ4EMAQIBMCcG
A1UdHwQgMB4wHKAaoBiGFmh0dHA6Ly94MS5jLmxlbmNyLm9yZy8wDQYJKoZIhvcN
AQELBQADggIBAE7iiV0KAxyQOND1H/lxXPjDj7I3iHpvsCUf7b632IYGjukJhM1y
v4Hz/MrPU0jtvfZpQtSlET41yBOykh0FX+ou1Nj4ScOt9ZmWnO8m2OG0JAtIIE38
01S0qcYhyOE2G/93ZCkXufBL713qzXnQv5C/viOykNpKqUgxdKlEC+Hi9i2DcaR1
e9KUwQUZRhy5j/PEdEglKg3l9dtD4tuTm7kZtB8v32oOjzHTYw+7KdzdZiw/sBtn
UfhBPORNuay4pJxmY/WrhSMdzFO2q3Gu3MUBcdo27goYKjL9CTF8j/Zz55yctUoV
aneCWs/ajUX+HypkBTA+c8LGDLnWO2NKq0YD/pnARkAnYGPfUDoHR9gVSp/qRx+Z
WghiDLZsMwhN1zjtSC0uBWiugF3vTNzYIEFfaPG7Ws3jDrAMMYebQ95JQ+HIBD/R
PBuHRTBpqKlyDnkSHDHYPiNX3adPoPAcgdF3H2/W0rmoswMWgTlLn1Wu0mrks7/q
pdWfS6PJ1jty80r2VKsM/Dj3YIDfbjXKdaFU5C+8bhfJGqU3taKauuz0wHVGT3eo
6FlWkWYtbt4pgdamlwVeZEW+LM7qZEJEsMNPrfC03APKmZsJgpWCDWOKZvkZcvjV
uYkQ4omYCTX5ohy+knMjdOmdH9c7SpqEWBDC86fiNex+O0XOMEZSa8DA
-----END CERTIFICATE-----

View File

@ -0,0 +1,28 @@
-----BEGIN PRIVATE KEY-----
MIIEvQIBADANBgkqhkiG9w0BAQEFAASCBKcwggSjAgEAAoIBAQC0fGLzjEtQISZV
8lP6R717DbdZU2ljaq0dqXGN3yG55XB3Zxa2d4NVXlb9I7GoYIejhBR35ZzCnnBq
BsrPazyGfOhXhBp2gYmioMjwOXso4mC7u7hnzQSSIgNM456U1NwLRkGVHgGxhEFa
7ht/NgQ/ZBjDB6cJFa5zj+iOlzmc8yeeJZbSfeI2GE+wNetTR4Lnb2or3KNFT6hv
lpurZ/+nciDGza2xzN0dUq4lj4hUMc3ReuSgAE/Xq8uV/eYbOqDLMSwFePRxss0o
WzX2w65ci2f09uAiX/iUiS/O9aRTCEJEER4X3ibKb87lRo674PfD05ACxousHOUt
IQWT0KJPAgMBAAECggEAWab32Ba2SmVND6BByq7cFkXn730ZeoLA310NxPUzYY3w
4b7Zb2XKXRtxhmi2lPbuKXEwYaYyyhG9sU0SbRnNhNiC6QX7xyXtYlUPuxQyc0qP
1nEW1qjmPlia3xSp7zAU5ZzOcu3m1XDmre5cxkHktkBYdHhWppKYh1rSoBHKUoU3
wvkIKT0RJKwbmTJRqTNA/11uuNZjq8ysNt9cimF3aXX2ZobqRIi9VwYlvavq2hXo
pdEky9H965+v9FwTTMqLOwx0WgQJ3KPsn0WiJ0aH0JJfsomk9bhlROJUS/79keXr
OGjW/uAv2UNwSmBH1o3mKUnRTZipZ9VNEseZYuNhoQKBgQDpS69KaGaZFXvd83Tm
ekdwx6fKTjgQh5yTCDjFLtdPjBGI9gHGvA9NFr6+KLffrafyNDFbKBX7VwhBicBi
ZW+12WuI2Q+KPHMw3NRz1qEZjBYQ9g3vFR9PKF8OUQsVaFqEe1aA2vPS2kluNw+T
k1OqDqbBZ6yesuyI5BVTgS+lpwKBgQDGDQCVV3m06VWGoRLoCcvSX/YRVtDv5RA1
ETamJEEHV14j/i0j8MtHNU8XWwXMvuEH49vxAxlzzb051grXrG+WNOqGCsRiixOI
E1RK5v2s7pNQkljA/iy+JD6xggrh04QJ0zQBYOgi0yNMbLhXAG7BM+S04UNMI01a
8qXnjxODGQKBgEbhv/iTj9ijNmdROQtty5bwkoJdEZu0GFZ0AQuoF7MLk6hRVmjT
arK5XmrYZEWJtaVZRkW0ADnFT7TZ7aH3v+E4lfuWN6qAg18tOT+Yzom8jlfI6qLh
gAnE8lyfMwbmFdp6vuWXoM1HlVfvUsQ71wesO+43WbM+Ga/d3LzqW1exAoGAYX0d
AGZi3o7NLswzBk1sK05ZTgeyKaRT6gtjHz1RVU/IY2dGyR5Kse6n1BNWM4byNnQP
W//uk3Z+4u1dwPR8qS+7EehS6z8SijUZlRVHYcy+bzbawYVceOxWgAJHYQpBQKTa
QKN3IU1VXtVVmF36JthoiDEqc1wdQ9uVlvpy3GECgYEAoEM/5sw/8emBUPQrsnKT
ytDziHkDr/rTsgrYkbAc1xttaFF6eSaghuzujuzOA81sKOMuoXqeiPfAxr61CsU9
0Sc/T2IBFH9qjEnc186eCLaKn9EOqr6n7/HR+w4Sb6F2+j6G9z3fHZwJnR8J2eps
CAgOgNOx8Il49o/yRgmXpbc=
-----END PRIVATE KEY-----

1948
package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

31
package.json Normal file
View File

@ -0,0 +1,31 @@
{
"name": "api",
"version": "1.0.0",
"description": "",
"main": "src/app.ts",
"scripts": {
"start": "NODE_ENV=prod nodemon src/app.ts",
"test": "NODE_ENV=test nodemon src/app.ts",
"dev": "NODE_ENV=dev nodemon --exec \"node --require ts-node/register --inspect=192.168.0.15:9229 src/app.ts\"",
"build": "tsc --project ./"
},
"author": "",
"license": "ISC",
"keywords": [],
"dependencies": {
"canvas": "^2.11.2",
"dateformat": "^4.5.1",
"dayjs": "^1.11.7",
"dotenv": "^16.0.3",
"express": "^4.20.0",
"fs": "^0.0.1-security",
"multer": "^1.4.5-lts.1",
"nodemon": "^3.1.4"
},
"devDependencies": {
"@types/dateformat": "^5.0.0",
"@types/express": "^4.17.15",
"@types/multer": "^1.4.12",
"@types/node": "^22.5.4"
}
}

53
src/app.ts Normal file
View File

@ -0,0 +1,53 @@
// 背景執行 forever start -c ts-node -a -l canvas.log src/app.ts
// 重新背景執行 forever restart -a -l canvas.log src/app.ts
// 監聽檔案變化 "npm start"
// 連線Debug "npm run dev"
import dayjs from "dayjs";
import "dayjs/locale/zh-tw";
import dotenv from "dotenv";
import dateFormat from "dateformat";
import express from 'express';
import fs from "fs";
import https from 'https';
import multer from 'multer';
import { generateImage } from "./canvas";
dayjs.locale("zh-tw");
if (process.env.NODE_ENV) {
dotenv.config({ path: `./.env.${process.env.NODE_ENV}` });
} else {
dotenv.config();
}
const path: string = process.env.URLPATH || "/";
const port: number = +process.env.PORT || 3000;
const app = express();
const upload = multer({ dest: `${path}/` }); // 用于处理文件上传
//讀取憑證及金鑰
const prikey: string = fs.readFileSync(process.env.prikeyPath, "utf8");
const cert: string = fs.readFileSync(process.env.certPath, "utf8");
const cafile: string = fs.readFileSync(process.env.cafilePath, "utf-8");
//建立憑證及金鑰
const credentials: Object = {
key: prikey,
cert: cert,
ca: cafile
};
// 处理 POST 请求并返回生成的图片
app.post(`/${path}`, upload.single('file'), (req, res) => {
const imageBuffer = generateImage();
res.set('Content-Type', 'image/png');
res.send(imageBuffer);
});
// 创建 HTTPS 服务器
https.createServer(credentials, app).listen(port, () => {
let dateTime: string = dateFormat(new Date(), "yyyy-mm-dd HH:MM:ss");
console.log(`${dateTime} listening on ${port} path: ${path}`);
console.log(`${dateTime} [api已準備就緒]`);
});

121
src/canvas.ts Normal file
View File

@ -0,0 +1,121 @@
import { createCanvas } from 'canvas';
// 生成图片的函数
export function generateImage() {
const boys = ['男生A', '男生B', '男生C'];
const girls = ['女生A', '女生B', '女生C'];
const rounds = 3; // 可以根据需要设置轮数
const person = boys.length > girls.length ? boys.length : girls.length;
if (boys.length > girls.length) {
girls.push("那個")
} else if (boys.length < girls.length) {
boys.push("那個")
}
const canvasWidth = 800; // 根据需要调整宽度
const canvasHeight = (rounds * 100) + (person * 10); // 根据需要调整高度
const canvas = createCanvas(canvasWidth, canvasHeight);
const ctx = canvas.getContext('2d');
const font = "PMingLiU";
// 设定背景颜色
ctx.fillStyle = 'white';
ctx.fillRect(0, 0, canvas.width, canvas.height);
// 追踪已使用的组合
let usedCombinations = new Set<string>();
// 生成不重复的队伍组合并确保每人上场
function generateTeams(): { teamA: string, teamB: string }[] {
const roundTeams: { teamA: string, teamB: string }[] = [];
const availableBoys = [...boys];
const availableGirls = [...girls];
let attempts = 0;
while (roundTeams.length < boys.length) {
const boyIndex = Math.floor(Math.random() * availableBoys.length);
const girlIndex = Math.floor(Math.random() * availableGirls.length);
const team = `${availableBoys[boyIndex]}-${availableGirls[girlIndex]}`;
// 如果组合未使用过,添加到队伍列表
if (!usedCombinations.has(team)) {
roundTeams.push({
teamA: availableBoys[boyIndex],
teamB: availableGirls[girlIndex]
});
usedCombinations.add(team);
availableBoys.splice(boyIndex, 1);
availableGirls.splice(girlIndex, 1);
}
// 如果尝试过多次未找到合适组合,强制生成剩余队伍
attempts++;
if (attempts > 10 && roundTeams.length < boys.length) {
for (let i = 0; i < availableBoys.length; i++) {
roundTeams.push({
teamA: availableBoys[i],
teamB: availableGirls[i % availableGirls.length]
});
}
break;
}
}
return roundTeams;
}
// 生成所有轮次的队伍组合
let teamCombinations: { teamA: string, teamB: string }[][] = [];
for (let i = 0; i < rounds; i++) {
const newRound = generateTeams();
teamCombinations.push(newRound);
// 如果所有组合都用过了,重置组合记录
if (usedCombinations.size >= boys.length * girls.length) {
usedCombinations.clear();
}
}
// 绘制标题和日期
ctx.fillStyle = 'black';
ctx.font = `bold 24px "${font}"`;
ctx.textAlign = 'center';
ctx.fillText('比賽隊伍', canvasWidth / 2, 40);
ctx.font = `italic 16px "${font}"`;
ctx.fillText('2024-09-09', canvasWidth / 2, 70);
const roundTitleYStart = 120;
const teamYStart = 60;
const lineHeight = 30;
const roundsPerRow = 3;
const rowHeight = 300;
teamCombinations.forEach((roundTeams, roundIndex) => {
const rowIndex = Math.floor(roundIndex / roundsPerRow);
const colIndex = roundIndex % roundsPerRow;
const roundXPosition = (canvasWidth / roundsPerRow) * colIndex + (canvasWidth / roundsPerRow) / 2;
const roundYPosition = roundTitleYStart + rowIndex * rowHeight;
ctx.font = `bold 18px "${font}"`;
ctx.textAlign = 'center';
ctx.fillText(`${roundIndex + 1}`, roundXPosition, roundYPosition);
ctx.font = `16px "${font}"`;
roundTeams.forEach((team, teamIndex) => {
const teamYPosition = roundYPosition + teamYStart + teamIndex * lineHeight;
if (teamIndex === 0) {
ctx.fillText('1號隊友', roundXPosition - 50, teamYPosition);
ctx.fillText('2號隊友', roundXPosition + 50, teamYPosition);
}
ctx.fillText(team.teamA, roundXPosition - 50, teamYPosition + lineHeight);
ctx.fillText(team.teamB, roundXPosition + 50, teamYPosition + lineHeight);
});
});
// 导出图片
const buffer = canvas.toBuffer('image/png');
// fs.writeFileSync('./team_combination.png', buffer);
console.log('圖片已生成!');
return buffer;
}

16
tsconfig.json Normal file
View File

@ -0,0 +1,16 @@
{
"compilerOptions": {
"target": "es5", /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */
"module": "commonjs", /* Specify what module code is generated. */
"lib": [
"es2015",
"es2017",
"dom"
],
"esModuleInterop": true, /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables `allowSyntheticDefaultImports` for type compatibility. */
"forceConsistentCasingInFileNames": true, /* Ensure that casing is correct in imports. */
"strict": false, /* Enable all strict type-checking options. */
"skipLibCheck": true /* Skip type checking all .d.ts files. */,
"outDir": "dist" // jsdist
}
}