[add] init
This commit is contained in:
commit
b437e53fe5
5
.env.dev
Normal file
5
.env.dev
Normal 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
5
.env.prod
Normal 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
367
.eslintrc.json
Normal 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
1
.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
||||
/node_modules
|
43
Dockerfile
Normal file
43
Dockerfile
Normal 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
29
certificate/RSA-cert.pem
Normal 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
29
certificate/RSA-chain.pem
Normal 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-----
|
28
certificate/RSA-privkey.pem
Normal file
28
certificate/RSA-privkey.pem
Normal 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
1948
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
31
package.json
Normal file
31
package.json
Normal 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
53
src/app.ts
Normal 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
121
src/canvas.ts
Normal 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
16
tsconfig.json
Normal 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資料夾中
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user