@dazhicheng/common 1.0.3 → 1.0.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/.env ADDED
@@ -0,0 +1,20 @@
1
+ # 【通用】环境变量
2
+
3
+ # 版本号
4
+ VITE_VERSION = 1.0.0
5
+
6
+
7
+ # 应用部署基础路径(如部署在子目录 /admin,则设置为 /admin/)
8
+ VITE_BASE_URL = /
9
+
10
+ # 权限模式【 frontend 前端模式 / backend 后端模式 】
11
+ VITE_ACCESS_MODE = backend
12
+
13
+ # 跨域请求时是否携带 Cookie(开启前需确保后端支持)
14
+ VITE_WITH_CREDENTIALS = false
15
+
16
+ # 是否打开路由信息
17
+ VITE_OPEN_ROUTE_INFO = false
18
+
19
+ # 锁屏加密密钥
20
+ VITE_LOCK_ENCRYPT_KEY = s3cur3k3y4adpro
package/.gitattributes ADDED
@@ -0,0 +1,3 @@
1
+ * text=auto
2
+ *.html linguist-detectable=false
3
+ *.vue linguist-detectable=true
package/.gitignore ADDED
@@ -0,0 +1,8 @@
1
+ node_modules
2
+ .DS_Store
3
+ dist
4
+ dist-ssr
5
+ *.local
6
+ .cursorrules
7
+ .vite
8
+ .claude
package/.prettierrc ADDED
@@ -0,0 +1,20 @@
1
+ {
2
+ "printWidth": 100,
3
+ "tabWidth": 2,
4
+ "useTabs": false,
5
+ "semi": true,
6
+ "vueIndentScriptAndStyle": true,
7
+ "singleQuote": true,
8
+ "quoteProps": "as-needed",
9
+ "bracketSpacing": true,
10
+ "trailingComma": "all",
11
+ "bracketSameLine": false,
12
+ "jsxSingleQuote": false,
13
+ "arrowParens": "always",
14
+ "insertPragma": false,
15
+ "requirePragma": false,
16
+ "proseWrap": "never",
17
+ "htmlWhitespaceSensitivity": "strict",
18
+ "endOfLine": "auto",
19
+ "rangeStart": 0
20
+ }
@@ -0,0 +1,5 @@
1
+ dist
2
+ node_modules
3
+ public
4
+ .husky
5
+ .vscode
@@ -0,0 +1,76 @@
1
+ module.exports = {
2
+ extends: [
3
+ 'stylelint-config-standard',
4
+ 'stylelint-config-recommended-scss',
5
+ 'stylelint-config-recommended-vue/scss',
6
+ 'stylelint-config-html/vue',
7
+ 'stylelint-config-recess-order',
8
+ ],
9
+ overrides: [
10
+ {
11
+ files: ['**/*.{vue,html}'],
12
+ customSyntax: 'postcss-html',
13
+ },
14
+ {
15
+ files: ['**/*.{css,scss}'],
16
+ customSyntax: 'postcss-scss',
17
+ },
18
+ ],
19
+ rules: {
20
+ 'import-notation': 'string',
21
+ 'selector-class-pattern': null,
22
+ 'custom-property-pattern': null,
23
+ 'keyframes-name-pattern': null,
24
+ 'no-descending-specificity': null,
25
+ 'no-empty-source': null,
26
+ 'property-no-vendor-prefix': null,
27
+ 'selector-pseudo-class-no-unknown': [
28
+ true,
29
+ {
30
+ ignorePseudoClasses: ['global', 'export', 'deep'],
31
+ },
32
+ ],
33
+ 'property-no-unknown': [
34
+ true,
35
+ {
36
+ ignoreProperties: [],
37
+ },
38
+ ],
39
+ 'at-rule-no-unknown': [
40
+ true,
41
+ {
42
+ ignoreAtRules: [
43
+ 'apply',
44
+ 'use',
45
+ 'mixin',
46
+ 'include',
47
+ 'extend',
48
+ 'each',
49
+ 'if',
50
+ 'else',
51
+ 'for',
52
+ 'while',
53
+ 'reference',
54
+ ],
55
+ },
56
+ ],
57
+ 'scss/at-rule-no-unknown': [
58
+ true,
59
+ {
60
+ ignoreAtRules: [
61
+ 'apply',
62
+ 'use',
63
+ 'mixin',
64
+ 'include',
65
+ 'extend',
66
+ 'each',
67
+ 'if',
68
+ 'else',
69
+ 'for',
70
+ 'while',
71
+ 'reference',
72
+ ],
73
+ },
74
+ ],
75
+ },
76
+ };
package/Dockerfile ADDED
@@ -0,0 +1,33 @@
1
+ # 阶段 1:构建前端项目
2
+ FROM node:24-alpine AS builder
3
+
4
+ WORKDIR /app
5
+
6
+ # 安装 pnpm 并配置镜像源
7
+ RUN npm install -g pnpm && \
8
+ pnpm config set registry https://registry.npmmirror.com
9
+
10
+ ARG NODE_ENV=dev
11
+
12
+ # 先复制依赖文件并安装依赖
13
+ COPY package*.json pnpm-lock.yaml* pnpm-workspace.yaml ./
14
+ RUN pnpm install
15
+
16
+
17
+ # 再复制全部源代码,进行构建
18
+ COPY . .
19
+ RUN pnpm "build:${NODE_ENV}"
20
+
21
+ # 阶段 2:部署到 Nginx
22
+ FROM nginx:stable-alpine
23
+
24
+ ARG NODE_ENV=dev
25
+
26
+ # 复制 Nginx 配置
27
+ COPY "nginx_${NODE_ENV}.conf" /etc/nginx/conf.d/default.conf
28
+
29
+ # 复制构建产物
30
+ COPY --from=builder /app/dist /usr/share/nginx/html
31
+
32
+ # 启动 Nginx
33
+ CMD ["nginx", "-g", "daemon off;"]
@@ -0,0 +1,87 @@
1
+ import pluginJs from '@eslint/js';
2
+ import eslintPluginPrettierRecommended from 'eslint-plugin-prettier/recommended';
3
+ import pluginVue from 'eslint-plugin-vue';
4
+ import globals from 'globals';
5
+ import tseslint from 'typescript-eslint';
6
+
7
+ export default [
8
+ // 指定文件匹配规则
9
+ {
10
+ files: ['**/*.{js,mjs,cjs,ts,vue}'],
11
+ },
12
+ // 指定全局变量和环境
13
+ {
14
+ languageOptions: {
15
+ globals: {
16
+ ...globals.browser,
17
+ ...globals.node,
18
+ },
19
+ },
20
+ },
21
+ // 扩展配置
22
+ pluginJs.configs.recommended,
23
+ ...tseslint.configs.recommended,
24
+ ...pluginVue.configs['flat/essential'],
25
+ // 自定义规则
26
+ {
27
+ // 针对所有 JavaScript、TypeScript 和 Vue 文件应用以下配置
28
+ files: ['**/*.{js,mjs,cjs,ts,vue}'],
29
+
30
+ languageOptions: {
31
+ globals: {
32
+ // 合并从 autoImportConfig 中读取的全局变量配置
33
+ ...autoImportConfig.globals,
34
+ },
35
+ },
36
+ rules: {
37
+ 'no-undef': 'off', // TypeScript 自身已处理未定义变量检查
38
+ quotes: ['error', 'single'], // 使用单引号
39
+ semi: ['error', 'never'], // 语句末尾不加分号
40
+ 'no-var': 'error', // 要求使用 let 或 const 而不是 var
41
+ '@typescript-eslint/no-explicit-any': 'off', // 禁用 any 检查
42
+ 'vue/multi-word-component-names': 'off', // 禁用对 Vue 组件名称的多词要求检查
43
+ 'no-multiple-empty-lines': ['warn', { max: 1, maxEOF: 1 }], // 不允许多个空行
44
+ 'no-unexpected-multiline': 'error', // 禁止空余的多行
45
+ 'no-unused-vars': 'off', // 禁用未使用的变量检查
46
+ '@typescript-eslint/no-unused-vars': 'off', // 禁用 TypeScript 未使用的变量检查
47
+ },
48
+ },
49
+ {
50
+ files: ['**/*.vue'],
51
+ languageOptions: {
52
+ parserOptions: {
53
+ parser: tseslint.parser,
54
+ ecmaFeatures: {
55
+ jsx: true,
56
+ },
57
+ },
58
+ },
59
+ },
60
+ // vue 规则
61
+ {
62
+ files: ['**/*.vue'],
63
+ languageOptions: {
64
+ parserOptions: {
65
+ parser: tseslint.parser,
66
+ ecmaFeatures: {
67
+ jsx: true,
68
+ },
69
+ },
70
+ },
71
+ },
72
+ // 忽略文件
73
+ {
74
+ ignores: [
75
+ 'node_modules',
76
+ 'dist',
77
+ 'public',
78
+ '.vscode/**',
79
+ 'src/assets/**',
80
+ 'src/utils/console.ts',
81
+ 'src/api',
82
+ 'src/api/**',
83
+ ],
84
+ },
85
+ // prettier 配置
86
+ eslintPluginPrettierRecommended,
87
+ ];
package/nginx_dev.conf ADDED
@@ -0,0 +1,26 @@
1
+ # HTTP 服务器 - 保持原有配置
2
+ server_name localhost;
3
+ charset utf-8;
4
+
5
+ root /usr/share/nginx/html;
6
+ index index.html;
7
+
8
+ location / {
9
+ try_files $uri $uri/ /index.html;
10
+ # CORS 配置
11
+ add_header 'Access-Control-Allow-Origin' '*' always;
12
+ add_header 'Access-Control-Allow-Methods' 'GET, POST, OPTIONS' always;
13
+ add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization' always;
14
+ }
15
+
16
+ location /dev-api/ {
17
+ proxy_set_header Host $host;
18
+ proxy_set_header X-Real-Ip $remote_addr;
19
+ proxy_set_header X-Forwarded-For $remote_addr;
20
+ proxy_pass http://116.6.49.180:11000/;
21
+ }
22
+
23
+ error_page 500 502 503 504 /50x.html;
24
+ location = /50x.html {
25
+ root html;
26
+ }
@@ -0,0 +1,26 @@
1
+ # HTTP 服务器 - 保持原有配置
2
+ server_name localhost;
3
+ charset utf-8;
4
+
5
+ root /usr/share/nginx/html;
6
+ index index.html;
7
+
8
+ location / {
9
+ try_files $uri $uri/ /index.html;
10
+ # CORS 配置
11
+ add_header 'Access-Control-Allow-Origin' '*' always;
12
+ add_header 'Access-Control-Allow-Methods' 'GET, POST, OPTIONS' always;
13
+ add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization' always;
14
+ }
15
+
16
+ location /test-api/ {
17
+ proxy_set_header Host $host;
18
+ proxy_set_header X-Real-Ip $remote_addr;
19
+ proxy_set_header X-Forwarded-For $remote_addr;
20
+ proxy_pass http://116.6.49.180:11001/;
21
+ }
22
+
23
+ error_page 500 502 503 504 /50x.html;
24
+ location = /50x.html {
25
+ root html;
26
+ }
package/package.json CHANGED
@@ -1,12 +1,29 @@
1
1
  {
2
2
  "name": "@dazhicheng/common",
3
- "version": "1.0.3",
3
+ "version": "1.0.5",
4
4
  "description": "共享配置文件",
5
5
  "type": "module",
6
+ "types": "./src/types/index.d.ts",
6
7
  "files": [
7
8
  "commitlint.config.cjs",
8
- ".husky"
9
+ ".husky",
10
+ ".prettierrc",
11
+ ".stylelintrc.cjs",
12
+ ".stylelintignore",
13
+ ".gitignore",
14
+ ".gitattributes",
15
+ ".env",
16
+ "eslint.config.mjs",
17
+ "tsconfig.json",
18
+ "Dockerfile",
19
+ "nginx_dev.conf",
20
+ "nginx_test.conf",
21
+ "sync-common.mjs",
22
+ "src/types/index.d.ts"
9
23
  ],
24
+ "exports": {
25
+ "./sync-common.mjs": "./sync-common.mjs"
26
+ },
10
27
  "publishConfig": {
11
28
  "access": "public"
12
29
  },
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,277 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import { createRequire } from 'module';
4
+ import { fileURLToPath } from 'url';
5
+
6
+ const __filename = fileURLToPath(import.meta.url);
7
+ const __dirname = path.dirname(__filename);
8
+ const require = createRequire(import.meta.url);
9
+ const commonDir = __dirname;
10
+
11
+ const markerPattern = /^<<<(before|common|after)>>>$/gm;
12
+ const textLikeFiles = new Set([
13
+ '.gitignore',
14
+ '.stylelintignore',
15
+ '.gitattributes',
16
+ '.prettierrc',
17
+ '.env',
18
+ ]);
19
+ const directLikeFiles = new Set([
20
+ 'Dockerfile',
21
+ 'nginx_dev.conf',
22
+ 'nginx_test.conf',
23
+ 'commit-msg',
24
+ 'pre-commit',
25
+ '.stylelintrc.cjs',
26
+ 'commitlint.config.cjs',
27
+ 'eslint.config.mjs',
28
+ 'tsconfig.json',
29
+ ]);
30
+ const defaultManagedFiles = [
31
+ '.gitignore',
32
+ '.stylelintignore',
33
+ '.gitattributes',
34
+ '.prettierrc',
35
+ '.env',
36
+ '.stylelintrc.cjs',
37
+ 'commitlint.config.cjs',
38
+ 'Dockerfile',
39
+ 'eslint.config.mjs',
40
+ 'tsconfig.json',
41
+ { name: 'nginx_dev.conf' },
42
+ { name: 'nginx_test.conf' },
43
+ { name: 'commit-msg', dir: '.husky' },
44
+ { name: 'pre-commit', dir: '.husky' },
45
+ ];
46
+
47
+ const getFileRelativePath = (file) => (file.dir ? path.join(file.dir, file.name) : file.name);
48
+ const getFileKey = (file) => `${file.dir ?? ''}::${file.name}`;
49
+
50
+ const ensureCommonFile = (file) => {
51
+ const filePath = path.join(commonDir, getFileRelativePath(file));
52
+ if (!fs.existsSync(filePath)) {
53
+ throw new Error(`Missing common file: ${filePath}`);
54
+ }
55
+ return filePath;
56
+ };
57
+
58
+ const parseLocalSections = (filePath) => {
59
+ if (!filePath || !fs.existsSync(filePath)) {
60
+ return { before: '', after: '', hasMarkers: false };
61
+ }
62
+
63
+ const rawContent = fs.readFileSync(filePath, 'utf8').trim();
64
+ if (!rawContent) {
65
+ return { before: '', after: '', hasMarkers: false };
66
+ }
67
+
68
+ const markers = [...rawContent.matchAll(markerPattern)];
69
+ if (markers.length === 0) {
70
+ return { before: '', after: rawContent, hasMarkers: false };
71
+ }
72
+
73
+ const sections = {
74
+ before: '',
75
+ common: '',
76
+ after: '',
77
+ };
78
+
79
+ for (let index = 0; index < markers.length; index += 1) {
80
+ const current = markers[index];
81
+ const next = markers[index + 1];
82
+ const sectionName = current[1];
83
+ const start = current.index + current[0].length;
84
+ const end = next ? next.index : rawContent.length;
85
+ sections[sectionName] = rawContent.slice(start, end).trim();
86
+ }
87
+
88
+ return {
89
+ before: sections.before,
90
+ after: sections.after,
91
+ hasMarkers: true,
92
+ };
93
+ };
94
+
95
+ const getDefaultHeader = (fileName) => {
96
+ if (!textLikeFiles.has(fileName)) {
97
+ return '';
98
+ }
99
+
100
+ return `# base\n# 由 scripts/sync-common.mjs 根据 @dazhicheng/common/${fileName} 自动生成,请勿直接修改\n\n`;
101
+ };
102
+
103
+ const getDefaultLocalFiles = (file) => ({
104
+ merged: path.join('local', getFileRelativePath(file)),
105
+ });
106
+
107
+ const getDefaultBlockHeaders = (fileName) => {
108
+ if (textLikeFiles.has(fileName)) {
109
+ return {
110
+ before: '# project specific\n',
111
+ common: '',
112
+ after: '\n\n# project specific\n',
113
+ };
114
+ }
115
+
116
+ if (directLikeFiles.has(fileName)) {
117
+ return {
118
+ before: '# 这里是增量\n\n',
119
+ common: '# 这里是 common 的内容\n\n',
120
+ after: '\n\n# 这里还是增量\n\n',
121
+ };
122
+ }
123
+
124
+ return {
125
+ before: '',
126
+ common: '',
127
+ after: '',
128
+ };
129
+ };
130
+
131
+ const getDefaultFileConfig = (file) => ({
132
+ name: file.name,
133
+ dir: file.dir,
134
+ header: getDefaultHeader(file.name),
135
+ localFiles: getDefaultLocalFiles(file),
136
+ blockHeaders: getDefaultBlockHeaders(file.name),
137
+ });
138
+
139
+ const mergeFileConfig = (defaultFile, projectFile = {}) => ({
140
+ ...defaultFile,
141
+ ...projectFile,
142
+ localFiles: {
143
+ ...(defaultFile.localFiles ?? {}),
144
+ ...(projectFile.localFiles ?? {}),
145
+ },
146
+ blockHeaders: {
147
+ ...(defaultFile.blockHeaders ?? {}),
148
+ ...(projectFile.blockHeaders ?? {}),
149
+ },
150
+ });
151
+
152
+ const normalizeFileEntry = (file) => {
153
+ if (typeof file === 'string') {
154
+ return { name: file };
155
+ }
156
+
157
+ return file;
158
+ };
159
+
160
+ const normalizeFiles = (config = {}) => {
161
+ if (Array.isArray(config)) {
162
+ return config.map(normalizeFileEntry);
163
+ }
164
+
165
+ if (Array.isArray(config.files)) {
166
+ return config.files.map(normalizeFileEntry);
167
+ }
168
+
169
+ return [
170
+ ...(config.textFiles ?? []).map((file) => ({ type: 'textFiles', ...normalizeFileEntry(file) })),
171
+ ...(config.directFiles ?? []).map((file) => ({
172
+ type: 'directFiles',
173
+ ...normalizeFileEntry(file),
174
+ })),
175
+ ];
176
+ };
177
+
178
+ const mergeManagedFiles = (projectConfig = {}, runtimeConfig = {}) => {
179
+ const defaultFiles = normalizeFiles(defaultManagedFiles).map((file) =>
180
+ mergeFileConfig(getDefaultFileConfig(file), file),
181
+ );
182
+ const projectFiles = normalizeFiles(projectConfig);
183
+ const runtimeFiles = normalizeFiles(runtimeConfig);
184
+ const mergedMap = new Map(defaultFiles.map((file) => [getFileKey(file), { ...file }]));
185
+
186
+ for (const projectFile of projectFiles) {
187
+ const key = getFileKey(projectFile);
188
+ const current = mergedMap.get(key) ?? getDefaultFileConfig(projectFile);
189
+ mergedMap.set(key, mergeFileConfig(current, projectFile));
190
+ }
191
+
192
+ for (const runtimeFile of runtimeFiles) {
193
+ const key = getFileKey(runtimeFile);
194
+ const current = mergedMap.get(key) ?? getDefaultFileConfig(runtimeFile);
195
+ mergedMap.set(key, mergeFileConfig(current, runtimeFile));
196
+ }
197
+
198
+ return [...mergedMap.values()].filter((file) => file.enabled !== false);
199
+ };
200
+
201
+ const logGenerated = ({ rootDir, targetPath, commonPath, localPaths, usedLocalPaths }) => {
202
+ const targetName = path.relative(rootDir, targetPath).replaceAll('\\', '/');
203
+ const commonName = `@dazhicheng/common/${path.relative(commonDir, commonPath).replaceAll('\\', '/')}`;
204
+ const sourceText = [commonName, ...usedLocalPaths].join(' + ');
205
+
206
+ console.log(`[sync:common] generated ${targetName} <- ${sourceText}`);
207
+ if (localPaths.length > 0) {
208
+ console.log(`[sync:common] 如需项目增量,请修改 ${localPaths.join(' / ')}`);
209
+ console.log(
210
+ '[sync:common] 单个 local 文件可使用 <<<before>>> / <<<common>>> / <<<after>>> 标记',
211
+ );
212
+ }
213
+ };
214
+
215
+ const renderFile = ({ file, commonContent, localSections, usedLocalPaths }) => {
216
+ let output = file.header ?? '';
217
+
218
+ if (localSections.before) {
219
+ usedLocalPaths.push(file.localFiles.merged);
220
+ output += `${file.blockHeaders.before ?? ''}${localSections.before}\n\n`;
221
+ }
222
+
223
+ output += `${file.blockHeaders.common ?? ''}${commonContent}`;
224
+
225
+ if (localSections.after) {
226
+ if (!usedLocalPaths.includes(file.localFiles.merged)) {
227
+ usedLocalPaths.push(file.localFiles.merged);
228
+ }
229
+ output += `${file.blockHeaders.after ?? ''}${localSections.after}\n`;
230
+ } else {
231
+ output += '\n';
232
+ }
233
+
234
+ return output;
235
+ };
236
+
237
+ const syncFiles = ({ files, rootDir }) => {
238
+ for (const file of files) {
239
+ const commonPath = ensureCommonFile(file);
240
+ const targetPath = path.join(rootDir, getFileRelativePath(file));
241
+ const targetDir = path.dirname(targetPath);
242
+ if (!fs.existsSync(targetDir)) {
243
+ fs.mkdirSync(targetDir, { recursive: true });
244
+ }
245
+ const mergedPath = file.localFiles?.merged ? path.join(rootDir, file.localFiles.merged) : '';
246
+ const localSections = parseLocalSections(mergedPath);
247
+ const commonContent = fs.readFileSync(commonPath, 'utf8').trimEnd();
248
+ const usedLocalPaths = [];
249
+ const output = renderFile({ file, commonContent, localSections, usedLocalPaths });
250
+
251
+ fs.writeFileSync(targetPath, output);
252
+ logGenerated({
253
+ rootDir,
254
+ targetPath,
255
+ commonPath,
256
+ localPaths: Object.values(file.localFiles ?? {}).filter(Boolean),
257
+ usedLocalPaths,
258
+ });
259
+ }
260
+ };
261
+
262
+ export const runSyncCommon = async ({ rootDir, files = [], config = {} }) => {
263
+ const runtimeConfig = {
264
+ ...(config ?? {}),
265
+ files: [...normalizeFiles(config ?? {}), ...normalizeFiles(files)],
266
+ };
267
+ const managedFiles = mergeManagedFiles({}, runtimeConfig);
268
+
269
+ syncFiles({ files: managedFiles, rootDir });
270
+ };
271
+
272
+ const isDirectRun = process.argv[1] && path.resolve(process.argv[1]) === __filename;
273
+
274
+ if (isDirectRun) {
275
+ const cwd = process.cwd();
276
+ await runSyncCommon({ rootDir: cwd });
277
+ }
package/tsconfig.json ADDED
@@ -0,0 +1,29 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "esnext",
4
+ "module": "esnext",
5
+ "moduleResolution": "node",
6
+ "strict": true,
7
+ "jsx": "preserve",
8
+ "jsxImportSource": "vue",
9
+ "sourceMap": true,
10
+ "resolveJsonModule": true,
11
+ "esModuleInterop": true,
12
+ "lib": ["esnext", "dom"],
13
+ "types": ["vite/client", "node", "element-plus/global"],
14
+ "skipLibCheck": true,
15
+ "baseUrl": ".",
16
+ "paths": {
17
+ "@/*": ["src/*"],
18
+ "@views/*": ["src/views/*"],
19
+ "@imgs/*": ["src/assets/images/*"],
20
+ "@icons/*": ["src/assets/icons/*"],
21
+ "@utils/*": ["src/utils/*"],
22
+ "@stores/*": ["src/store/*"],
23
+ "@plugins/*": ["src/plugins/*"],
24
+ "@styles/*": ["src/assets/styles/*"]
25
+ }
26
+ },
27
+ "include": ["src/**/*", "src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue"],
28
+ "exclude": ["node_modules", "dist", "**/*.js"]
29
+ }