[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, //switch中的case标签不能重复
"no-else-return": "off", //如果if语句里面有return,后面不能跟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", //禁止else语句内只有if语句
"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()之前不能使用this或super
"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", //禁止不必要的call和apply
"no-void": "off", //禁用void操作符
"no-var": 0, //禁用var用let和const代替
"no-warning-comments": "off", //不能有警告备注
"no-array-constructor": "error", // 禁止使用数组构造器
"no-caller": "error", // 禁止使用arguments.caller或arguments.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", // label名不能与var声明的变量名相同
"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", // 禁止使用new创建包装实例new 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", // 禁止比较时使用NaN只能用isNaN()
"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, //非派生类不能调用super派生类必须调用super
"default-case": "off", //switch语句最后必须有default
"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 in循环要用if语句过滤
"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", //函数名首行大写必须使用new方式调用首行小写必须用不带new方式调用
"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" // 將編譯過後的js檔放到dist資料夾中
}
}