Browse Source

feat: PHP 运行环境增加扩展源选择 (#2340)

release-1.6 v1.6.2
zhengkunwang 1 year ago committed by wanghe-fit2cloud
parent
commit
04edce94f9
  1. 2
      backend/app/dto/request/runtime.go
  2. 1
      backend/app/dto/response/runtime.go
  3. 7
      backend/app/service/runtime.go
  4. 3
      backend/app/service/runtime_utils.go
  5. 2
      frontend/src/api/interface/runtime.ts
  6. 9
      frontend/src/lang/modules/en.ts
  7. 9
      frontend/src/lang/modules/tw.ts
  8. 9
      frontend/src/lang/modules/zh.ts
  9. 155
      frontend/src/views/website/runtime/create/index.vue

2
backend/app/dto/request/runtime.go

@ -17,6 +17,7 @@ type RuntimeCreate struct {
Image string `json:"image"` Image string `json:"image"`
Type string `json:"type"` Type string `json:"type"`
Version string `json:"version"` Version string `json:"version"`
Source string `json:"source"`
} }
type RuntimeDelete struct { type RuntimeDelete struct {
@ -30,4 +31,5 @@ type RuntimeUpdate struct {
Image string `json:"image"` Image string `json:"image"`
Version string `json:"version"` Version string `json:"version"`
Rebuild bool `json:"rebuild"` Rebuild bool `json:"rebuild"`
Source string `json:"source"`
} }

1
backend/app/dto/response/runtime.go

@ -6,4 +6,5 @@ type RuntimeRes struct {
model.Runtime model.Runtime
AppParams []AppParam `json:"appParams"` AppParams []AppParam `json:"appParams"`
AppID uint `json:"appId"` AppID uint `json:"appId"`
Source string `json:"source"`
} }

7
backend/app/service/runtime.go

