mirror of
https://github.com/router-for-me/Cli-Proxy-API-Management-Center.git
synced 2026-02-02 19:00:49 +08:00
feat: implement versioning system by extracting version from environment, git tags, or package.json, and display app version in MainLayout; enhance ConfigPage with search functionality and CodeMirror integration for YAML editing
This commit is contained in:
@@ -421,6 +421,9 @@ export function MainLayout() {
|
|||||||
<span>
|
<span>
|
||||||
{t('footer.api_version')}: {serverVersion || t('system_info.version_unknown')}
|
{t('footer.api_version')}: {serverVersion || t('system_info.version_unknown')}
|
||||||
</span>
|
</span>
|
||||||
|
<span>
|
||||||
|
{t('footer.version')}: {__APP_VERSION__ || t('system_info.version_unknown')}
|
||||||
|
</span>
|
||||||
<span>
|
<span>
|
||||||
{t('footer.build_date')}:{' '}
|
{t('footer.build_date')}:{' '}
|
||||||
{serverBuildDate ? new Date(serverBuildDate).toLocaleString(i18n.language) : t('system_info.version_unknown')}
|
{serverBuildDate ? new Date(serverBuildDate).toLocaleString(i18n.language) : t('system_info.version_unknown')}
|
||||||
|
|||||||
@@ -550,7 +550,11 @@
|
|||||||
"status_save_failed": "Save failed",
|
"status_save_failed": "Save failed",
|
||||||
"save_success": "Configuration saved successfully",
|
"save_success": "Configuration saved successfully",
|
||||||
"error_yaml_not_supported": "Server did not return YAML. Verify the /config.yaml endpoint is available.",
|
"error_yaml_not_supported": "Server did not return YAML. Verify the /config.yaml endpoint is available.",
|
||||||
"editor_placeholder": "key: value"
|
"editor_placeholder": "key: value",
|
||||||
|
"search_placeholder": "Search config... (Enter for next, Shift+Enter for previous)",
|
||||||
|
"search_no_results": "No results",
|
||||||
|
"search_prev": "Previous",
|
||||||
|
"search_next": "Next"
|
||||||
},
|
},
|
||||||
"system_info": {
|
"system_info": {
|
||||||
"title": "Management Center Info",
|
"title": "Management Center Info",
|
||||||
|
|||||||
@@ -550,7 +550,11 @@
|
|||||||
"status_save_failed": "保存失败",
|
"status_save_failed": "保存失败",
|
||||||
"save_success": "配置已保存",
|
"save_success": "配置已保存",
|
||||||
"error_yaml_not_supported": "服务器未返回 YAML 格式,请确认 /config.yaml 接口可用",
|
"error_yaml_not_supported": "服务器未返回 YAML 格式,请确认 /config.yaml 接口可用",
|
||||||
"editor_placeholder": "key: value"
|
"editor_placeholder": "key: value",
|
||||||
|
"search_placeholder": "搜索配置内容... (Enter 下一个, Shift+Enter 上一个)",
|
||||||
|
"search_no_results": "无结果",
|
||||||
|
"search_prev": "上一个",
|
||||||
|
"search_next": "下一个"
|
||||||
},
|
},
|
||||||
"system_info": {
|
"system_info": {
|
||||||
"title": "管理中心信息",
|
"title": "管理中心信息",
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
@use '../styles/mixins' as *;
|
||||||
|
|
||||||
.container {
|
.container {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
@@ -21,6 +23,54 @@
|
|||||||
gap: $spacing-lg;
|
gap: $spacing-lg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.searchBar {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: $spacing-sm;
|
||||||
|
|
||||||
|
@include mobile {
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: stretch;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchInputWrapper {
|
||||||
|
flex: 1;
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchInput {
|
||||||
|
flex: 1;
|
||||||
|
padding-right: 80px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchCount {
|
||||||
|
position: absolute;
|
||||||
|
right: 12px;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
padding: 2px 8px;
|
||||||
|
border-radius: $radius-sm;
|
||||||
|
pointer-events: none;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchActions {
|
||||||
|
display: flex;
|
||||||
|
gap: 4px;
|
||||||
|
flex-shrink: 0;
|
||||||
|
|
||||||
|
button {
|
||||||
|
min-width: 32px;
|
||||||
|
padding: 0 8px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.controls {
|
.controls {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
@@ -56,10 +106,74 @@
|
|||||||
border: 1px solid var(--border-color);
|
border: 1px solid var(--border-color);
|
||||||
border-radius: $radius-lg;
|
border-radius: $radius-lg;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
|
||||||
|
// CodeMirror theme overrides
|
||||||
|
:global {
|
||||||
|
.cm-editor {
|
||||||
|
height: 100%;
|
||||||
|
font-size: 14px;
|
||||||
|
font-family: 'Consolas', 'Monaco', 'Menlo', monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-scroller {
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-gutters {
|
||||||
|
border-right: 1px solid var(--border-color);
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-lineNumbers .cm-gutterElement {
|
||||||
|
padding: 0 8px 0 12px;
|
||||||
|
min-width: 40px;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-activeLine {
|
||||||
|
background: var(--bg-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-activeLineGutter {
|
||||||
|
background: var(--bg-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-selectionMatch {
|
||||||
|
background: rgba(255, 193, 7, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-searchMatch {
|
||||||
|
background: rgba(255, 193, 7, 0.4);
|
||||||
|
outline: 1px solid rgba(255, 193, 7, 0.6);
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-searchMatch-selected {
|
||||||
|
background: rgba(255, 152, 0, 0.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dark theme adjustments
|
||||||
|
[data-theme='dark'] & {
|
||||||
|
.cm-gutters {
|
||||||
|
background: var(--bg-tertiary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.cm-selectionMatch {
|
||||||
|
background: rgba(255, 193, 7, 0.2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.actions {
|
.actions {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: $spacing-sm;
|
gap: $spacing-sm;
|
||||||
justify-content: flex-end;
|
justify-content: flex-end;
|
||||||
|
|
||||||
|
@include mobile {
|
||||||
|
justify-content: stretch;
|
||||||
|
|
||||||
|
button {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,21 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
|
import CodeMirror, { ReactCodeMirrorRef } from '@uiw/react-codemirror';
|
||||||
|
import { yaml } from '@codemirror/lang-yaml';
|
||||||
|
import { search, searchKeymap, highlightSelectionMatches } from '@codemirror/search';
|
||||||
|
import { keymap } from '@codemirror/view';
|
||||||
import { Card } from '@/components/ui/Card';
|
import { Card } from '@/components/ui/Card';
|
||||||
import { Button } from '@/components/ui/Button';
|
import { Button } from '@/components/ui/Button';
|
||||||
import { useNotificationStore, useAuthStore } from '@/stores';
|
import { Input } from '@/components/ui/Input';
|
||||||
|
import { useNotificationStore, useAuthStore, useThemeStore } from '@/stores';
|
||||||
import { configFileApi } from '@/services/api/configFile';
|
import { configFileApi } from '@/services/api/configFile';
|
||||||
|
import styles from './ConfigPage.module.scss';
|
||||||
|
|
||||||
export function ConfigPage() {
|
export function ConfigPage() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { showNotification } = useNotificationStore();
|
const { showNotification } = useNotificationStore();
|
||||||
const connectionStatus = useAuthStore((state) => state.connectionStatus);
|
const connectionStatus = useAuthStore((state) => state.connectionStatus);
|
||||||
|
const theme = useThemeStore((state) => state.theme);
|
||||||
|
|
||||||
const [content, setContent] = useState('');
|
const [content, setContent] = useState('');
|
||||||
const [loading, setLoading] = useState(true);
|
const [loading, setLoading] = useState(true);
|
||||||
@@ -16,71 +23,264 @@ export function ConfigPage() {
|
|||||||
const [error, setError] = useState('');
|
const [error, setError] = useState('');
|
||||||
const [dirty, setDirty] = useState(false);
|
const [dirty, setDirty] = useState(false);
|
||||||
|
|
||||||
|
// Search state
|
||||||
|
const [searchQuery, setSearchQuery] = useState('');
|
||||||
|
const [searchResults, setSearchResults] = useState<{ current: number; total: number }>({ current: 0, total: 0 });
|
||||||
|
const editorRef = useRef<ReactCodeMirrorRef>(null);
|
||||||
|
|
||||||
const disableControls = connectionStatus !== 'connected';
|
const disableControls = connectionStatus !== 'connected';
|
||||||
|
|
||||||
const loadConfig = async () => {
|
const loadConfig = useCallback(async () => {
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
setError('');
|
setError('');
|
||||||
try {
|
try {
|
||||||
const data = await configFileApi.fetchConfigYaml();
|
const data = await configFileApi.fetchConfigYaml();
|
||||||
setContent(data);
|
setContent(data);
|
||||||
setDirty(false);
|
setDirty(false);
|
||||||
} catch (err: any) {
|
} catch (err: unknown) {
|
||||||
setError(err?.message || t('notification.refresh_failed'));
|
const message = err instanceof Error ? err.message : t('notification.refresh_failed');
|
||||||
|
setError(message);
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
}
|
}
|
||||||
};
|
}, [t]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
loadConfig();
|
loadConfig();
|
||||||
}, []);
|
}, [loadConfig]);
|
||||||
|
|
||||||
const handleSave = async () => {
|
const handleSave = async () => {
|
||||||
setSaving(true);
|
setSaving(true);
|
||||||
try {
|
try {
|
||||||
await configFileApi.saveConfigYaml(content);
|
await configFileApi.saveConfigYaml(content);
|
||||||
setDirty(false);
|
setDirty(false);
|
||||||
showNotification(t('notification.saved_success'), 'success');
|
showNotification(t('config_management.save_success'), 'success');
|
||||||
} catch (err: any) {
|
} catch (err: unknown) {
|
||||||
showNotification(`${t('notification.save_failed')}: ${err?.message || ''}`, 'error');
|
const message = err instanceof Error ? err.message : '';
|
||||||
|
showNotification(`${t('notification.save_failed')}: ${message}`, 'error');
|
||||||
} finally {
|
} finally {
|
||||||
setSaving(false);
|
setSaving(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleChange = useCallback((value: string) => {
|
||||||
|
setContent(value);
|
||||||
|
setDirty(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// Search functionality
|
||||||
|
const performSearch = useCallback((query: string, direction: 'next' | 'prev' = 'next') => {
|
||||||
|
if (!query || !editorRef.current?.view) return;
|
||||||
|
|
||||||
|
const view = editorRef.current.view;
|
||||||
|
const doc = view.state.doc.toString();
|
||||||
|
const matches: number[] = [];
|
||||||
|
const lowerQuery = query.toLowerCase();
|
||||||
|
const lowerDoc = doc.toLowerCase();
|
||||||
|
|
||||||
|
let pos = 0;
|
||||||
|
while (pos < lowerDoc.length) {
|
||||||
|
const index = lowerDoc.indexOf(lowerQuery, pos);
|
||||||
|
if (index === -1) break;
|
||||||
|
matches.push(index);
|
||||||
|
pos = index + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (matches.length === 0) {
|
||||||
|
setSearchResults({ current: 0, total: 0 });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find current match based on cursor position
|
||||||
|
const cursorPos = view.state.selection.main.head;
|
||||||
|
let currentIndex = 0;
|
||||||
|
|
||||||
|
if (direction === 'next') {
|
||||||
|
// Find next match after cursor
|
||||||
|
for (let i = 0; i < matches.length; i++) {
|
||||||
|
if (matches[i] > cursorPos) {
|
||||||
|
currentIndex = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// If no match after cursor, wrap to first
|
||||||
|
if (i === matches.length - 1) {
|
||||||
|
currentIndex = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Find previous match before cursor
|
||||||
|
for (let i = matches.length - 1; i >= 0; i--) {
|
||||||
|
if (matches[i] < cursorPos) {
|
||||||
|
currentIndex = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// If no match before cursor, wrap to last
|
||||||
|
if (i === 0) {
|
||||||
|
currentIndex = matches.length - 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const matchPos = matches[currentIndex];
|
||||||
|
setSearchResults({ current: currentIndex + 1, total: matches.length });
|
||||||
|
|
||||||
|
// Scroll to and select the match
|
||||||
|
view.dispatch({
|
||||||
|
selection: { anchor: matchPos, head: matchPos + query.length },
|
||||||
|
scrollIntoView: true
|
||||||
|
});
|
||||||
|
view.focus();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleSearchChange = useCallback((value: string) => {
|
||||||
|
setSearchQuery(value);
|
||||||
|
if (value) {
|
||||||
|
performSearch(value);
|
||||||
|
} else {
|
||||||
|
setSearchResults({ current: 0, total: 0 });
|
||||||
|
}
|
||||||
|
}, [performSearch]);
|
||||||
|
|
||||||
|
const handleSearchKeyDown = useCallback((e: React.KeyboardEvent) => {
|
||||||
|
if (e.key === 'Enter') {
|
||||||
|
e.preventDefault();
|
||||||
|
performSearch(searchQuery, e.shiftKey ? 'prev' : 'next');
|
||||||
|
}
|
||||||
|
}, [searchQuery, performSearch]);
|
||||||
|
|
||||||
|
const handlePrevMatch = useCallback(() => {
|
||||||
|
performSearch(searchQuery, 'prev');
|
||||||
|
}, [searchQuery, performSearch]);
|
||||||
|
|
||||||
|
const handleNextMatch = useCallback(() => {
|
||||||
|
performSearch(searchQuery, 'next');
|
||||||
|
}, [searchQuery, performSearch]);
|
||||||
|
|
||||||
|
// CodeMirror extensions
|
||||||
|
const extensions = useMemo(() => [
|
||||||
|
yaml(),
|
||||||
|
search(),
|
||||||
|
highlightSelectionMatches(),
|
||||||
|
keymap.of(searchKeymap)
|
||||||
|
], []);
|
||||||
|
|
||||||
|
// Status text
|
||||||
|
const getStatusText = () => {
|
||||||
|
if (disableControls) return t('config_management.status_disconnected');
|
||||||
|
if (loading) return t('config_management.status_loading');
|
||||||
|
if (error) return t('config_management.status_load_failed');
|
||||||
|
if (saving) return t('config_management.status_saving');
|
||||||
|
if (dirty) return t('config_management.status_dirty');
|
||||||
|
return t('config_management.status_loaded');
|
||||||
|
};
|
||||||
|
|
||||||
|
const getStatusClass = () => {
|
||||||
|
if (error) return styles.error;
|
||||||
|
if (dirty) return styles.modified;
|
||||||
|
if (!loading && !saving) return styles.saved;
|
||||||
|
return '';
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card
|
<div className={styles.container}>
|
||||||
title={t('nav.config_management')}
|
<h1 className={styles.pageTitle}>{t('config_management.title')}</h1>
|
||||||
extra={
|
<p className={styles.description}>{t('config_management.description')}</p>
|
||||||
<div style={{ display: 'flex', gap: 8 }}>
|
|
||||||
<Button variant="secondary" size="sm" onClick={loadConfig} disabled={loading}>
|
<Card>
|
||||||
{t('common.refresh')}
|
<div className={styles.content}>
|
||||||
|
{/* Search bar */}
|
||||||
|
<div className={styles.searchBar}>
|
||||||
|
<div className={styles.searchInputWrapper}>
|
||||||
|
<Input
|
||||||
|
value={searchQuery}
|
||||||
|
onChange={(e) => handleSearchChange(e.target.value)}
|
||||||
|
onKeyDown={handleSearchKeyDown}
|
||||||
|
placeholder={t('config_management.search_placeholder', { defaultValue: '搜索配置内容... (Enter 下一个, Shift+Enter 上一个)' })}
|
||||||
|
disabled={disableControls || loading}
|
||||||
|
className={styles.searchInput}
|
||||||
|
/>
|
||||||
|
{searchQuery && (
|
||||||
|
<span className={styles.searchCount}>
|
||||||
|
{searchResults.total > 0
|
||||||
|
? `${searchResults.current} / ${searchResults.total}`
|
||||||
|
: t('config_management.search_no_results', { defaultValue: '无结果' })}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className={styles.searchActions}>
|
||||||
|
<Button
|
||||||
|
variant="secondary"
|
||||||
|
size="sm"
|
||||||
|
onClick={handlePrevMatch}
|
||||||
|
disabled={!searchQuery || searchResults.total === 0}
|
||||||
|
title={t('config_management.search_prev', { defaultValue: '上一个' })}
|
||||||
|
>
|
||||||
|
↑
|
||||||
</Button>
|
</Button>
|
||||||
<Button size="sm" onClick={handleSave} loading={saving} disabled={disableControls || loading || !dirty}>
|
<Button
|
||||||
{t('common.save')}
|
variant="secondary"
|
||||||
|
size="sm"
|
||||||
|
onClick={handleNextMatch}
|
||||||
|
disabled={!searchQuery || searchResults.total === 0}
|
||||||
|
title={t('config_management.search_next', { defaultValue: '下一个' })}
|
||||||
|
>
|
||||||
|
↓
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
}
|
</div>
|
||||||
>
|
|
||||||
|
{/* Editor */}
|
||||||
{error && <div className="error-box">{error}</div>}
|
{error && <div className="error-box">{error}</div>}
|
||||||
<div className="form-group">
|
<div className={styles.editorWrapper}>
|
||||||
<label>{t('nav.config_management')}</label>
|
<CodeMirror
|
||||||
<textarea
|
ref={editorRef}
|
||||||
className="input"
|
|
||||||
rows={20}
|
|
||||||
value={content}
|
value={content}
|
||||||
onChange={(e) => {
|
onChange={handleChange}
|
||||||
setContent(e.target.value);
|
extensions={extensions}
|
||||||
setDirty(true);
|
theme={theme === 'dark' ? 'dark' : 'light'}
|
||||||
|
editable={!disableControls && !loading}
|
||||||
|
placeholder={t('config_management.editor_placeholder')}
|
||||||
|
height="100%"
|
||||||
|
style={{ height: '100%' }}
|
||||||
|
basicSetup={{
|
||||||
|
lineNumbers: true,
|
||||||
|
highlightActiveLineGutter: true,
|
||||||
|
highlightActiveLine: true,
|
||||||
|
foldGutter: true,
|
||||||
|
dropCursor: true,
|
||||||
|
allowMultipleSelections: true,
|
||||||
|
indentOnInput: true,
|
||||||
|
bracketMatching: true,
|
||||||
|
closeBrackets: true,
|
||||||
|
autocompletion: false,
|
||||||
|
rectangularSelection: true,
|
||||||
|
crosshairCursor: false,
|
||||||
|
highlightSelectionMatches: true,
|
||||||
|
closeBracketsKeymap: true,
|
||||||
|
searchKeymap: true,
|
||||||
|
foldKeymap: true,
|
||||||
|
completionKeymap: false,
|
||||||
|
lintKeymap: true
|
||||||
}}
|
}}
|
||||||
disabled={disableControls}
|
|
||||||
placeholder="config.yaml"
|
|
||||||
/>
|
/>
|
||||||
<div className="hint">
|
</div>
|
||||||
{dirty ? t('system_info.version_current_missing') : loading ? t('common.loading') : t('system_info.version_is_latest')}
|
|
||||||
|
{/* Controls */}
|
||||||
|
<div className={styles.controls}>
|
||||||
|
<span className={`${styles.status} ${getStatusClass()}`}>
|
||||||
|
{getStatusText()}
|
||||||
|
</span>
|
||||||
|
<div className={styles.actions}>
|
||||||
|
<Button variant="secondary" size="sm" onClick={loadConfig} disabled={loading}>
|
||||||
|
{t('config_management.reload')}
|
||||||
|
</Button>
|
||||||
|
<Button size="sm" onClick={handleSave} loading={saving} disabled={disableControls || loading || !dirty}>
|
||||||
|
{t('config_management.save')}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Card>
|
</Card>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
3
src/types/style.d.ts
vendored
3
src/types/style.d.ts
vendored
@@ -2,3 +2,6 @@ declare module '*.module.scss' {
|
|||||||
const classes: Record<string, string>;
|
const classes: Record<string, string>;
|
||||||
export default classes;
|
export default classes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Global constants injected by Vite at build time
|
||||||
|
declare const __APP_VERSION__: string;
|
||||||
|
|||||||
@@ -2,6 +2,38 @@ import { defineConfig } from 'vite';
|
|||||||
import react from '@vitejs/plugin-react';
|
import react from '@vitejs/plugin-react';
|
||||||
import { viteSingleFile } from 'vite-plugin-singlefile';
|
import { viteSingleFile } from 'vite-plugin-singlefile';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
|
import { execSync } from 'child_process';
|
||||||
|
import fs from 'fs';
|
||||||
|
|
||||||
|
// Get version from environment, git tag, or package.json
|
||||||
|
function getVersion(): string {
|
||||||
|
// 1. Environment variable (set by GitHub Actions)
|
||||||
|
if (process.env.VERSION) {
|
||||||
|
return process.env.VERSION;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. Try git tag
|
||||||
|
try {
|
||||||
|
const gitTag = execSync('git describe --tags --exact-match 2>/dev/null || git describe --tags 2>/dev/null || echo ""', { encoding: 'utf8' }).trim();
|
||||||
|
if (gitTag) {
|
||||||
|
return gitTag;
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// Git not available or no tags
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. Fall back to package.json version
|
||||||
|
try {
|
||||||
|
const pkg = JSON.parse(fs.readFileSync(path.resolve(__dirname, 'package.json'), 'utf8'));
|
||||||
|
if (pkg.version && pkg.version !== '0.0.0') {
|
||||||
|
return pkg.version;
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// package.json not readable
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'dev';
|
||||||
|
}
|
||||||
|
|
||||||
// https://vitejs.dev/config/
|
// https://vitejs.dev/config/
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
@@ -11,6 +43,9 @@ export default defineConfig({
|
|||||||
removeViteModuleLoader: true
|
removeViteModuleLoader: true
|
||||||
})
|
})
|
||||||
],
|
],
|
||||||
|
define: {
|
||||||
|
__APP_VERSION__: JSON.stringify(getVersion())
|
||||||
|
},
|
||||||
resolve: {
|
resolve: {
|
||||||
alias: {
|
alias: {
|
||||||
'@': path.resolve(__dirname, './src')
|
'@': path.resolve(__dirname, './src')
|
||||||
|
|||||||
Reference in New Issue
Block a user