@ -94,7 +94,7 @@ func (r *RuntimeService) Create(create request.RuntimeCreate) (err error) {
return return
} }
} }
composeContent, envContent, forms, err := handleParams(create.Image, create.Type, newNameDir, create.Params) composeContent, envContent, forms, err := handleParams(create.Image, create.Type, newNameDir, create.Source, create.Params)
if err != nil { if err != nil {
return return
} }
@ -197,6 +197,9 @@ func (r *RuntimeService) Get(id uint) (*response.RuntimeRes, error) {
if err != nil { if err != nil {
return nil, err return nil, err
} }
if v, ok := envs["CONTAINER_PACKAGE_URL"]; ok {
res.Source = v
}
for _, form := range appForm.FormFields { for _, form := range appForm.FormFields {
if v, ok := envs[form.EnvKey]; ok { if v, ok := envs[form.EnvKey]; ok {
appParam := response.AppParam{ appParam := response.AppParam{
@ -252,7 +255,7 @@ func (r *RuntimeService) Update(req request.RuntimeUpdate) error {
return buserr.New(constant.ErrImageExist) return buserr.New(constant.ErrImageExist)
} }
runtimeDir := path.Join(constant.RuntimeDir, runtime.Type, runtime.Name) runtimeDir := path.Join(constant.RuntimeDir, runtime.Type, runtime.Name)
composeContent, envContent, _, err := handleParams(req.Image, runtime.Type, runtimeDir, req.Params) composeContent, envContent, _, err := handleParams(req.Image, runtime.Type, runtimeDir, req.Source, req.Params)
if err != nil { if err != nil {
return err return err
} }

3
backend/app/service/runtime_utils.go

@ -89,7 +89,7 @@ func buildRuntime(runtime *model.Runtime, oldImageID string, rebuild bool) {
_ = runtimeRepo.Save(runtime) _ = runtimeRepo.Save(runtime)
} }
func handleParams(image, runtimeType, runtimeDir string, params map[string]interface{}) (composeContent []byte, envContent []byte, forms []byte, err error) { func handleParams(image, runtimeType, runtimeDir, source string, params map[string]interface{}) (composeContent []byte, envContent []byte, forms []byte, err error) {
fileOp := files.NewFileOp() fileOp := files.NewFileOp()
composeContent, err = fileOp.GetContent(path.Join(runtimeDir, "docker-compose.yml")) composeContent, err = fileOp.GetContent(path.Join(runtimeDir, "docker-compose.yml"))
if err != nil { if err != nil {
@ -114,6 +114,7 @@ func handleParams(image, runtimeType, runtimeDir string, params map[string]inter
params["PHP_EXTENSIONS"] = strings.Join(strArray, ",") params["PHP_EXTENSIONS"] = strings.Join(strArray, ",")
} }
} }
params["CONTAINER_PACKAGE_URL"] = source
} }
newMap := make(map[string]string) newMap := make(map[string]string)
handleMap(params, newMap) handleMap(params, newMap)

2
frontend/src/api/interface/runtime.ts

@ -23,6 +23,7 @@ export namespace Runtime {
export interface RuntimeDTO extends Runtime { export interface RuntimeDTO extends Runtime {
appParams: App.InstallParams[]; appParams: App.InstallParams[];
appId: number; appId: number;
source?: string;
} }
export interface RuntimeCreate { export interface RuntimeCreate {
@ -36,6 +37,7 @@ export namespace Runtime {
appId?: number; appId?: number;
version?: string; version?: string;
rebuild?: boolean; rebuild?: boolean;
source?: string;
} }
export interface RuntimeUpdate { export interface RuntimeUpdate {

9
frontend/src/lang/modules/en.ts

@ -1658,6 +1658,15 @@ const message = {
phpPluginHelper: phpPluginHelper:
'<a target="_blank" href="https://1panel.cn/docs/user_manual/websites/php/#php_1">View PHP extension list</a> ', '<a target="_blank" href="https://1panel.cn/docs/user_manual/websites/php/#php_1">View PHP extension list</a> ',
rebuild: 'Rebuild PHP App', rebuild: 'Rebuild PHP App',
source: 'PHP extension source',
ustc: 'University of Science and Technology of China',
netease: 'Netease',
aliyun: 'Alibaba Cloud',
default: 'default',
tsinghua: 'Tsinghua University',
xtomhk: 'XTOM Mirror Station (Hong Kong)',
xtom: 'XTOM Mirror Station (Global)',
phpsourceHelper: 'Choose the appropriate source according to your network environment',
}, },
process: { process: {
pid: 'Process ID', pid: 'Process ID',

9
frontend/src/lang/modules/tw.ts

@ -1569,6 +1569,15 @@ const message = {
phpPluginHelper: phpPluginHelper:
"<a target=“_blank” href='https://1panel.cn/docs/user_manual/websites/php/#php_1'>查看 PHP 擴展列表</a> ", "<a target=“_blank” href='https://1panel.cn/docs/user_manual/websites/php/#php_1'>查看 PHP 擴展列表</a> ",
rebuild: '重建 PHP 應用', rebuild: '重建 PHP 應用',
source: 'PHP 擴展源',
ustc: '中國科學技術大學',
netease: '網易',
aliyun: '阿里雲',
default: '默認',
tsinghua: '清華大學',
xtomhk: 'XTOM 鏡像站香港',
xtom: 'XTOM 鏡像站全球',
phpsourceHelper: '根據你的網絡環境選擇合適的源',
}, },
process: { process: {
pid: '進程ID', pid: '進程ID',

9
frontend/src/lang/modules/zh.ts

@ -1569,6 +1569,15 @@ const message = {
phpPluginHelper: phpPluginHelper:
"<a target=“_blank” href='https://1panel.cn/docs/user_manual/websites/php/#php_1'>查看 PHP 扩展列表</a> ", "<a target=“_blank” href='https://1panel.cn/docs/user_manual/websites/php/#php_1'>查看 PHP 扩展列表</a> ",
rebuild: '重建 PHP 应用', rebuild: '重建 PHP 应用',
source: 'PHP 扩展源',
ustc: '中国科学技术大学',
netease: '网易',
aliyun: '阿里云',
default: '默认',
tsinghua: '清华大学',
xtomhk: 'XTOM 镜像站香港',
xtom: 'XTOM 镜像站全球',
phpsourceHelper: '根据你的网络环境选择合适的源',
}, },
process: { process: {
pid: '进程ID', pid: '进程ID',

155
frontend/src/views/website/runtime/create/index.vue

@ -63,44 +63,60 @@
</el-col> </el-col>
</el-row> </el-row>
</el-form-item> </el-form-item>
<el-form-item :label="$t('runtime.image')" prop="image">
<el-input v-model="runtime.image"></el-input>
</el-form-item>
<div v-if="initParam"> <div v-if="initParam">
<el-form-item v-if="runtime.type === 'php'"> <div v-if="runtime.type === 'php'">
<el-alert :title="$t('runtime.buildHelper')" type="warning" :closable="false" /> <el-form-item :label="$t('runtime.image')" prop="image">
</el-form-item> <el-input v-model="runtime.image"></el-input>
<Params </el-form-item>
v-if="mode === 'create'" <el-form-item :label="$t('runtime.source')" prop="source">
v-model:form="runtime.params" <el-select v-model="runtime.source" filterable allow-create default-first-option>
v-model:params="appParams" <el-option
v-model:rules="rules" v-for="(source, index) in phpSources"
></Params> :key="index"
<EditParams :label="source.label + ' [' + source.value + ']'"
v-if="mode === 'edit'" :value="source.value"
v-model:form="runtime.params" ></el-option>
v-model:params="editParams" </el-select>
v-model:rules="rules" <span class="input-help">
></EditParams> {{ $t('runtime.phpsourceHelper') }}
<el-form-item v-if="runtime.type === 'php'"> </span>
<el-alert type="info" :closable="false"> </el-form-item>
<span>{{ $t('runtime.extendHelper') }}</span>
<span v-html="$t('runtime.phpPluginHelper')"></span> <Params
<br /> v-if="mode === 'create'"
</el-alert> v-model:form="runtime.params"
</el-form-item> v-model:params="appParams"
<div v-if="runtime.type === 'php' && mode == 'edit'"> v-model:rules="rules"
></Params>
<EditParams
v-if="mode === 'edit'"
v-model:form="runtime.params"
v-model:params="editParams"
v-model:rules="rules"
></EditParams>
<el-form-item> <el-form-item>
<el-checkbox v-model="runtime.rebuild"> <el-alert :title="$t('runtime.buildHelper')" type="warning" :closable="false" />
{{ $t('runtime.rebuild') }}
</el-checkbox>
</el-form-item> </el-form-item>
<el-form-item> <el-form-item>
<el-alert type="info" :closable="false"> <el-alert type="info" :closable="false">
<span>{{ $t('runtime.rebuildHelper') }}</span> <span>{{ $t('runtime.extendHelper') }}</span>
<span v-html="$t('runtime.phpPluginHelper')"></span>
<br /> <br />
</el-alert> </el-alert>
</el-form-item> </el-form-item>
<div v-if="mode == 'edit'">
<el-form-item>
<el-checkbox v-model="runtime.rebuild">
{{ $t('runtime.rebuild') }}
</el-checkbox>
</el-form-item>
<el-form-item>
<el-alert type="info" :closable="false">
<span>{{ $t('runtime.rebuildHelper') }}</span>
<br />
</el-alert>
</el-form-item>
</div>
</div> </div>
</div> </div>
</div> </div>
@ -160,23 +176,59 @@ const appReq = reactive({
page: 1, page: 1,
pageSize: 20, pageSize: 20,
}); });
const runtime = ref<Runtime.RuntimeCreate>({ const initData = (type: string) => ({
name: '', name: '',
appDetailId: undefined, appDetailId: undefined,
image: '', image: '',
params: {}, params: {},
type: 'php', type: type,
resource: 'appstore', resource: 'appstore',
rebuild: false, rebuild: false,
source: 'mirrors.ustc.edu.cn',
}); });
let runtime = reactive<Runtime.RuntimeCreate>(initData('php'));
const rules = ref<any>({ const rules = ref<any>({
name: [Rules.appName], name: [Rules.appName],
resource: [Rules.requiredInput], resource: [Rules.requiredInput],
appId: [Rules.requiredSelect], appId: [Rules.requiredSelect],
version: [Rules.requiredInput, Rules.paramCommon], version: [Rules.requiredInput, Rules.paramCommon],
image: [Rules.requiredInput, Rules.imageName], image: [Rules.requiredInput, Rules.imageName],
source: [Rules.requiredSelect],
}); });
const phpSources = [
{
label: i18n.global.t('runtime.ustc'),
value: 'mirrors.ustc.edu.cn',
},
{
label: i18n.global.t('runtime.netease'),
value: 'mirrors.163.com',
},
{
label: i18n.global.t('runtime.aliyun'),
value: 'mirrors.aliyun.com',
},
{
label: i18n.global.t('runtime.tsinghua'),
value: 'mirrors.tuna.tsinghua.edu.cn',
},
{
label: i18n.global.t('runtime.xtomhk'),
value: 'mirrors.xtom.com.hk',
},
{
label: i18n.global.t('runtime.xtom'),
value: 'mirrors.xtom.com',
},
{
label: i18n.global.t('runtime.default'),
value: 'dl-cdn.alpinelinux.org',
},
];
const em = defineEmits(['close']); const em = defineEmits(['close']);
const handleClose = () => { const handleClose = () => {
@ -186,12 +238,12 @@ const handleClose = () => {
const changeResource = (resource: string) => { const changeResource = (resource: string) => {
if (resource === 'local') { if (resource === 'local') {
runtime.value.appDetailId = undefined; runtime.appDetailId = undefined;
runtime.value.version = ''; runtime.version = '';
runtime.value.params = {}; runtime.params = {};
runtime.value.image = ''; runtime.image = '';
} else { } else {
runtime.value.version = ''; runtime.version = '';
searchApp(null); searchApp(null);
} }
}; };
@ -201,7 +253,7 @@ const searchApp = (appId: number) => {
apps.value = res.data.items || []; apps.value = res.data.items || [];
if (res.data && res.data.items && res.data.items.length > 0) { if (res.data && res.data.items && res.data.items.length > 0) {
if (appId == null) { if (appId == null) {
runtime.value.appId = res.data.items[0].id; runtime.appId = res.data.items[0].id;
getApp(res.data.items[0].key, mode.value); getApp(res.data.items[0].key, mode.value);
} else { } else {
res.data.items.forEach((item) => { res.data.items.forEach((item) => {
@ -227,10 +279,10 @@ const changeApp = (appId: number) => {
const changeVersion = () => { const changeVersion = () => {
loading.value = true; loading.value = true;
initParam.value = false; initParam.value = false;
GetAppDetail(runtime.value.appId, runtime.value.version, 'runtime') GetAppDetail(runtime.appId, runtime.version, 'runtime')
.then((res) => { .then((res) => {
runtime.value.appDetailId = res.data.id; runtime.appDetailId = res.data.id;
runtime.value.image = res.data.image + ':' + runtime.value.version; runtime.image = res.data.image + ':' + runtime.version;
appParams.value = res.data.params; appParams.value = res.data.params;
initParam.value = true; initParam.value = true;
}) })
@ -243,7 +295,7 @@ const getApp = (appkey: string, mode: string) => {
GetApp(appkey).then((res) => { GetApp(appkey).then((res) => {
appVersions.value = res.data.versions || []; appVersions.value = res.data.versions || [];
if (res.data.versions.length > 0) { if (res.data.versions.length > 0) {
runtime.value.version = res.data.versions[0]; runtime.version = res.data.versions[0];
if (mode === 'create') { if (mode === 'create') {
changeVersion(); changeVersion();
} else { } else {
@ -261,7 +313,7 @@ const submit = async (formEl: FormInstance | undefined) => {
} }
if (mode.value == 'create') { if (mode.value == 'create') {
loading.value = true; loading.value = true;
CreateRuntime(runtime.value) CreateRuntime(runtime)
.then(() => { .then(() => {
MsgSuccess(i18n.global.t('commons.msg.createSuccess')); MsgSuccess(i18n.global.t('commons.msg.createSuccess'));
handleClose(); handleClose();
@ -271,7 +323,7 @@ const submit = async (formEl: FormInstance | undefined) => {
}); });
} else { } else {
loading.value = true; loading.value = true;
UpdateRuntime(runtime.value) UpdateRuntime(runtime)
.then(() => { .then(() => {
MsgSuccess(i18n.global.t('commons.msg.updateSuccess')); MsgSuccess(i18n.global.t('commons.msg.updateSuccess'));
handleClose(); handleClose();
@ -287,7 +339,7 @@ const getRuntime = async (id: number) => {
try { try {
const res = await GetRuntime(id); const res = await GetRuntime(id);
const data = res.data; const data = res.data;
runtime.value = { Object.assign(runtime, {
id: data.id, id: data.id,
name: data.name, name: data.name,
appDetailId: data.appDetailId, appDetailId: data.appDetailId,
@ -298,7 +350,8 @@ const getRuntime = async (id: number) => {
appId: data.appId, appId: data.appId,
version: data.version, version: data.version,
rebuild: true, rebuild: true,
}; source: data.source,
});
editParams.value = data.appParams; editParams.value = data.appParams;
if (mode.value == 'create') { if (mode.value == 'create') {
searchApp(data.appId); searchApp(data.appId);
@ -312,16 +365,10 @@ const acceptParams = async (props: OperateRrops) => {
mode.value = props.mode; mode.value = props.mode;
initParam.value = false; initParam.value = false;
if (props.mode === 'create') { if (props.mode === 'create') {
runtime.value = { Object.assign(runtime, initData(props.type));
name: '',
appDetailId: undefined,
image: '',
params: {},
type: props.type,
resource: 'appstore',
};
searchApp(null); searchApp(null);
} else { } else {
searchApp(null);
getRuntime(props.id); getRuntime(props.id);
} }
open.value = true; open.value = true;

Loading…
Cancel
Save