Compare commits

..

70 Commits

Author SHA1 Message Date
Bryan
19b1dc0dbc Merge pull request #4367 from jumpserver/dev
merge: from dev to master
2024-09-19 19:36:42 +08:00
Bryan
77ef172a23 Merge pull request #4320 from jumpserver/dev
v4.1.0
2024-08-15 21:42:42 +08:00
Bryan
4596887bf1 Merge pull request #4178 from jumpserver/dev
v4.0.0
2024-07-03 19:06:07 +08:00
Bryan
0a3dc30c85 Merge pull request #4072 from jumpserver/dev
v3.10.11-lts
2024-06-19 16:04:12 +08:00
Bryan
51d24bc8e5 Merge pull request #3941 from jumpserver/dev
v3.10.10-lts
2024-05-16 16:05:04 +08:00
Bryan
1b15a4d043 Merge pull request #3871 from jumpserver/dev
v3.10.9 (dev to master)
2024-04-22 19:44:33 +08:00
Bryan
7d3f818242 Merge pull request #3864 from jumpserver/v3.10
v3.10.8
2024-04-18 17:58:05 +08:00
Bryan
4e26f18d77 Merge pull request #3862 from jumpserver/dev
v3.10.8
2024-04-18 17:17:36 +08:00
Bryan
b22613617a Revert "build(deps): bump follow-redirects from 1.15.3 to 1.15.4"
This reverts commit e971cbf4a8.
2024-03-27 16:16:07 +08:00
dependabot[bot]
e971cbf4a8 build(deps): bump follow-redirects from 1.15.3 to 1.15.4
Bumps [follow-redirects](https://github.com/follow-redirects/follow-redirects) from 1.15.3 to 1.15.4.
- [Release notes](https://github.com/follow-redirects/follow-redirects/releases)
- [Commits](https://github.com/follow-redirects/follow-redirects/compare/v1.15.3...v1.15.4)

---
updated-dependencies:
- dependency-name: follow-redirects
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-03-27 16:11:26 +08:00
wangruidong
4672abae35 fix: 刷新页面根据搜索条件过滤出对应的资源 2024-03-04 19:17:05 +08:00
Bryan
ba36d72602 Merge pull request #3761 from jumpserver/master
v3.10.4 (branch-v3.10)
2024-02-29 16:26:13 +08:00
Bryan
4bfbbba4c5 Merge pull request #3760 from jumpserver/dev
v3.10.4
2024-02-29 16:15:33 +08:00
Bryan
ea038ce43a Merge pull request #3697 from jumpserver/master
v3.10.2
2024-01-17 13:34:12 +00:00
Bryan
e16b19666c Merge pull request #3696 from jumpserver/dev
v3.10.2
2024-01-17 13:33:22 +00:00
Bryan
c7f5409eb6 Merge pull request #3694 from jumpserver/master
v3.10.2
2024-01-17 07:35:33 -04:00
Bryan
fdbd7d2222 Merge pull request #3693 from jumpserver/dev
v3.10.2
2024-01-17 07:24:50 -04:00
Bryan
ddbaeeafea Merge pull request #3668 from jumpserver/master
v3.10.1
2023-12-29 11:34:04 +05:00
Bryan
efb0e9dacb Merge pull request #3665 from jumpserver/dev
v3.10.1
2023-12-29 11:14:54 +05:00
huailei
f6f8301ad5 Revert "perf: 账号收集翻译"
This reverts commit 9a63ae63d4.
2023-12-22 15:25:31 +08:00
“huailei000”
9a63ae63d4 perf: 账号收集翻译 2023-12-22 11:31:45 +08:00
Bryan
1e007ccda3 Merge pull request #3642 from jumpserver/dev
v3.10
2023-12-21 15:15:52 +05:00
老广
d1d0b06b53 Merge pull request #3546 from jumpserver/dev
v3.9.0
2023-11-16 18:25:10 +08:00
Bryan
5fb70d2f24 Merge pull request #3450 from jumpserver/dev
v3.8.0
2023-10-19 03:33:53 -05:00
Bryan
b54a95430f Merge pull request #3404 from jumpserver/dev
v3.7.0
2023-09-21 17:04:42 +08:00
Bryan
4d8b4c45af Merge pull request #3355 from jumpserver/dev
v3.6.0
2023-08-17 14:00:33 +05:00
Bryan
a6d642df60 Merge pull request #3283 from jumpserver/dev
v3.5.0
2023-07-20 19:04:29 +08:00
Jiangjie.Bai
2e74f1522f Merge pull request #3222 from jumpserver/dev
v3.4.0
2023-06-15 14:51:36 +08:00
Jiangjie.Bai
fe615e0314 Merge pull request #3219 from jumpserver/dev
v3.4.0
2023-06-15 14:17:46 +08:00
Jiangjie.Bai
09f734e6fc Merge pull request #3135 from jumpserver/dev
v3.3.0
2023-05-18 19:18:11 +08:00
Jiangjie.Bai
3117046342 Merge pull request #3061 from jumpserver/dev
v3.2.0
2023-04-20 18:40:08 +08:00
Bai
b68aecb5cc fix: 批量更新资产平台help-text 2023-04-20 18:39:22 +08:00
Jiangjie.Bai
1c9b155d97 Merge pull request #3057 from jumpserver/dev
v3.2.0
2023-04-20 18:22:46 +08:00
Jiangjie.Bai
75b1be9864 Merge pull request #3019 from jumpserver/dev
v3.2.0 rc2
2023-04-14 19:01:37 +08:00
Jiangjie.Bai
615c3c1cf4 Merge pull request #3014 from jumpserver/dev
v3.2.0 rc1
2023-04-13 20:02:38 +08:00
Jiangjie.Bai
4d82231af4 Merge pull request #3012 from jumpserver/dev
v3.2.0 rc1
2023-04-13 19:22:38 +08:00
“huailei000”
c6cf6571b6 perf: ldap导入用户列表-组织下拉框设置最大宽度 2023-03-16 16:44:36 +08:00
Bai
8ea990d070 fix: 修复创建资产添加账号模版报错问题 2023-03-16 16:44:36 +08:00
“huailei000”
f4a32170d5 perf: message 2023-03-16 16:44:36 +08:00
ibuler
073508675e perf: 添加默认的信息 2023-03-16 16:44:36 +08:00
Jiangjie.Bai
1d6ca0a93a Merge pull request #2924 from jumpserver/dev
v3.1.0 rc4
2023-03-15 19:46:31 +08:00
Jiangjie.Bai
36aea652d6 Merge pull request #2788 from jumpserver/dev
v3.0.0
2023-02-23 20:16:41 +08:00
Jiangjie.Bai
1a42ce90ab Merge pull request #2760 from jumpserver/dev
v3.0.0-rc-latest
2023-02-22 22:21:54 +08:00
Jiangjie.Bai
31a401b55d Merge pull request #2463 from jumpserver/dev
v3.0.0-rc4
2023-01-31 18:55:34 +08:00
Jiangjie.Bai
582a84178d Merge pull request #2187 from jumpserver/dev
v2.28.0
2022-11-17 17:44:19 +08:00
Jiangjie.Bai
9b9f7c936c Merge pull request #2184 from jumpserver/dev
v2.28.0-rc5
2022-11-17 14:18:15 +08:00
Jiangjie.Bai
2a6100957f Merge pull request #2182 from jumpserver/dev
v2.28.0-rc4
2022-11-16 21:08:55 +08:00
Jiangjie.Bai
16606d6a27 Merge pull request #2176 from jumpserver/dev
v2.28.0-rc2
2022-11-14 10:01:05 +08:00
Jiangjie.Bai
0a612f50e6 Merge pull request #2164 from jumpserver/dev
v2.28.0-rc1
2022-11-10 17:45:47 +08:00
Jiangjie.Bai
fe36fa9390 Merge pull request #2117 from jumpserver/dev
v2.27.0-rc4
2022-10-18 21:02:10 +08:00
Jiangjie.Bai
ba109900ec Merge pull request #2113 from jumpserver/dev
v2.27.0-rc3
2022-10-18 11:20:57 +08:00
Jiangjie.Bai
ec7768267f Merge pull request #2105 from jumpserver/dev
v2.27.0-rc2
2022-10-14 11:01:32 +08:00
Jiangjie.Bai
cc58b374ab Merge pull request #2101 from jumpserver/dev
v2.27.0-rc1
2022-10-13 17:44:53 +08:00
Jiangjie.Bai
04ffbb8fd6 Merge pull request #2097 from jumpserver/dev
v2.27.0-rc1
2022-10-13 15:14:40 +08:00
Jiangjie.Bai
49880f6739 Merge pull request #2059 from jumpserver/dev
v2.26.0
2022-09-15 17:49:44 +08:00
Jiangjie.Bai
e6f98d58c4 Merge pull request #2057 from jumpserver/dev
v2.26.0-rc4
2022-09-15 16:18:03 +08:00
Jiangjie.Bai
fd1f16d43c Merge pull request #2050 from jumpserver/dev
v2.26.0-rc2
2022-09-13 17:41:39 +08:00
Jiangjie.Bai
968b2415b1 Merge pull request #2043 from jumpserver/dev
v2.26.0-rc1
2022-09-08 15:46:44 +08:00
Jiangjie.Bai
776090d6ba Merge pull request #2001 from jumpserver/dev
v2.25.0
2022-08-18 16:12:45 +08:00
Jiangjie.Bai
3a37952288 Merge pull request #1996 from jumpserver/dev
v2.25.0-rc4
2022-08-17 16:53:23 +08:00
Jiangjie.Bai
62b8fc0e3b Merge pull request #1994 from jumpserver/dev
v2.25.0-rc3
2022-08-16 19:08:23 +08:00
Jiangjie.Bai
b2028869cb Merge pull request #1986 from jumpserver/dev
v2.25.0-rc2
2022-08-12 18:06:56 +08:00
Jiangjie.Bai
5277a725f8 Merge pull request #1973 from jumpserver/dev
v2.25.0-rc1
2022-08-11 14:11:59 +08:00
Jiangjie.Bai
f137788c1a Merge pull request #1957 from jumpserver/dev
v2.24.0-rc5
2022-07-20 19:06:03 +08:00
Jiangjie.Bai
f7d17c8de7 Merge pull request #1954 from jumpserver/dev
v2.24.0-rc4
2022-07-19 16:18:13 +08:00
Jiangjie.Bai
feea70b0be Merge pull request #1944 from jumpserver/dev
v2.24.0-rc3
2022-07-18 12:05:42 +08:00
Jiangjie.Bai
04696ef3d6 Merge pull request #1940 from jumpserver/dev
v2.24.0-rc2
2022-07-15 18:07:37 +08:00
Jiangjie.Bai
1731f4f788 Merge pull request #1934 from jumpserver/dev
v2.24.0-rc1
2022-07-14 18:27:51 +08:00
Jiangjie.Bai
6f25d93909 Merge pull request #1931 from jumpserver/dev
v2.24.0-rc1
2022-07-14 17:51:58 +08:00
Jiangjie.Bai
46461ec324 Merge pull request #1925 from jumpserver/dev
v2.24.0-rc1
2022-07-14 15:12:15 +08:00
108 changed files with 562 additions and 2142 deletions

View File

@@ -1,28 +0,0 @@
name: LLM Code Review
permissions:
contents: read
pull-requests: write
on:
pull_request:
types: [opened, reopened, synchronize]
jobs:
llm-code-review:
runs-on: ubuntu-latest
steps:
- uses: fit2cloud/LLM-CodeReview-Action@main
env:
GITHUB_TOKEN: ${{ secrets.FIT2CLOUDRD_LLM_CODE_REVIEW_TOKEN }}
OPENAI_API_KEY: ${{ secrets.ALIYUN_LLM_API_KEY }}
LANGUAGE: English
OPENAI_API_ENDPOINT: https://dashscope.aliyuncs.com/compatible-mode/v1
MODEL: qwen2-1.5b-instruct
PROMPT: "Please check the following code differences for any irregularities, potential issues, or optimization suggestions, and provide your answers in English."
top_p: 1
temperature: 1
# max_tokens: 10000
MAX_PATCH_LENGTH: 10000
IGNORE_PATTERNS: "/node_modules,*.md,/dist,/.github"
FILE_PATTERNS: "*.java,*.go,*.py,*.vue,*.ts,*.js,*.css,*.scss,*.html"

View File

@@ -66,7 +66,7 @@
"normalize.css": "7.0.0",
"npm": "^7.8.0",
"nprogress": "0.2.0",
"path-to-regexp": "3.3.0",
"path-to-regexp": "2.4.0",
"v-sanitize": "^0.0.13",
"vue": "2.6.10",
"vue-codemirror": "4.0.6",
@@ -117,7 +117,7 @@
"sass-loader": "^7.1.0",
"script-ext-html-webpack-plugin": "2.1.3",
"script-loader": "0.7.2",
"serve-static": "^1.16.0",
"serve-static": "^1.13.2",
"strip-ansi": "^7.1.0",
"svg-sprite-loader": "4.1.3",
"svgo": "1.2.2",

View File

@@ -53,7 +53,7 @@ export function createJob(form) {
})
}
export function stopJob(form) {
export function StopJob(form) {
return request({
url: '/api/v1/ops/job-executions/stop/',
method: 'post',
@@ -71,10 +71,3 @@ export function JobUploadFile(form) {
})
}
export function auditUpdateJob(id, form) {
return request({
url: `/api/v1/audits/jobs/${id}/`,
method: 'patch',
data: form
})
}

View File

@@ -15,7 +15,6 @@ export function testEmailSetting(data) {
data: data
})
}
export function importLicense(formData) {
return request({
url: '/api/v1/xpack/license/import',
@@ -26,7 +25,6 @@ export function importLicense(formData) {
data: formData
})
}
export function testLdapSetting(data, refresh = true) {
let url = '/api/v1/settings/ldap/testing/config/'
if (refresh) {
@@ -98,17 +96,9 @@ export function getPublicSettings(isOpen) {
method: 'get'
})
}
export function getLogo() {
return request({
url: '/api/v1/xpack/interface/setting/',
method: 'get'
})
}
export function getPreference() {
return request({
url: '/api/v1/users/preference/?category=luna',
method: 'get'
})
}

View File

@@ -32,10 +32,6 @@ export default {
type: String,
default: '/api/v1/assets/assets/'
},
typeUrl: {
type: String,
default: '/api/v1/assets/nodes/category/tree/'
},
nodeUrl: {
type: String,
default: '/api/v1/assets/nodes/'
@@ -109,9 +105,9 @@ export default {
showAssets: false,
showSearch: false,
customTreeHeaderName: this.$t('TypeTree'),
url: this.typeUrl,
url: '/api/v1/assets/nodes/category/tree/',
nodeUrl: this.treeSetting?.nodeUrl || this.nodeUrl,
treeUrl: `${this.typeUrl}?assets=${showAssets ? '1' : '0'}&count_resource=${this.treeSetting.countResource || 'asset'}`,
treeUrl: `/api/v1/assets/nodes/category/tree/?assets=${showAssets ? '1' : '0'}&count_resource=${this.treeSetting.countResource || 'asset'}`,
callback: {
onSelected: (event, treeNode) => this.getAssetsUrl(treeNode)
}

View File

@@ -8,7 +8,7 @@
:visible.sync="visible"
class="dialog-content"
v-bind="$attrs"
width="740px"
width="600px"
@confirm="visible = false"
v-on="$listeners"
>
@@ -52,21 +52,11 @@
<el-row :gutter="24" style="margin: 0 auto;">
<el-col :md="24" :sm="24" style="display: flex; align-items: center; margin-bottom: 20px;">
<el-input
v-if="subTypeSelected !== 'face'"
v-model="secretValue"
:placeholder="inputPlaceholder"
:show-password="showPassword"
@keyup.enter.native="handleConfirm"
/>
<iframe
v-if="isFaceCaptureVisible && subTypeSelected ==='face' && faceCaptureUrl"
:src="faceCaptureUrl"
allow="camera"
sandbox="allow-scripts allow-same-origin"
style="width: 100%; height: 800px;border: none;"
/>
<span v-if="subTypeSelected === 'sms'" style="margin: -1px 0 0 20px;">
<el-button
:disabled="smsBtnDisabled"
@@ -82,24 +72,9 @@
</el-row>
<el-row :gutter="24" style="margin: 10px auto;">
<el-col :md="24" :sm="24">
<el-button
v-if="subTypeSelected!=='face'"
class="confirm-btn"
size="mini"
type="primary"
@click="handleConfirm"
>
<el-button class="confirm-btn" size="mini" type="primary" @click="handleConfirm">
{{ this.$t('Confirm') }}
</el-button>
<el-button
v-if="subTypeSelected==='face'&&!isFaceCaptureVisible"
class="confirm-btn"
size="mini"
type="primary"
@click="handleFaceCapture"
>
开始人脸识别
</el-button>
</el-col>
</el-row>
</div>
@@ -138,10 +113,7 @@ export default {
visible: false,
callback: null,
cancel: null,
processing: false,
isFaceCaptureVisible: false,
faceToken: null,
faceCaptureUrl: null
processing: false
}
},
computed: {
@@ -157,10 +129,6 @@ export default {
},
methods: {
handleSubTypeChange(val) {
if (val !== 'face') {
this.isFaceCaptureVisible = false
}
this.inputPlaceholder = this.subTypeChoices.filter(item => item.name === val)[0]?.placeholder
this.smsWidth = val === 'sms' ? 6 : 0
},
@@ -224,29 +192,6 @@ export default {
this.$message.error(this.$tc('FailedToSendVerificationCode'))
})
},
startFaceCapture() {
const url = '/api/v1/authentication/mfa/face/context/'
this.$axios.post(url).then(data => {
const token = data['token']
this.faceCaptureUrl = '/facelive/capture?token=' + token
this.isFaceCaptureVisible = true
const timer = setInterval(() => {
this.$axios.get(url + `?token=${token}`).then(data => {
if (data['is_finished']) {
clearInterval(timer)
this.isFaceCaptureVisible = false
this.handleConfirm()
}
})
}, 1000)
}).catch(() => {
this.$message.error(this.$tc('FailedToStartFaceCapture'))
})
},
handleFaceCapture() {
this.startFaceCapture()
},
handleConfirm() {
if (this.confirmTypeRequired === 'relogin') {
return this.logout()
@@ -269,8 +214,6 @@ export default {
})
}).catch((err) => {
this.$message.error(err.message || this.$tc('ConfirmFailed'))
this.faceCaptureUrl = null
this.isFaceCaptureVisible = false
})
}
}
@@ -278,21 +221,21 @@ export default {
</script>
<style lang="scss" scoped>
.dialog-content ::v-deep .el-dialog__footer {
padding: 0;
}
.dialog-content ::v-deep .el-dialog {
padding: 8px;
.el-dialog__body {
padding-top: 30px;
padding-bottom: 30px;
.dialog-content ::v-deep .el-dialog__footer {
padding: 0;
}
}
.confirm-btn {
width: 100%;
line-height: 20px;
}
.dialog-content ::v-deep .el-dialog {
padding: 8px;
.el-dialog__body {
padding-top: 30px;
padding-bottom: 30px;
}
}
.confirm-btn {
width: 100%;
line-height: 20px;
}
</style>

View File

@@ -1,111 +0,0 @@
<template>
<AutoDataForm
ref="AutoDataForm"
class="variable-add"
:submit-btn-text="submitBtnText"
v-bind="$data"
@submit="confirm"
/>
</template>
<script>
import AutoDataForm from '@/components/Form/AutoDataForm/index.vue'
export default {
name: 'VariableCreateForm',
components: {
AutoDataForm
},
props: {
variable: {
type: Object,
default: () => ({})
}
},
data() {
return {
submitBtnText: this.$t('Confirm'),
url: '/api/v1/ops/variables/',
form: Object.assign({ 'on_invalid': 'error' }, this.variable || {}),
fields: [
['', ['name', 'var_name', 'type', 'text_default_value', 'select_default_value', 'extra_args', 'tips', 'required']]
],
fieldsMeta: {
text_default_value: {
label: this.$t('DefaultValue'),
hidden: (formValue) => {
return formValue.type !== 'text'
},
el: {
type: 'input'
}
},
select_default_value: {
label: this.$t('DefaultValue'),
hidden: (formValue) => {
return formValue.type !== 'select'
},
el: { type: 'input' }
},
extra_args: {
hidden: (formValue) => {
return formValue.type !== 'select'
},
el: { type: 'textarea', rows: 4, placeholder: this.$t('ExtraArgsPlaceholder') },
rules: [
{
validator: (rule, value, callback) => {
try {
const lines = value.split('\n')
const regex = /^[^:]+:[^:]+$/
for (const line of lines) {
if (!regex.test(line.trim())) {
callback(new Error(this.$t('ExtraArgsFormatError')))
}
}
} catch (e) {
callback(new Error(this.$t('ExtraArgsFormatError')))
}
callback()
}
}
]
}
},
hasSaveContinue: false,
method: 'get'
}
},
methods: {
confirm(form) {
if (this.variable?.name) {
this.$emit('edit', form)
} else {
this.$emit('add', form)
}
}
}
}
</script>
<style lang='scss' scoped>
.variable-add {
::v-deep .el-form-item {
margin-bottom: 5px;
.help-block {
margin-bottom: 5px;
}
}
::v-deep .form-group-header {
.hr-line-dashed {
margin: 5px 0;
}
h3 {
margin-bottom: 10px;
}
}
}
</style>

View File

@@ -1,74 +0,0 @@
<template>
<AutoDataForm
ref="AutoDataForm"
class="variable-set"
:submit-btn-text="submitBtnText"
v-bind="$data"
:fields="fields"
@submit="confirm"
/>
</template>
<script>
import AutoDataForm from '@/components/Form/AutoDataForm/index.vue'
export default {
name: 'VariableSetForm',
components: {
AutoDataForm
},
props: {
formData: {
type: Array,
default: () => ([])
},
queryParam: {
type: String,
default: ''
}
},
data() {
return {
submitBtnText: this.$t('Confirm'),
// 防止缓存form remoteMeta
url: `/api/v1/ops/variables/form-data/?t=${new Date().getTime()}&` + this.queryParam,
form: {},
hasSaveContinue: false,
method: 'get',
hasReset: false
}
},
computed: {
fields() {
return [['', this.formData.map(item => item.var_name)]]
}
},
methods: {
confirm(form) {
this.$emit('confirm', form)
}
}
}
</script>
<style lang='scss' scoped>
.variable-set {
::v-deep .el-form-item {
margin-bottom: 5px;
.help-block {
margin-bottom: 5px;
}
}
::v-deep .form-group-header {
.hr-line-dashed {
margin: 5px 0;
}
h3 {
margin-bottom: 10px;
}
}
}
</style>

View File

@@ -78,7 +78,7 @@ export default {
formatterData = data
}
return (
<span style={{ whiteSpace: 'pre-wrap', lineHeight: '1.2' }}>{formatterData}</span>
<span>{formatterData}</span>
)
}
if (this.value instanceof Array) {
@@ -92,7 +92,7 @@ export default {
)
}
return (
<span style='white-space: pre-wrap;' title={this.displayValue}>{this.displayValue}</span>
<span title={this.displayValue}>{this.displayValue}</span>
)
}
}

View File

@@ -123,7 +123,6 @@ export default {
&__body {
padding: 20px 30px;
font-size: 13px;
&:has(.el-table) {
background: #f3f3f4;

View File

@@ -7,7 +7,7 @@
v-bind="$attrs"
v-on="$listeners"
>
<div
<span
v-for="(group, i) in groups"
:key="'group-'+group.name"
:slot="'id:'+group.name"
@@ -18,7 +18,7 @@
:index="i"
:line="i !== 0 && !groupHidden(groups[i - 1], i - 1)"
/>
</div>
</span>
</DataForm>
</template>

View File

@@ -7,6 +7,7 @@ import BasicTree from '@/components/Form/FormFields/BasicTree.vue'
import JsonEditor from '@/components/Form/FormFields/JsonEditor.vue'
import { assignIfNot, toSentenceCase } from '@/utils/common'
import TagInput from '@/components/Form/FormFields/TagInput.vue'
import TransferSelect from '@/components/Form/FormFields/TransferSelect.vue'
import i18n from '@/i18n/i18n'
export class FormFieldGenerator {
@@ -134,6 +135,9 @@ export class FormFieldGenerator {
case 'comment':
field.el.type = 'textarea'
break
case 'users':
field.component = TransferSelect
field.el.label = field.label
}
return field
}

View File

@@ -17,7 +17,7 @@
placement="right"
popper-class="help-tips"
>
<div slot="content" v-sanitize="data.helpTip" class="help-tip-content" /> <!-- Noncompliant -->
<div slot="content" v-sanitize="data.helpTip" /> <!-- Noncompliant -->
<i class="fa fa-question-circle-o help-tip-icon" />
</el-tooltip>
</template>
@@ -322,9 +322,4 @@ export default {
cursor: pointer;
}
}
.help-tip-content {
white-space: pre-wrap;
word-wrap: break-word;
}
</style>

View File

@@ -245,7 +245,7 @@ export default {
},
set(val) {
this.$emit('update:value', val)
this.$emit('input', val)
this.$emit('change', val)
}
},
iOptions() {

View File

@@ -1,5 +1,5 @@
<template>
<el-input v-bind="$attrs" v-on="$listeners">
<el-input v-model="iValue" v-bind="$attrs" v-on="$listeners">
<template slot="append">{{ iUnit }}</template>
</el-input>
</template>
@@ -30,6 +30,9 @@ export default {
computed: {
iUnit() {
return this.displayMapper[this.unit] || this.unit
},
iValue() {
return this.$attrs.value ? this.$attrs.value : this.defaultValue
}
}
}

View File

@@ -48,8 +48,7 @@ export default {
type: 'input-number',
el: {
min: 8,
max: 36,
size: 'mini'
max: 30
}
},
{

View File

@@ -1,5 +1,5 @@
<template>
<div :class="showSetting ? 'show-setting' : 'hide-setting'">
<div v-if="!loading" :class="showSetting ? 'show-setting' : 'hide-setting'">
<div v-for="(item, index) in items" :key="item.name" class="protocol-item">
<el-input
v-model="item.port"
@@ -114,7 +114,8 @@ export default {
name: '',
items: [],
currentProtocol: {},
showDialog: false
showDialog: false,
loading: false
}
},
computed: {
@@ -143,8 +144,10 @@ export default {
watch: {
choices: {
handler(value, oldValue) {
this.loading = true
setTimeout(() => {
this.setDefaultItems(value)
this.loading = false
},)
},
deep: true,

View File

@@ -42,7 +42,7 @@ export default {
patterns.push([/\d/, i18n.t('NUMBER_REQUIRED')])
}
if (passwordRule['SECURITY_PASSWORD_SPECIAL_CHAR']) {
const pattern = new RegExp("[`~!@#$^&*()=|{}':;',\\[\\].<>/?~@#¥……&*()——|{}【】‘;:”“'。,、?_+-]")
const pattern = new RegExp("[`~!@#$^&*()=|{}':;',\\[\\].<>/?~@#¥……&*()——|{}【】‘;:”“'。,、?]")
patterns.push([pattern, i18n.t('SPECIAL_CHAR_REQUIRED')])
}
for (const [pattern, msg] of patterns) {

View File

@@ -39,22 +39,6 @@ export default {
</script>
<style scoped>
html:lang(en) .quick-actions ::v-deep button {
width: 80px;
}
html:lang(ja) .quick-actions ::v-deep button {
width: 100px;
}
html:lang(zh-tw) .quick-actions ::v-deep button {
width: 65px;
}
html:lang(zh-cn) .quick-actions ::v-deep button {
width: 65px;
}
.quick-actions ::v-deep table {
width: 100%;
}
@@ -74,6 +58,7 @@ export default {
.quick-actions ::v-deep button {
padding: 4px 5px;
font-size: 13px;
width: 65px;
span {
overflow: hidden;

View File

@@ -3,7 +3,7 @@
v-if="showColumnSettingPopover"
:cancel-title="$tc('RestoreDefault')"
:destroy-on-close="true"
:title="$tc('ListPreference')"
:title="$tc('TableSetting')"
:visible.sync="showColumnSettingPopover"
top="10%"
width="50%"

View File

@@ -14,7 +14,7 @@
@select="selectStrategy.onSelect"
v-on="$listeners"
@selection-change="selectStrategy.onSelectionChange"
@select-all="handleSelectAll($event, canSelect)"
@select-all="selectStrategy.onSelectAll($event, canSelect)"
@sort-change="onSortChange"
>
<!--TODO 不用jsx写, 感觉template逻辑有点不清晰了-->
@@ -1106,12 +1106,6 @@ export default {
this.page = val
this.getList()
},
handleSelectAll(e, canSelect) {
this.loading = true
this.selectStrategy.onSelectAll(e, canSelect, () => {
this.loading = false
})
},
/**
* 切换某一行的选中状态,如果使用了第二个参数,则是设置这一行选中与否
*

View File

@@ -6,8 +6,6 @@
* 多选策略接口
*/
class StrategyAbstract {
cachedSelectedData = []
constructor(elDataTable) {
this.elDataTable = elDataTable
// 绑定this后可直接在template中使用
@@ -36,14 +34,12 @@ class StrategyNormal extends StrategyAbstract {
onSelectionChange(val) {
this.elDataTable.selected = val
}
/**
* toggleRowSelection和clearSelection的表现与el-table一致
*/
toggleRowSelection(...args) {
return this.elTable.toggleRowSelection(...args)
}
clearSelection() {
return this.elTable.clearSelection()
}
@@ -54,12 +50,12 @@ class StrategyNormal extends StrategyAbstract {
*/
class StrategyPersistSelection extends StrategyAbstract {
/**
* el-tableselection-change 事件不适用于开启跨页保存的情况。
* 比如,当开启 persistSelection时发生以下两个场景
* el-tableselection-change事件不适用于开启跨页保存的情况。
* 比如当开启persistSelection时发生以下两个场景
* 1. 用户点击翻页
* 2. 用户点击行首的切换全选项按钮,清空当前页多选项数据
* 其中场景 1 应该保持 selected 不变;而场景 2 只应该从 selected 移除当前页所有行,保留其他页面的多选状态。
* 但 el-tableselection-change 事件在两个场景中无差别发生,所以这里不处理这个事件
* 其中场景1应该保持selected不变而场景2只应该从selected移除当前页所有行保留其他页面的多选状态。
* 但el-tableselection-change事件在两个场景中无差别发生所以这里不处理这个事件
*/
/**
@@ -67,108 +63,53 @@ class StrategyPersistSelection extends StrategyAbstract {
*/
onSelect(selection, row) {
const isChosen = selection.indexOf(row) > -1
this.toggleRowSelection(row, isChosen)
}
/**
* 用户切换当前页的多选
*/
onSelectAll(selection, selectable = () => true, callback) {
const { id, selected, data } = this.elDataTable
const selectedIds = new Set(selected.map(r => r[id]))
// 获取当前所有已选择的项
const selectedRows = data.filter(r => selection.includes(r))
// 判断是否已全选
const isSelected = data.every(r => selectable(r) && selectedRows.includes(r))
const rowsToSelect = []
const rowsToDeselect = []
data.forEach(r => {
onSelectAll(selection, selectable = () => true) {
const isSelected = !!selection.length
this.elDataTable.data.forEach(r => {
if (selectable(r)) {
const isRowSelected = selectedIds.has(r[id])
if (isSelected && !isRowSelected) {
rowsToSelect.push(r)
} else if (!isSelected && isRowSelected) {
rowsToDeselect.push(r)
}
this.toggleRowSelection(r, isSelected)
}
})
if (isSelected) {
rowsToSelect.forEach(row => {
selected.push(row)
selectedIds.add(row[id])
})
rowsToDeselect.forEach(row => {
this.elDataTable.toggleRowSelection(row, true)
})
} else {
rowsToDeselect.forEach(row => {
const index = selected.findIndex(item => item[id] === row[id])
if (index !== -1) {
selected.splice(index, 1)
}
selectedIds.delete(row[id])
})
rowsToSelect.forEach(row => {
this.elDataTable.toggleRowSelection(row, false)
})
}
// this.elTable.selected = Array.from(selectedIds).map(id => {
// return data.find(r => r[id] === id)
// })
callback()
}
/**
* toggleRowSelectionclearSelection 管理 elDataTableselected 数组
* 记得最后要将状态同步到 el-table
* toggleRowSelectionclearSelection管理elDataTableselected数组
* 记得最后要将状态同步到el-table中
*/
toggleRowSelection(row, isSelected) {
const { id, selected } = this.elDataTable
const foundIndex = selected.findIndex(r => r[id] === row[id])
if (typeof isSelected === 'undefined') {
isSelected = foundIndex <= -1
}
if (isSelected && foundIndex === -1) {
selected.push(row)
} else if (!isSelected && foundIndex > -1) {
selected.splice(foundIndex, 1)
}
this.elDataTable.$emit('toggle-row-selection', isSelected, row)
this.updateElTableSelection()
}
clearSelection() {
this.elDataTable.selected = []
this.updateElTableSelection()
}
/**
* 将selected状态同步到el-table中
*/
updateElTableSelection() {
const { data, id, selected } = this.elDataTable
// 历史勾选的行已经不在当前页了所以要将当前页的行数据和selected合并
const mergeData = _.uniqWith([...data, ...selected], _.isEqual)
mergeData.forEach(r => {
const isSelected = !!selected.find(r2 => r[id] === r2[id])
if (!this.elTable) {
return
}
this.elTable.toggleRowSelection(r, isSelected)
})
}

View File

@@ -106,7 +106,7 @@ export default {
{
name: 'actionSetting',
icon: 'system-setting',
tip: this.$t('ListPreference'),
tip: this.$t('TableSetting'),
has: this.hasColumnSetting,
callback: this.handleTableSettingClick.bind(this)
},

View File

@@ -131,16 +131,6 @@ export default {
}
},
computed: {},
watch: {
cellValue: {
handler(newValue) {
if (newValue) {
this.initial = this.formatterArgs.getLabels(this.cellValue)
this.iLabels = [...this.initial]
}
}
}
},
mounted() {
this.initial = this.formatterArgs.getLabels(this.cellValue)
this.iLabels = [...this.initial]

View File

@@ -24,9 +24,6 @@ export default {
},
isDisplay(row) {
return true
},
callback({ row }) {
return null
}
}
}
@@ -52,11 +49,10 @@ export default {
methods: {
onChange(val) {
this.$axios.patch(this.patchUrl, this.patchData).then(res => {
this.formatterArgs.callback(this.row)
this.$message.success(this.$t('UpdateSuccessMsg'))
this.$message.success(this.$t('updateSuccessMsg'))
}).catch(err => {
this.value = !val
this.$message.error(this.$t('UpdateErrorMsg' + ' ' + err))
this.$message.error(this.$t('updateErrorMsg' + ' ' + err))
})
}
}

View File

@@ -234,10 +234,12 @@ export default {
delete routeFilter.search
}
const asFilterTags = _.cloneDeep(this.filterTags)
this.filterTags = {
...asFilterTags,
...routeFilter
}
setTimeout(() => {
this.filterTags = {
...asFilterTags,
...routeFilter
}
}, 100)
},
getValueLabel(key, value) {
for (const field of this.options) {

View File

@@ -1,17 +1,12 @@
import VueCookie from 'vue-cookie'
import store from '@/store'
export function getLangCode(withInternalCode = false) {
export function getLangCode() {
const cookieLang = VueCookie.get('django_language')
let lang = cookieLang || navigator.language.toLowerCase()
if (withInternalCode) {
const languages = store.getters.publicSettings['LANGUAGES']
for (const langObj of languages) {
if (langObj['other_codes'].indexOf(lang) > -1) {
lang = langObj['code']
break
}
}
const browserLang = navigator.systemLanguage || navigator.language || navigator.userLanguage
let lang = cookieLang || browserLang || 'en'
lang = lang.replace('-', '_')
if (lang !== 'zh_hant') {
lang = lang.slice(0, 2)
}
return lang
}

View File

@@ -52,7 +52,7 @@ export default {
<style lang="scss" scoped>
.app-main {
background-color: #f3f3f4;
height: 100vh !important;
height: 100% !important;
width: 100%;
position: relative;
overflow: auto;

View File

@@ -25,6 +25,7 @@ export default {
return {
langCookeName: 'django_language', // 后端Django需要的COOKIE KEY
supportLanguages: [],
currentLangCode: '',
defaultLang: {
title: 'English',
code: 'en',
@@ -40,25 +41,41 @@ export default {
}, {})
},
currentLang() {
const lang = getLangCode(true)
return this.supportedLangMapper[lang] || this.defaultLang
const lang = this.supportedLangMapper[this.currentLangCode] || this.defaultLang
return lang
}
},
mounted() {
this.currentLangCode = getLangCode()
this.supportLanguages = store.getters.publicSettings['LANGUAGES'].map(item => {
let code = item.code.replace('-', '_')
if (code !== 'zh_hant') {
code = code.slice(0, 2)
}
return {
title: item.name,
code: item.code,
code: code,
cookieCode: item.code
}
})
this.changeMomentLang()
},
methods: {
changeLang() {
if (this.currentLang.code !== this.$i18n.locale) {
this.changeLangTo(this.currentLang)
}
},
changeMomentLang() {
const lang = getLangCode()
this.$moment.locale(lang)
document.documentElement.lang = lang
if (this.currentLang.code.indexOf('en') > -1) {
this.$moment.locale('en')
} else if (this.currentLang.code.indexOf('ja') > -1) {
this.$moment.locale('ja')
} else if (this.currentLang.code.indexOf('zh_hant') > -1) {
this.$moment.locale('zh-tw')
} else {
this.$moment.locale('zh-cn')
}
},
changeLangTo(item) {
this.$axios.get(`/core/i18n/${item.cookieCode}/`).then(() => {

View File

@@ -99,11 +99,8 @@ export default {
},
currentOrgId() {
const usingOrgIds = this.usingOrgs.map(o => o.id)
let currentOrgId = this.currentOrg.id
const find = usingOrgIds.indexOf(currentOrgId) > -1
if (!find) {
currentOrgId = null
}

View File

@@ -176,7 +176,7 @@ export default {
// 未找到与之对应的
& ::v-deep .el-submenu__title {
font-family: "Open Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
font-family: "open sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
padding: 0 8px;
line-height: $headerHeight;
height: $headerHeight;

View File

@@ -116,6 +116,7 @@ export default {
<style lang="scss" scoped>
.page {
height: calc(100vh - 50px);
overflow-y: hidden;
overflow-x: hidden;

View File

@@ -80,5 +80,15 @@ export default [
}
}
]
},
{
path: 'job-execution-log',
name: 'JobExecutionLog',
component: () => import('@/views/audits/JobExecutionLogList'),
meta: {
title: i18n.t('JobExecutionLog'),
icon: 'task',
permissions: ['audits.view_joblog']
}
}
]

View File

@@ -3,7 +3,6 @@ import i18n from '@/i18n/i18n'
import SessionRoutes from './sessions'
import LogRoutes from './audits'
import JobRoutes from './jobs'
import empty from '@/layout/empty'
import store from '@/store'
@@ -55,18 +54,6 @@ export default {
permissions: []
},
children: LogRoutes
},
{
path: '/audit/jobs',
component: empty,
redirect: '',
name: 'AuditsJobs',
meta: {
title: i18n.t('JobsAudit'),
icon: 'job',
permissions: ['audits.view_joblog']
},
children: JobRoutes
}
]
}

View File

@@ -1,53 +0,0 @@
import i18n from '@/i18n/i18n'
import empty from '@/layout/empty'
export default [
{
path: 'job-list',
name: 'JobList',
component: empty,
redirect: {
name: 'JobList'
},
meta: {
title: i18n.t('JobList'),
icon: 'task',
permissions: ['audits.view_joblog']
},
children: [
{
path: '',
name: 'AuditJobList',
component: () => import('@/views/audits/JobPeriodTaskList'),
meta: {
title: i18n.t('JobList'),
permissions: []
}
}
]
},
{
path: 'job-execution-log',
name: 'JobExecutionLog',
component: empty,
redirect: {
name: 'AuditJobExecutionLog'
},
meta: {
title: i18n.t('JobExecutionLog'),
icon: 'history',
permissions: ['audits.view_joblog']
},
children: [
{
path: '',
name: 'AuditJobExecutionLog',
component: () => import('@/views/audits/JobExecutionLogList'),
meta: {
title: i18n.t('JobExecutionLog'),
permissions: ['audits.view_joblog']
}
}
]
}
]

View File

@@ -148,11 +148,11 @@ const actions = {
}
commit('SET_CURRENT_ORG', systemOrg)
},
leaveSettingOrg({ commit, state }) {
const preOrg = getPreOrgLocal(state.profile.username)
if (!preOrg) return
leaveSettingOrg({ commit }) {
const preOrg = store.state.users.preOrg
if (!preOrg) {
return
}
commit('SET_CURRENT_ORG', preOrg)
},
enterGlobalOrg({ commit }) {

View File

@@ -38,7 +38,7 @@ body {
-moz-osx-font-smoothing: grayscale;
-webkit-font-smoothing: auto;
color: var(--color-text-primary);
font-family: "Open Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
font-family: "open sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
font-size: 13px;
line-height: 1.428;
}
@@ -593,9 +593,3 @@ li.rmenu i.fa {
height: 6px; /* 设置水平滚动条的高度 */
}
}
.black-theme-popover {
width: 300px;
max-height: 700px;
overflow-y: scroll;
}

View File

@@ -23,10 +23,6 @@ $single-menu-height: 38px;
}
}
.el-menu--vertical {
background-color: #fff;
}
.el-menu {
border-right: none !important;
background-color: inherit !important;
@@ -143,10 +139,6 @@ $single-menu-height: 38px;
.nest-menu .level2-menu {
line-height: $single-menu-height;
}
.el-tooltip {
width: 55px !important;
}
}
}
}

View File

@@ -128,8 +128,6 @@ export function getErrorResponseMsg(error) {
}).filter(i => i).join('; ')
} else if (typeof data === 'string') {
return data
} else {
msg = error.toString()
}
return msg
}
@@ -312,26 +310,4 @@ export function toSentenceCase(string) {
}).join(' ')
return s[0].toUpperCase() + s.slice(1)
}
export { BASE_URL }
export function openNewWindow(url) {
let count
let top = 50
count = parseInt(window.sessionStorage.getItem('newWindowCount'), 10)
if (isNaN(count)) {
count = 0
}
let left = 100 + count * 100
top = 50 + count * 50
if (left + screen.width / 3 > screen.width) {
// 支持两排足以
top = screen.height / 3
count = 1
left = 100
}
let params = 'toolbar=yes,scrollbars=yes,resizable=yes'
params = params + `,top=${top},left=${left},width=${screen.width / 3},height=${screen.height / 3}`
window.sessionStorage.setItem('newWindowCount', `${count + 1}`)
window.open(url, '_blank', params)
}

View File

@@ -1,6 +1,5 @@
const moment = require('moment')
import { getLangCode } from '@/i18n/utils'
import store from '@/store'
function getTimeUnits(u) {
const units = {
@@ -9,7 +8,7 @@ function getTimeUnits(u) {
'm': '分',
's': '秒'
}
if (getLangCode(true) === 'zh-hans') {
if (getLangCode() === 'zh') {
return units[u]
} else {
return u
@@ -125,8 +124,3 @@ export function formatDate(inputTime) {
// return y + '-' + m + '-' + d + ' ' + h + ':' + minute + ':' + second
return y + '-' + m + '-' + d + 'T' + h + ':' + minute + ':' + second
}
export function getDefaultExpiredDays() {
const years = store.getters.publicSettings.DEFAULT_EXPIRED_YEARS
return getDayFuture(years * 365, new Date()).toISOString()
}

View File

@@ -19,7 +19,7 @@ export default {
initial: {
is_periodic: false,
password_rules: {
length: 36
length: 16
},
interval: 24,
accounts: [],

View File

@@ -66,8 +66,7 @@ export const getChangeSecretFields = () => {
is_periodic,
accounts: {
label: i18n.t('Accounts'),
component: TagInput,
helpText: i18n.t('ChangeSecretAccountHelpText')
component: TagInput
}
}
}

View File

@@ -21,7 +21,7 @@ export default {
initial: {
is_periodic: false,
password_rules: {
length: 36
length: 30
},
interval: 24,
secret_type: 'password',

View File

@@ -4,12 +4,10 @@ import InputWithUnit from '@/components/Form/FormFields/InputWithUnit.vue'
import store from '@/store'
const validatorInterval = (rule, value, callback) => {
// value只能是 null 或者 >=1
if (value === null || (!isNaN(value) && Number(value) >= 1)) {
callback()
} else {
callback(new Error(i18n.t('EnsureThisValueIsGreaterThanOrEqualTo1')))
if (parseInt(value) < 1) {
return callback(new Error(i18n.t('EnsureThisValueIsGreaterThanOrEqualTo1')))
}
callback()
}
export const crontab = {

View File

@@ -2,6 +2,7 @@
<TabPage
:active-menu.sync="config.activeMenu"
:submenu="config.submenu"
@tab-click="handleTabClick"
/>
</template>
@@ -34,6 +35,18 @@ export default {
]
}
}
},
methods: {
handleTabClick(tab) {
const query = _.cloneDeep(this.$route.query)
const newQuery = {
...query,
tab: tab.name
}
this.$nextTick(() => {
this.$router.replace({ query: newQuery })
})
}
}
}
</script>

View File

@@ -48,8 +48,7 @@ export default {
},
tableConfig: {
url: tableUrl,
category: 'all',
extraQuery: { 'order': '-date_updated' }
category: 'all'
},
headerActions: {
handleImportClick: ({ selectedRows }) => {

View File

@@ -76,20 +76,14 @@ export default {
},
callbacks: {
change: function(val) {
this.updateTaskData({ 'sync_ip_type': val })
}.bind(this)
}
},
{
title: this.$t('ReleaseAssets'),
type: 'switch',
attrs: {
model: this.object.task.release_assets,
disabled: !this.hasEditPerm()
},
callbacks: {
change: function(val) {
this.updateTaskData({ 'release_assets': val })
this.$axios.patch(
`/api/v1/xpack/cloud/sync-instance-tasks/${this.object.task.id}/`,
{ 'sync_ip_type': val }
).then(res => {
this.$message.success(this.$t('UpdateSuccessMsg'))
}).catch(err => {
this.$message.error(this.$t('UpdateErrorMsg' + ' ' + err))
})
}.bind(this)
}
}
@@ -203,16 +197,6 @@ export default {
methods: {
hasEditPerm() {
return this.$hasPerm('xpack.change_account') && this.$hasPerm('xpack.change_syncinstancetask')
},
updateTaskData(data) {
this.$axios.patch(
`/api/v1/xpack/cloud/sync-instance-tasks/${this.object.task.id}/`,
data
).then(res => {
this.$message.success(this.$tc('UpdateSuccessMsg'))
}).catch(err => {
this.$message.error(this.$tc('UpdateErrorMsg' + ' ' + err))
})
}
}
}

View File

@@ -73,7 +73,13 @@ export default {
btn.loading = true
}
})
form.value.interval = parseInt(form.value.interval, 10)
if (form.value.interval && typeof form.value.interval === 'number') {
form.value.interval = parseInt(form.value.interval, 10)
} else {
form.value.interval = 24
}
this.$refs.form.$refs.form.dataForm.submitForm('form', false)
},
handleSubmitSuccess(res) {

View File

@@ -39,8 +39,7 @@ export default {
submitBtnSize: 'mini',
submitBtnText: this.$t('Add'),
hasReset: false,
onSubmit: () => {
},
onSubmit: () => {},
submitMethod: () => 'post',
getUrl: () => '',
cleanFormValue(data) {
@@ -87,11 +86,7 @@ export default {
this.formConfig.fieldsMeta.protocols.el.hidden = true
}
this.resourceType = val
if (url) {
this.formConfig.fieldsMeta.value.el.ajax.url = url
} else {
this.formConfig.fieldsMeta.attr.el.remote = false
}
this.formConfig.fieldsMeta.value.el.ajax.url = url
this.formConfig.fieldsMeta.value.el.options = options
}
}
@@ -156,31 +151,21 @@ export default {
tableConfig: {
columns: [
{ prop: 'attr', label: this.$t('ResourceType'), formatter: tableFormatter('resource_type') },
{
prop: 'value', label: this.$t('Resource'), formatter: tableFormatter('resource', () => {
return this.globalResource
})
},
{ prop: 'value', label: this.$t('Resource'), formatter: tableFormatter('resource', () => { return this.globalResource }) },
{ prop: 'protocols', label: this.$t('Other'), formatter: tableFormatter('protocols') },
{
prop: 'action',
label: this.$t('Action'),
align: 'center',
width: '100px',
formatter: (row, col, cellValue, index) => {
return (
<div className='input-button'>
<el-button
icon='el-icon-minus'
size='mini'
style={{ 'flexShrink': 0 }}
type='danger'
onClick={this.handleDelete(index)}
/>
</div>
)
}
}
{ prop: 'action', label: this.$t('Action'), align: 'center', width: '100px', formatter: (row, col, cellValue, index) => {
return (
<div className='input-button'>
<el-button
icon='el-icon-minus'
size='mini'
style={{ 'flexShrink': 0 }}
type='danger'
onClick={ this.handleDelete(index) }
/>
</div>
)
} }
],
totalData: this.value || [],
hasPagination: false
@@ -192,9 +177,7 @@ export default {
},
methods: {
init() {
this.nameOptions.map((o) => {
this.globalResource[o.value] = o.label
})
this.nameOptions.map((o) => { this.globalResource[o.value] = o.label })
},
onSubmit() {
this.$emit('input', this.tableConfig.totalData)
@@ -235,11 +218,9 @@ export default {
::v-deep .el-form-item:nth-child(-n+3) {
width: 43.5%;
}
::v-deep .el-form-item:last-child {
width: 6%;
}
.action-input {
margin-top: -10px;
}

View File

@@ -162,7 +162,7 @@ export default {
},
activated() {
setTimeout(() => {
this.tab.activeMenu = window.localStorage.getItem('lastTab') || 'host'
this.tab.activeMenu = window.localStorage.getItem('lastTab')
this.$refs.genericListTable.reloadTable()
}, 300)
},

View File

@@ -42,7 +42,7 @@ function updatePlatformProtocols(vm, platformType, updateForm, isPlatformChanged
'username_selector': setting.username_selector
})
}
vm.iConfig.fieldsMeta.protocols.el.choices = platformProtocols
vm.iConfig.fieldsMeta.protocols.el.choices = platformProtocols.map(item => ({ name: item.name, port: item.port }))
updateForm({ protocols: [] })
}), 100)
}

View File

@@ -1,6 +1,6 @@
<template>
<div>
<GenericListPage ref="ListPage" :header-actions="headerActions" :table-config="tableConfig" />
<GenericListPage :header-actions="headerActions" :table-config="tableConfig" />
</div>
</template>
@@ -8,7 +8,6 @@
import GenericListPage from '@/layout/components/GenericListPage'
import { ActionsFormatter } from '@/components/Table/TableFormatters'
import { openTaskPage } from '@/utils/jms'
import { stopJob } from '@/api/ops'
export default {
components: {
@@ -46,20 +45,6 @@ export default {
callback: ({ row }) => {
openTaskPage(row.task_id)
}
},
{
title: this.$t('Stop'),
name: 'stop',
can: ({ row }) => {
return !row.is_finished
},
type: 'danger',
callback: ({ row }) => {
stopJob({ task_id: row.task_id }).then(() => {
this.$refs.ListPage.reloadTable()
this.$message.success(this.$t('StopJobMsg'))
})
}
}
]
}

View File

@@ -1,88 +0,0 @@
<template>
<div>
<GenericListPage ref="ListPage" :header-actions="headerActions" :table-config="tableConfig" />
</div>
</template>
<script type="text/jsx">
import GenericListPage from '@/layout/components/GenericListPage'
import { SwitchFormatter } from '@/components/Table/TableFormatters'
export default {
components: {
GenericListPage
},
data() {
const vm = this
return {
tableConfig: {
url: '/api/v1/audits/jobs/',
columnsShow: {
min: ['name', 'material'],
default: [
'name', 'material', 'type', 'crontab', 'interval', 'created_by', 'is_periodic_display', 'is_periodic'
]
},
columns: [
'name', 'args', 'material', 'type', 'crontab', 'interval', 'date_last_run', 'summary',
'created_by', 'is_periodic_display', 'is_periodic'
],
columnsMeta: {
actions: {
has: false
},
name: {
formatter: (row) => row.name
},
material: {
width: '200px',
label: this.$t('Command')
},
summary: {
width: '130px',
label: `${this.$t('Success')}/${this.$t('Total')}`,
formatter: (row) => {
return <div>
<span Class='text-primary'>{row.summary.success}</span>/
<span>{row.summary.total}</span>
</div>
}
},
is_periodic_display: {
width: '100px',
label: this.$t('Periodic')
},
is_periodic: {
width: '140px',
label: `${this.$t('Enable')}/${this.$t('Disable')}`,
formatter: SwitchFormatter,
formatterArgs: {
isDisplay(row) {
return row.is_periodic
},
getPatchUrl(row) {
return `/api/v1/audits/jobs/${row.id}/`
},
getPatchData(row) {
return {
is_periodic: !row.is_periodic
}
},
callback() {
vm.$refs.ListPage.reloadTable()
}
}
}
}
},
headerActions: {
hasLeftActions: false,
hasImport: false
}
}
}
}
</script>
<style>
</style>

View File

@@ -128,8 +128,7 @@ export default {
tip += current.label + '' + current.total + '<br/>'
}
return tip
},
appendToBody: true
}
},
grid: {
top: '60%',

View File

@@ -52,7 +52,6 @@ export default {
title: this.$tc('Select'),
name: 'select',
can: true,
type: 'primary',
callback: ({ row }) => {
this.$emit('select', row)
this.iVisible = false

View File

@@ -8,12 +8,6 @@
:visible.sync="showOpenAdhocSaveDialog"
/>
<VariableHelpDialog :visible.sync="showHelpDialog" />
<SetVariableDialog
:form-data="variableFormData"
:query-param="variableQueryParam"
:visible.sync="showSetVariableDialog"
@submit="onSubmitVariable"
/>
<AssetTreeTable ref="TreeTable" :tree-setting="treeSetting">
<template slot="table">
<div class="transition-box" style="width: calc(100% - 17px);">
@@ -41,7 +35,6 @@
</template>
<script>
import $ from '@/utils/jquery-vendor.js'
import AssetTreeTable from '@/components/Apps/AssetTreeTable'
import QuickJobTerm from '@/views/ops/Adhoc/components/QuickJobTerm.vue'
import CodeEditor from '@/components/Form/FormFields/CodeEditor'
@@ -49,8 +42,7 @@ import Page from '@/layout/components/Page'
import AdhocOpenDialog from './AdhocOpenDialog.vue'
import AdhocSaveDialog from './AdhocSaveDialog.vue'
import VariableHelpDialog from './VariableHelpDialog.vue'
import SetVariableDialog from '@/views/ops/Template/components/SetVariableDialog.vue'
import { createJob, getJob, getTaskDetail, stopJob } from '@/api/ops'
import { createJob, getJob, getTaskDetail, StopJob } from '@/api/ops'
export default {
name: 'CommandExecution',
@@ -58,7 +50,6 @@ export default {
VariableHelpDialog,
AdhocSaveDialog,
AdhocOpenDialog,
SetVariableDialog,
AssetTreeTable,
Page,
QuickJobTerm,
@@ -78,7 +69,6 @@ export default {
showHelpDialog: false,
showOpenAdhocDialog: false,
showOpenAdhocSaveDialog: false,
showSetVariableDialog: false,
DataZTree: 0,
runas: '',
runasPolicy: 'skip',
@@ -102,10 +92,6 @@ export default {
type: 'primary'
},
callback: () => {
if (this.variableFormData.length !== 0) {
this.showSetVariableDialog = true
return
}
this.execute()
}
},
@@ -300,19 +286,9 @@ export default {
view: {
dblClickExpand: false,
showLine: true
},
callback: {
onCheck: function(_event, treeId, treeNode) {
const treeObj = $.fn.zTree.getZTreeObj(treeId)
if (treeNode.checked) {
treeObj.expandNode(treeNode, true, false, true)
}
}
}
},
iShowTree: true,
variableFormData: [],
variableQueryParam: ''
iShowTree: true
}
},
computed: {
@@ -323,13 +299,6 @@ export default {
return this.$refs.TreeTable.$refs.TreeList.$refs.AutoDataZTree.$refs.AutoDataZTree.$refs.dataztree.$refs.ztree
}
},
watch: {
command(iNew, iOld) {
if (iNew === '') {
this.variableFormData = []
}
}
},
mounted() {
this.enableWS()
this.initData()
@@ -361,10 +330,6 @@ export default {
}
},
onSelectAdhoc(adhoc) {
this.variableFormData = adhoc?.variable.map((data) => {
return data.form_data
})
this.variableQueryParam = 'adhoc=' + adhoc.id
this.command = adhoc.args
},
enableWS() {
@@ -458,6 +423,7 @@ export default {
this.$message.error(this.$tc('RequiredRunas'))
return
}
const data = {
assets: hosts,
nodes: nodes,
@@ -472,9 +438,6 @@ export default {
if (this.chdir) {
data.chdir = this.chdir
}
if (this.parameters) {
data.parameters = this.parameters
}
createJob(data).then(res => {
this.executionInfo.timeCost = 0
this.executionInfo.status = { value: 'running', label: this.$t('Running') }
@@ -486,9 +449,9 @@ export default {
})
},
stop() {
stopJob({ task_id: this.currentTaskId }).then(() => {
StopJob({ task_id: this.currentTaskId }).then(() => {
this.xterm.write('\x1b[31m' +
this.$tc('StopLogOutput').replace('currentTaskId', this.currentTaskId) + '\x1b[0m')
this.$tc('StopLogOutput').replace('currentTaskId', this.currentTaskId) + '\x1b[0m')
this.xterm.write(this.wrapperError(''))
this.getTaskStatus()
}).catch((e) => {
@@ -504,11 +467,6 @@ export default {
}
this.toolbar.left.run.isVisible = this.executionInfo.status.value === 'running'
this.toolbar.left.stop.isVisible = this.executionInfo.status.value !== 'running'
},
onSubmitVariable(parameters) {
this.parameters = parameters
this.showSetVariableDialog = false
this.execute()
}
}
}

View File

@@ -75,7 +75,6 @@ export default {
fontFamily: 'monaco, Consolas, "Lucida Console", monospace',
lineHeight: 1.2,
fontSize: 13,
scrollback: 9999999,
rightClickSelectsWord: true,
theme: {
background: '#fff',

View File

@@ -28,14 +28,11 @@ export default {
columnsShow: {
min: ['material', 'actions'],
default: [
'id', 'material', 'job_type', 'is_finished', 'is_success',
'id', 'job', 'material', 'job_type', 'is_finished', 'is_success',
'time_cost', 'date_created', 'actions'
]
},
columnsMeta: {
material: {
width: '500px'
},
id: {
formatter(row) {
return row.id.slice(0, 8)
@@ -47,7 +44,6 @@ export default {
}
},
is_finished: {
width: '100px',
formatter: (row) => {
if (row.is_finished) {
return <i Class='fa fa-check text-primary'/>
@@ -56,7 +52,6 @@ export default {
}
},
is_success: {
width: '100px',
formatter: (row) => {
if (!row.is_finished) {
return <i Class='fa fa fa-spinner fa-spin'/>

View File

@@ -535,7 +535,7 @@ export default {
.empty-file-tip {
position: absolute;
right: calc(50% - 230px);
right: 20%;
top: 50%;
font-size: 18px;
color: #c5c9cc;

View File

@@ -3,25 +3,14 @@
<el-col :md="16" :sm="24">
<AutoDetailCard :fields="detailFields" :object="object" :url="url" />
</el-col>
<el-col :md="8" :sm="24">
<IBox :title="$tc('Variable')">
<Variable
:value.sync="object.variable"
@input="updateVariable"
/>
</IBox>
</el-col>
</el-row>
</template>
<script type="text/jsx">
import AutoDetailCard from '@/components/Cards/DetailCard/auto'
import Variable from '@/views/ops/Template/components/Variable.vue'
import { IBox } from '@/components'
export default {
components: {
IBox, Variable,
AutoDetailCard
},
props: {
@@ -48,15 +37,7 @@ export default {
return this.object.name
}
},
methods: {
updateVariable() {
const url = `/api/v1/ops/jobs/${this.object.id}/`
this.$axios.patch(url, { variable: this.object.variable }).then(() => {
this.$message.success(this.$tc('UpdateSuccessMsg'))
})
}
}
methods: {}
}
</script>

View File

@@ -13,8 +13,6 @@ import i18n from '@/i18n/i18n'
import VariableHelpDialog from '@/views/ops/Adhoc/VariableHelpDialog.vue'
import { Required } from '@/components/Form/DataForm/rules'
import { crontab, interval } from '@/views/accounts/const'
import LoadTemplateLink from '@/views/ops/Job/components/LoadTemplateLink.vue'
import Variable from '@/views/ops/Template/components/Variable'
export default {
components: {
@@ -29,8 +27,8 @@ export default {
url: '/api/v1/ops/jobs/',
fields: [
[this.$t('Basic'), ['name', 'type', 'instant']],
[this.$t('Asset'), ['assets', 'nodes', 'runas', 'runas_policy']],
[this.$t('Task'), ['module', 'argsLoadFromTemplate', 'args', 'playbook', 'variable', 'chdir', 'timeout', 'parameters']],
[this.$t('Task'), ['module', 'args', 'playbook', 'chdir', 'timeout']],
[this.$t('Asset'), ['assets', 'runas', 'runas_policy']],
[this.$t('Plan'), ['run_after_save', 'is_periodic', 'interval', 'crontab']],
[this.$t('Other'), ['comment']]
],
@@ -90,66 +88,19 @@ export default {
return { label: item.name, value: item.id }
}
}
},
on: {
change: ([event], updateForm) => {
this.queryParam = `playbook=${event.pk}`
this.$axios.get(`/api/v1/ops/playbooks/${event.pk}/`,
).then(data => {
data?.variable.map(item => {
delete item.job
delete item.playbook
delete item.id
return item
})
updateForm({ variable: data.variable })
})
}
}
},
assets: {
type: 'assetSelect',
component: AssetSelect,
label: this.$t('Asset'),
rules: [Required],
el: {
baseUrl: '/api/v1/perms/users/self/assets/',
baseNodeUrl: '/api/v1/perms/users/self/nodes/',
typeUrl: '/api/v1/perms/users/self/nodes/children-with-assets/category/tree',
value: []
}
},
nodes: {
el: {
value: [],
ajax: {
url: '/api/v1/perms/users/self/nodes/',
filterOption: (item) => {
if (item.value !== 'favorite') {
return item
}
},
transformOption: (item) => {
return { label: item.full_value || item.name, value: item.id }
}
}
}
},
argsLoadFromTemplate: {
label: this.$t('Templates'),
hidden: (formValue) => {
return formValue.type !== 'adhoc'
},
component: LoadTemplateLink,
on: {
change: ([event], updateForm) => {
updateForm({
args: event.args,
module: event.module.value,
variable: event.variable
})
}
}
},
args: {
rules: [Required],
hidden: (formValue) => {
@@ -173,9 +124,6 @@ export default {
]
}
},
variable: {
component: Variable
},
timeout: {
helpText: i18n.t('TimeoutHelpText')
},
@@ -193,7 +141,7 @@ export default {
run_after_save: {
type: 'checkbox',
hidden: (formValue) => {
return true
return this.instantTask
}
},
is_periodic: {
@@ -202,34 +150,11 @@ export default {
return this.instantTask
}
},
parameters: {
hidden: () => {
return true
}
},
interval,
crontab
},
createSuccessNextRoute: { name: 'JobManagement' },
updateSuccessNextRoute: { name: 'JobManagement' },
cleanFormValue: (data) => {
Object.assign(data, { periodic_variable: this.periodicVariableValue })
return data
},
moreButtons: [
{
title: this.$t('ExecuteAfterSaving'),
callback: (value, form, btn) => {
form.value.run_after_save = true
const parameters = form.value.variable.reduce((acc, item) => {
acc[item.var_name] = item.default_value
return acc
}, {})
form.value['parameters'] = parameters
this.submitForm(form, btn)
}
}
]
updateSuccessNextRoute: { name: 'JobManagement' }
}
},
mounted() {
@@ -266,16 +191,7 @@ export default {
this.ready = true
}
},
methods: {
submitForm(form, btn) {
form.validate((valid) => {
if (valid) {
btn.loading = true
}
})
this.$refs.form.$refs.createUpdateForm.$refs.form.$refs.dataForm.submitForm('form', false)
}
}
methods: {}
}
</script>

View File

@@ -1,16 +1,129 @@
<template>
<div>
<BaseJob />
<GenericListTable :header-actions="headerActions" :table-config="tableConfig" />
<JobRunDialog v-if="showJobRunDialog" :item="item" :visible.sync="showJobRunDialog" @submit="runJob" />
</div>
</template>
<script>
import BaseJob from './BaseJob'
import JobRunDialog from '@/views/ops/Job/JobRunDialog'
import GenericListTable from '@/layout/components/GenericListTable'
import { openTaskPage } from '@/utils/jms'
import { ActionsFormatter, DateFormatter, DetailFormatter } from '@/components/Table/TableFormatters'
export default {
name: 'Adhoc',
components: {
BaseJob
GenericListTable,
JobRunDialog
},
data() {
return {
item: {},
tableConfig: {
url: '/api/v1/ops/jobs/?type=adhoc',
columnsShow: {
min: ['name', 'actions'],
default: [
'name', 'type', 'asset_amount', 'average_time_cost',
'summary', 'date_last_run', 'actions'
]
},
columns: [
'name', 'type', 'summary', 'average_time_cost', 'asset_amount',
'date_last_run', 'comment', 'date_updated', 'date_created', 'actions'
],
columnsMeta: {
name: {
width: '140px',
formatter: DetailFormatter,
formatterArgs: {
can: true,
getRoute: ({ row }) => ({
name: 'JobDetail',
params: { id: row.id }
})
}
},
type: {
width: '96px',
formatter: (row) => {
return row.type.label
}
},
comment: {
width: '240px'
},
summary: {
label: this.$t('Summary'),
formatter: (row) => {
return row.summary['success'] + '/' + row.summary['total']
}
},
average_time_cost: {
formatter: (row) => {
return row.average_time_cost.toFixed(2) + 's'
}
},
asset_amount: {
label: this.$t('AssetsOfNumber'),
formatter: (row) => {
return row.assets.length
}
},
date_last_run: {
width: '140px',
formatter: DateFormatter
},
actions: {
formatter: ActionsFormatter,
formatterArgs: {
hasUpdate: true,
canUpdate: this.$hasPerm('ops.change_job') && !this.$store.getters.currentOrgIsRoot,
updateRoute: 'JobUpdate',
hasDelete: true,
canDelete: this.$hasPerm('ops.delete_job'),
hasClone: false,
extraActions: [
{
title: this.$t('Run'),
name: 'run',
can: this.$hasPerm('ops.add_jobexecution') && !this.$store.getters.currentOrgIsRoot,
callback: ({ row }) => {
if (row?.use_parameter_define && row?.parameters_define) {
const params = JSON.parse(row.parameters_define)
if (Object.keys(params).length > 0) {
this.item = row
this.showJobRunDialog = true
}
} else {
this.runJob(row)
}
}
}
]
}
}
}
},
headerActions: {
createRoute: 'JobCreate',
hasRefresh: true,
hasExport: false,
hasImport: false
},
showJobRunDialog: false
}
},
methods: {
runJob(row, parameters) {
this.$axios.post('/api/v1/ops/job-executions/', {
job: row.id,
parameters: parameters
}).then((resp) => {
openTaskPage(resp.task_id)
})
}
}
}
</script>

View File

@@ -1,167 +0,0 @@
<template>
<div>
<GenericListTable :header-actions="headerActions" :table-config="tableConfig" />
<JobRunDialog v-if="showJobRunDialog" :item="item" :visible.sync="showJobRunDialog" @submit="runJob" />
<SetVariableDialog
v-if="showVariableDialog"
:form-data="formData"
:query-param="'job=' + item.id"
:visible.sync="showVariableDialog"
@submit="runJobWithParams"
/>
</div>
</template>
<script>
import JobRunDialog from '@/views/ops/Job/JobRunDialog'
import GenericListTable from '@/layout/components/GenericListTable'
import SetVariableDialog from '@/views/ops/Template/components/SetVariableDialog.vue'
import { openTaskPage } from '@/utils/jms'
import { ActionsFormatter, DateFormatter, DetailFormatter } from '@/components/Table/TableFormatters'
export default {
components: {
GenericListTable,
JobRunDialog,
SetVariableDialog
},
props: {
type: {
type: String,
default: 'adhoc'
}
},
data() {
return {
item: {},
tableConfig: {
url: `/api/v1/ops/jobs/?type=${this.type}`,
columnsShow: {
min: ['name', 'actions'],
default: [
'name', 'type', 'asset_amount', 'average_time_cost',
'summary', 'date_last_run', 'actions'
]
},
columns: [
'name', 'type', 'summary', 'average_time_cost', 'asset_amount',
'date_last_run', 'comment', 'date_updated', 'date_created', 'actions'
],
columnsMeta: {
name: {
width: '140px',
formatter: DetailFormatter,
formatterArgs: {
can: true,
getRoute: ({ row }) => ({
name: 'JobDetail',
params: { id: row.id }
})
}
},
type: {
width: '96px',
formatter: (row) => {
return row.type.label
}
},
comment: {
width: '240px'
},
summary: {
label: this.$t('Summary'),
formatter: (row) => {
return row.summary['success'] + '/' + row.summary['total']
}
},
average_time_cost: {
formatter: (row) => {
return row.average_time_cost.toFixed(2) + 's'
}
},
asset_amount: {
label: this.$t('AssetsOfNumber'),
formatter: (row) => {
return row.assets.length
}
},
date_last_run: {
width: '140px',
formatter: DateFormatter
},
actions: {
formatter: ActionsFormatter,
formatterArgs: {
hasUpdate: true,
canUpdate: this.$hasPerm('ops.change_job') && !this.$store.getters.currentOrgIsRoot,
updateRoute: 'JobUpdate',
hasDelete: true,
canDelete: this.$hasPerm('ops.delete_job'),
hasClone: false,
extraActions: [
{
title: this.$t('Run'),
name: 'run',
order: 5,
type: 'primary',
can: this.$hasPerm('ops.add_jobexecution') && !this.$store.getters.currentOrgIsRoot,
callback: ({ row }) => {
this.item = row
if (row?.use_parameter_define && row?.parameters_define) {
const params = JSON.parse(row.parameters_define)
if (Object.keys(params).length > 0) {
this.showJobRunDialog = true
}
} else if (row?.variable?.length) {
this.showVariableDialog = true
} else {
this.runJob(row)
}
}
}
]
}
}
}
},
headerActions: {
createRoute: () => {
return {
name: 'JobCreate',
query: {
type: this.type
}
}
},
hasRefresh: true,
hasExport: false,
hasImport: false
},
showJobRunDialog: false,
showVariableDialog: false
}
},
computed: {
formData() {
return this.item.variable.map((data) => {
return data.form_data
})
}
},
methods: {
runJob(row, parameters) {
this.$axios.post('/api/v1/ops/job-executions/', {
job: row.id,
parameters: parameters
}).then((resp) => {
this.showVariableDialog = false
openTaskPage(resp.task_id)
})
},
runJobWithParams(parameters) {
this.runJob(this.item, parameters)
}
}
}
</script>

View File

@@ -1,41 +0,0 @@
<template>
<div>
<AdhocOpenDialog v-if="showOpenAdhocDialog" :visible.sync="showOpenAdhocDialog" @select="onSelectAdhoc" />
<el-link :underline="false" type="default" @click="onClick()">
{{ $t('LoadTemplate') }}
</el-link>
</div>
</template>
<script>
import AdhocOpenDialog from '@/views/ops/Adhoc/AdhocOpenDialog'
export default {
name: 'LoadTemplateLink',
components: {
AdhocOpenDialog
},
data() {
return {
showOpenAdhocDialog: false
}
},
methods: {
onClick() {
this.showOpenAdhocDialog = true
},
onSelectAdhoc(adhoc) {
adhoc?.variable.map(item => {
delete item.id
delete item.job
delete item.adhoc
return item
})
this.$emit('change', adhoc)
}
}
}
</script>
<style>
</style>

View File

@@ -1,16 +1,136 @@
<template>
<div>
<BaseJob :type="'playbook'" />
<GenericListTable :header-actions="headerActions" :table-config="tableConfig" />
<JobRunDialog v-if="showJobRunDialog" :item="item" :visible.sync="showJobRunDialog" @submit="runJob" />
</div>
</template>
<script>
import BaseJob from './BaseJob'
import JobRunDialog from '@/views/ops/Job/JobRunDialog'
import GenericListTable from '@/layout/components/GenericListTable'
import { openTaskPage } from '@/utils/jms'
import { ActionsFormatter, DateFormatter, DetailFormatter } from '@/components/Table/TableFormatters'
export default {
name: 'PlayBook',
components: {
BaseJob
GenericListTable,
JobRunDialog
},
data() {
return {
item: {},
tableConfig: {
url: '/api/v1/ops/jobs/?type=playbook',
columnsShow: {
min: ['name', 'actions'],
default: [
'name', 'type', 'asset_amount', 'average_time_cost',
'summary', 'date_last_run', 'actions'
]
},
columns: [
'name', 'type', 'summary', 'average_time_cost', 'asset_amount',
'date_last_run', 'comment', 'date_updated', 'date_created', 'actions'
],
columnsMeta: {
name: {
width: '140px',
formatter: DetailFormatter,
formatterArgs: {
can: true,
getRoute: ({ row }) => ({
name: 'JobDetail',
params: { id: row.id }
})
}
},
type: {
width: '96px',
formatter: (row) => {
return row.type.label
}
},
comment: {
width: '240px'
},
summary: {
label: this.$t('Summary'),
formatter: (row) => {
return row.summary['success'] + '/' + row.summary['total']
}
},
average_time_cost: {
formatter: (row) => {
return row.average_time_cost.toFixed(2) + 's'
}
},
asset_amount: {
label: this.$t('AssetsOfNumber'),
formatter: (row) => {
return row.assets.length
}
},
date_last_run: {
width: '140px',
formatter: DateFormatter
},
actions: {
formatter: ActionsFormatter,
formatterArgs: {
hasUpdate: true,
canUpdate: this.$hasPerm('ops.change_job') && !this.$store.getters.currentOrgIsRoot,
updateRoute: 'JobUpdate',
hasDelete: true,
canDelete: this.$hasPerm('ops.delete_job'),
hasClone: false,
extraActions: [
{
title: this.$t('Run'),
name: 'run',
can: this.$hasPerm('ops.add_jobexecution') && !this.$store.getters.currentOrgIsRoot,
callback: ({ row }) => {
if (row?.use_parameter_define && row?.parameters_define) {
const params = JSON.parse(row.parameters_define)
if (Object.keys(params).length > 0) {
this.item = row
this.showJobRunDialog = true
}
} else {
this.runJob(row)
}
}
}
]
}
}
}
},
headerActions: {
hasRefresh: true,
hasExport: false,
hasImport: false,
createRoute: () => {
return {
name: 'JobCreate',
query: {
type: 'playbook'
}
}
}
},
showJobRunDialog: false
}
},
methods: {
runJob(row, parameters) {
this.$axios.post('/api/v1/ops/job-executions/', {
job: row.id,
parameters: parameters
}).then((resp) => {
openTaskPage(resp.task_id)
})
}
}
}
</script>

View File

@@ -17,7 +17,7 @@ export default {
url: '/api/v1/ops/adhocs/',
columnsShow: {
min: ['name', 'actions'],
default: ['name', 'module', 'args', 'comment', 'scope', 'date_created', 'actions', 'created_by']
default: ['name', 'module', 'args', 'comment', 'scope', 'date_created', 'actions']
},
columnsMeta: {
name: {
@@ -38,8 +38,7 @@ export default {
canDelete: ({ row }) => {
return this.$hasPerm('ops.delete_adhoc') && row.creator === currentUserID
},
hasClone: true,
cloneRoute: 'AdhocCreate'
hasClone: false
}
}
}

View File

@@ -1,29 +1,16 @@
<template>
<el-row :gutter="20">
<el-col :md="16" :sm="24">
<AutoDetailCard :excludes="excludes" :object="object" :url="url" />
</el-col>
<el-col :md="8" :sm="24">
<IBox :title="$tc('Variable')">
<Variable
:value.sync="object.variable"
:disable-edit="disableEdit"
@input="updateVariable"
/>
</IBox>
<AutoDetailCard :fields="detailFields" :object="object" :url="url" />
</el-col>
</el-row>
</template>
<script type="text/jsx">
import AutoDetailCard from '@/components/Cards/DetailCard/auto'
import Variable from '@/views/ops/Template/components/Variable'
import { IBox } from '@/components'
export default {
components: {
IBox,
Variable,
AutoDetailCard
},
props: {
@@ -34,9 +21,7 @@ export default {
},
data() {
return {
url: `/api/v1/ops/adhocs/${this.object.id}/`,
excludes: ['variable'],
disableEdit: this.object.creator !== this.$store.state.users.profile.id
url: `/api/v1/ops/adhocs/${this.object.id}/`
}
},
computed: {
@@ -44,14 +29,7 @@ export default {
return this.object.name
}
},
methods: {
updateVariable() {
const url = `/api/v1/ops/adhocs/${this.object.id}/`
this.$axios.patch(url, { variable: this.object.variable }).then(() => {
this.$message.success(this.$tc('UpdateSuccessMsg'))
})
}
}
methods: {}
}
</script>

View File

@@ -5,7 +5,6 @@
<script>
import { GenericCreateUpdatePage } from '@/layout/components'
import CodeEditor from '@/components/Form/FormFields/CodeEditor'
import Variable from '@/views/ops/Template/components/Variable'
export default {
components: {
@@ -15,7 +14,7 @@ export default {
return {
url: '/api/v1/ops/adhocs/',
fields: [
[this.$t('Basic'), ['name', 'scope', 'module', 'args', 'variable', 'comment']]
[this.$t('Basic'), ['name', 'module', 'args', 'comment', 'scope']]
],
initial: {
module: 'shell',
@@ -24,9 +23,6 @@ export default {
fieldsMeta: {
args: {
component: CodeEditor
},
variable: {
component: Variable
}
},
createSuccessNextRoute: {
@@ -34,16 +30,6 @@ export default {
},
updateSuccessNextRoute: {
name: 'Template'
},
cleanFormValue(value) {
const isClone = this?.$route?.query.clone_from !== undefined
if (isClone) {
value?.variable.map((item) => {
delete item.id
delete item.adhoc
})
}
return value
}
}
}

View File

@@ -24,7 +24,7 @@ export default {
url: '/api/v1/ops/playbooks/',
columnsShow: {
min: ['name', 'actions'],
default: ['name', 'comment', 'scope', 'date_created', 'actions', 'created_by']
default: ['name', 'comment', 'scope', 'date_created', 'actions']
},
columnsMeta: {
name: {

View File

@@ -13,29 +13,13 @@ export default {
return {
url: '/api/v1/ops/playbooks/',
fields: [
[this.$t('Basic'), ['name', 'scope', 'comment', 'variable']]
[this.$t('Basic'), ['name', 'comment', 'scope']]
],
fieldsMeta: {
variable: {
hidden: () => true
}
},
createSuccessNextRoute: {
name: 'Template'
},
updateSuccessNextRoute: {
name: 'Template'
},
cleanFormValue(value) {
const isClone = this?.$route?.query.clone_from !== undefined
if (isClone) {
value?.variable.map((item) => {
delete item.id
delete item.playbook
delete item.job
})
}
return value
}
}
}

View File

@@ -6,7 +6,7 @@
:title="$tc('NewFile')"
:visible.sync="iVisible"
top="1vh"
width="40%"
width="20%"
@confirm="onConfirm"
>
<el-form>
@@ -45,8 +45,7 @@ export default {
}
}
},
mounted() {
},
mounted() {},
methods: {
onConfirm() {
this.$emit('confirm', this.name)

View File

@@ -23,7 +23,7 @@ export default {
return {
url: `/api/v1/ops/playbooks/${this.object.id}/`,
excludes: [
'variable', 'path', 'create_method', 'vcs_url'
'path', 'create_method', 'vcs_url'
]
}
},

View File

@@ -34,13 +34,6 @@
</el-tab-pane>
</el-tabs>
<div style="display: flex;margin-top:10px;justify-content: space-between" />
<el-form ref="form" label-position="left" label-width="30px">
<div class="form-content">
<el-form-item label="" prop="variable">
<Variable :value.sync="variables" :disable-edit.sync="disableEdit" @input="setVariable" />
</el-form-item>
</div>
</el-form>
</div>
</template>
</TreeTable>
@@ -53,15 +46,13 @@ import CodeEditor from '@/components/Form/FormFields/CodeEditor'
import item from '@/layout/components/NavLeft/Item'
import NewNodeDialog from '@/views/ops/Template/Playbook/PlaybookDetail/Editor/NewNodeDialog.vue'
import { renameFile } from '@/api/ops'
import Variable from '@/views/ops/Template/components/Variable'
export default {
name: 'CommandExecution',
components: {
NewNodeDialog,
TreeTable,
CodeEditor,
Variable
CodeEditor
},
props: {
object: {
@@ -155,8 +146,7 @@ export default {
},
iShowTree: true,
activeEditorId: '',
openedEditor: {},
variables: []
openedEditor: {}
}
},
computed: {
@@ -177,7 +167,6 @@ export default {
}
},
mounted() {
this.variables = this.object?.variable
this.onOpenEditor({ id: 'main.yml', name: 'main.yml' })
},
methods: {
@@ -297,15 +286,6 @@ export default {
},
hasChange(editor) {
return editor.value !== editor.originValue
},
setVariable(variables) {
if (this.disableEdit) {
return
}
this.$axios.patch(`/api/v1/ops/playbooks/${this.object.id}/`,
{ variable: variables }).catch(err => {
this.$message.error(this.$tc('UpdateErrorMsg') + ' ' + err)
})
}
}
}

View File

@@ -1,78 +0,0 @@
<template>
<Dialog
v-if="iVisible"
:destroy-on-close="true"
:show-cancel="false"
:show-confirm="false"
:title="$tc('AddVariable')"
:visible.sync="iVisible"
width="800px"
>
<VariableCreateForm
:variable="variable"
@add="addVariable"
@edit="editVariable"
/>
</Dialog>
</template>
<script>
import Dialog from '@/components/Dialog'
import VariableCreateForm from '@/components/Apps/VariableCreateUpdateForm'
export default {
name: 'AddVariableDialog',
components: {
Dialog,
VariableCreateForm
},
props: {
visible: {
type: Boolean,
default: false
},
variable: {
type: Object,
default: () => ({})
},
variables: {
type: Array,
default: () => ([])
}
},
computed: {
iVisible: {
get() {
return this.visible
},
set(val) {
this.$emit('update:visible', val)
}
}
},
methods: {
addVariable(variable) {
const i = this.variables.findIndex(item => item.name === variable.name || item.var_name === variable.var_name)
if (i !== -1) {
this.variables.splice(i, 1)
}
this.variables.push(variable)
this.iVisible = false
},
editVariable(form) {
const i = this.variables.findIndex(item => item.var_name === this.variable.var_name)
this.variables.splice(i, 1, form)
const count = this.variables.filter(value => value.var_name === form.var_name || value.name === form.name).length
// 不允许有相同的变量名
if (count > 1) {
this.variables.splice(i, 1)
}
this.iVisible = false
}
}
}
</script>
<style scoped>
</style>

View File

@@ -1,66 +0,0 @@
<template>
<Dialog
v-if="iVisible"
:destroy-on-close="true"
:show-cancel="false"
:show-confirm="false"
:title="$tc('setVariable')"
:visible.sync="iVisible"
width="800px"
>
<VariableSetForm
:form-data="formData"
:query-param="queryParam"
@confirm="handleConfirm"
/>
</Dialog>
</template>
<script>
import Dialog from '@/components/Dialog'
import VariableSetForm from '@/components/Apps/VariableSetForm'
export default {
name: 'SetVariableDialog',
components: {
Dialog,
VariableSetForm
},
props: {
visible: {
type: Boolean,
default: false
},
formData: {
type: Array,
default: () => ([])
},
queryParam: {
type: String,
default: ''
}
},
data() {
return {}
},
computed: {
iVisible: {
get() {
return this.visible
},
set(val) {
this.$emit('update:visible', val)
}
}
},
methods: {
handleConfirm(variable) {
this.$emit('submit', variable)
}
}
}
</script>
<style scoped>
</style>

View File

@@ -1,183 +0,0 @@
<template>
<div>
<div class="variables el-data-table">
<el-table :data="variables" class="el-table--fit el-table--border">
<el-table-column show-overflow-tooltip :label="$tc('Name')" prop="name" />
<el-table-column show-overflow-tooltip :label="$tc('VariableName')" prop="var_name" />
<el-table-column show-overflow-tooltip :label="$tc('DefaultValue')" prop="default_value" />
<el-table-column
v-if="!disableEdit"
:label="$tc('Actions')"
align="center"
class-name="buttons"
fixed="right"
width="135"
>
<template v-slot="scope">
<el-button icon="el-icon-minus" size="mini" type="danger" @click="removeVariable(scope.row)" />
<el-button
:disabled="!!scope.row.template"
icon="el-icon-edit"
size="mini"
type="primary"
@click="onEditClick(scope.row)"
/>
</template>
</el-table-column>
</el-table>
<div v-if="!disableEdit" class="actions">
<el-button size="mini" type="primary" @click="onAddClick">
{{ $t('Add') }}
</el-button>
</div>
<AddVariableDialog
:variable="variable"
:variables="variables"
:visible.sync="addVariableDialogVisible"
/>
</div>
</div>
</template>
<script>
import AddVariableDialog from './AddVariableDialog'
export default {
name: 'Variable',
components: {
AddVariableDialog
},
props: {
value: {
type: [Array],
default: () => []
},
disableEdit: {
type: Boolean,
default: false
}
},
data() {
return {
variable: {},
initial: false,
addVariableDialogVisible: false
}
},
computed: {
variables: {
get() {
return this.value
},
set(val) {
this.$emit('update:value', val)
this.$emit('change', val)
}
}
},
watch: {
variables: {
handler(newVal, oldVal) {
if (oldVal === undefined) return
if (newVal.length > 0 || !this.initial) {
newVal.map((item) => {
item.default_value = item.text_default_value || item.select_default_value
})
this.$emit('input', newVal)
}
if (newVal) {
this.initial = true
}
},
immediate: true,
deep: true
}
},
methods: {
removeVariable(variable) {
this.variables = this.variables.filter((item) => {
if (variable.id && item.id) {
return item.id !== variable.id
} else if (variable.var_name && item.var_name) {
return item.var_name !== variable.var_name
} else {
return variable.name !== item.name
}
})
},
onEditClick(variable) {
this.variable = variable
setTimeout(() => {
this.addVariableDialogVisible = true
})
},
onAddClick() {
this.variable = null
setTimeout(() => {
this.addVariableDialogVisible = true
})
}
}
}
</script>
<style lang="scss" scoped>
.el-data-table ::v-deep .el-table {
.table {
margin-top: 15px;
}
.el-table__row {
&.selected-row {
background-color: #f5f7fa;
}
& > td {
line-height: 1.5;
padding: 6px 0;
font-size: 13px;
border-right: none;
* {
vertical-align: middle;
}
.el-checkbox {
vertical-align: super;
}
& > div > span {
text-overflow: ellipsis;
overflow: hidden;
white-space: nowrap;
}
}
}
.el-table__header > thead > tr > th {
padding: 6px 0;
background-color: #ffffff;
font-size: 13px;
line-height: 1.5;
border-right: none;
.cell {
white-space: nowrap !important;
overflow: hidden;
text-overflow: ellipsis;
&:hover {
border-right: 2px solid #EBEEF5;
}
}
}
}
.el-data-table ::v-deep .el-table .el-table__header > thead > tr .is-sortable {
padding: 5px 0;
.cell {
padding-top: 3px !important;
}
}
</style>

View File

@@ -11,6 +11,7 @@
<script>
import { GenericCreateUpdatePage } from '@/layout/components'
import AssetSelect from '@/components/Apps/AssetSelect'
import { getDayFuture } from '@/utils/time'
import AccountFormatter from './components/AccountFormatter'
import { AllAccount } from '../const'
import ProtocolsSelect from '@/components/Form/FormFields/AllOrSpec.vue'
@@ -31,6 +32,9 @@ export default {
}
return {
initial: {
is_active: true,
date_start: new Date().toISOString(),
date_expired: getDayFuture(25550, new Date()).toISOString(),
nodes: nodesInitial,
assets: assetsInitial,
accounts: [AllAccount]

View File

@@ -37,12 +37,7 @@ export default {
notShowBuiltinTree: true,
url: '/api/v1/perms/asset-permissions/',
nodeUrl: '/api/v1/perms/asset-permissions/',
treeUrl: '/api/v1/assets/nodes/children/tree/?assets=1',
edit: {
drag: {
isMove: false
}
}
treeUrl: '/api/v1/assets/nodes/children/tree/?assets=1'
},
tableConfig: {
url: '/api/v1/perms/asset-permissions/',

View File

@@ -18,12 +18,6 @@
style="margin-top: 15px"
type="info"
/>
<QuickActions
:title="$tc('BiometricFeatures')"
type="warning"
:actions="biometricFeaturesActions"
style="margin-top: 15px"
/>
<IBox :title="$tc('InformationModification')" fa="fa-edit">
<table>
<tr>
@@ -82,21 +76,6 @@ export default {
url: `/api/v1/users/profile/`,
showPasswordDialog: false,
currentEdit: '',
biometricFeaturesActions: [
{
title: this.$t('Face'),
attrs: {
type: 'primary',
label: this.$store.state.users.profile.is_face_code_set ? this.$t('Unbind') : this.$t('Bind')
},
callbacks: {
click: () => {
const next_url = this.$store.state.users.profile.is_face_code_set ? '/core/auth/profile/face/disable/' : '/core/auth/profile/face/enable/'
window.open(next_url, '_self')
}
}
}
],
authQuickActions: [
{
title: this.$t('WeComOAuth'),

View File

@@ -5,7 +5,6 @@
<script>
import GenericListPage from '@/layout/components/GenericListPage'
import { download } from '@/utils/common'
import store from '@/store'
export default {
components: {
@@ -37,24 +36,7 @@ export default {
return row.has_file
},
tip: ({ row }) => {
const ftpFileMaxStore = store.getters.publicSettings['FTP_FILE_MAX_STORE']
const downloadTip = this.$t('Download')
const fileNotStoredTip = this.$t('FTPFileNotStored')
const storageNotEnabledTip = this.$t('FTPStorageNotEnabled')
const unknownStorageStateTip = this.$t('FTPUnknownStorageState')
if (row.has_file) {
return downloadTip
}
if (ftpFileMaxStore === 0) {
return storageNotEnabledTip
} else if (ftpFileMaxStore > 0) {
return fileNotStoredTip
} else {
return unknownStorageStateTip
}
return row.has_file ? this.$t('Download') : this.$t('DownloadFTPFileTip')
},
callback: function({ row }) {
// 跳转下载页面

View File

@@ -32,21 +32,13 @@ export default {
]
}
}
},
columnsMeta: {
type: Object,
default: () => {}
},
columnsExclude: {
type: Array,
default: () => []
}
},
data() {
return {
tableConfig: {
url: this.url,
columnsExclude: ['terminal', ...this.columnsExclude],
columnsExclude: ['terminal'],
columnsShow: this.columnsShow,
columnsMeta: {
id: {
@@ -145,8 +137,7 @@ export default {
hasUpdate: false,
extraActions: this.extraActions
}
},
...this.columnsMeta
}
}
},
headerActions: {

View File

@@ -1,5 +1,5 @@
<template>
<BaseList :extra-actions="extraActions" :url="url" :columns-meta="columnsMeta" :columns-exclude="columnsExclude" />
<BaseList :extra-actions="extraActions" :url="url" />
</template>
<script>
@@ -42,13 +42,7 @@ export default {
download(`/api/v1/terminal/sessions/${row.id}/replay/download/`)
}
}
],
columnsExclude: ['has_command'],
columnsMeta: {
command_amount: {
label: this.$t('CommandsTotal')
}
}
]
}
},
methods: {

View File

@@ -1,5 +1,5 @@
<template>
<BaseList :extra-actions="extraActions" :url="url" :columns-meta="columnsMeta" :columns-exclude="columnsExclude" />
<BaseList :extra-actions="extraActions" :url="url" />
</template>
<script>
@@ -100,14 +100,12 @@ export default {
window.open(monitorUrl, '_blank', 'height=600, width=850, top=400, left=400, toolbar=no, menubar=no, scrollbars=no, location=no, status=no')
}
}
],
columnsExclude: ['has_command'],
columnsMeta: {
command_amount: {
label: this.$t('CommandsTotal')
}
}
]
}
}
}
</script>
<style scoped>
</style>

View File

@@ -23,9 +23,6 @@ export default {
'AUTH_PASSKEY', 'FIDO_SERVER_ID', 'FIDO_SERVER_NAME'
],
fieldsMeta: {
'FIDO_SERVER_ID': {
placeholder: 'js.example.org'
}
},
submitMethod() {
return 'patch'
@@ -33,7 +30,8 @@ export default {
}
}
},
methods: {}
methods: {
}
}
</script>

View File

@@ -12,9 +12,6 @@ import { GenericCreateUpdateForm } from '@/layout/components'
import IBox from '@/components/IBox/index.vue'
import { openTaskPage } from '@/utils/jms'
import store from '@/store'
import HashiCorpKV from './HCP.vue'
import AzureKV from './Azure.vue'
import AwsSM from './Aws.vue'
export default {
components: {
@@ -28,6 +25,24 @@ export default {
hasReset: false,
helpText: this.$t('VaultHelpText'),
moreButtons: [
{
title: this.$t('Test'),
loading: false,
disabled: !store.getters.publicSettings['VAULT_ENABLED'],
callback: function(value, form, btn) {
btn.loading = true
vm.$axios.post(
'/api/v1/settings/vault/testing/',
value
).then(res => {
vm.$message.success(res['msg'])
}).catch(() => {
vm.$log.error('err occur')
}).finally(() => {
btn.loading = false
})
}
},
{
title: this.$t('Sync'),
loading: false,
@@ -47,9 +62,17 @@ export default {
}
}
],
encryptedFields: ['VAULT_HCP_TOKEN'],
fields: [
[this.$t('Basic'), ['VAULT_ENABLED', 'VAULT_BACKEND', 'HISTORY_ACCOUNT_CLEAN_LIMIT']],
[this.$t('Provider'), ['HCP', 'AZURE', 'AWS']]
[this.$t('Backend'),
[
'VAULT_ENABLED',
'VAULT_HCP_HOST',
'VAULT_HCP_TOKEN',
'VAULT_HCP_MOUNT_POINT'
]
],
[this.$t('History'), ['HISTORY_ACCOUNT_CLEAN_LIMIT']]
],
fieldsMeta: {
HISTORY_ACCOUNT_CLEAN_LIMIT: {
@@ -60,32 +83,21 @@ export default {
VAULT_ENABLED: {
disabled: true
},
VAULT_BACKEND: {
label: this.$t('Provider'),
VAULT_HCP_HOST: {
hidden: (formValue) => {
return !formValue.VAULT_ENABLED || formValue['VAULT_BACKEND'] === 'local'
},
disabled: true
},
HCP: {
label: this.$t('HashicorpVault'),
component: HashiCorpKV,
hidden: (formValue) => {
return !formValue.VAULT_ENABLED || formValue['VAULT_BACKEND'] !== 'hcp'
return !formValue.VAULT_ENABLED
}
},
AZURE: {
label: this.$t('AzureKeyVault'),
component: AzureKV,
VAULT_HCP_TOKEN: {
hidden: (formValue) => {
return !formValue.VAULT_ENABLED || formValue['VAULT_BACKEND'] !== 'azure'
return !formValue.VAULT_ENABLED
}
},
AWS: {
label: this.$t('AmazonSecretsManager'),
component: AwsSM,
VAULT_HCP_MOUNT_POINT: {
helpText: this.$t('VaultHCPMountPoint'),
helpTextAsTip: true,
hidden: (formValue) => {
return !formValue.VAULT_ENABLED || formValue['VAULT_BACKEND'] !== 'aws'
return !formValue.VAULT_ENABLED
}
}
},

View File

@@ -1,60 +0,0 @@
<template>
<BaseKV :config="$data" :title="$tc('AmazonSecretsManager')" />
</template>
<script>
import BaseKV from './Base.vue'
export default {
name: 'AmazonSecretsManager',
components: {
BaseKV
},
data() {
const vm = this
return {
url: `/api/v1/settings/setting/?category=aws_sm`,
hasDetailInMsg: false,
visible: false,
moreButtons: [
{
title: this.$t('Test'),
loading: false,
callback: function(value, form, btn) {
btn.loading = true
vm.$axios.post(
'/api/v1/settings/vault/aws/testing/',
value
).then(res => {
vm.$message.success(res['msg'])
}).catch(() => {
vm.$log.error('err occur')
}).finally(() => { btn.loading = false })
}
}
],
encryptedFields: ['VAULT_AWS_ACCESS_SECRET_KEY'],
fields: [
[this.$t('AccountStorage'),
[
'VAULT_AWS_REGION_NAME',
'VAULT_AWS_ACCESS_KEY_ID',
'VAULT_AWS_ACCESS_SECRET_KEY'
]
]
],
fieldsMeta: {
},
submitMethod() {
return 'patch'
}
}
},
computed: {},
methods: {}
}
</script>
<style scoped>
</style>

View File

@@ -1,61 +0,0 @@
<template>
<BaseKV :config="$data" :title="$tc('AzureKeyVault')" />
</template>
<script>
import BaseKV from './Base.vue'
export default {
name: 'AzureKV',
components: {
BaseKV
},
data() {
const vm = this
return {
url: `/api/v1/settings/setting/?category=azure_kv`,
hasDetailInMsg: false,
visible: false,
moreButtons: [
{
title: this.$t('Test'),
loading: false,
callback: function(value, form, btn) {
btn.loading = true
vm.$axios.post(
'/api/v1/settings/vault/azure/testing/',
value
).then(res => {
vm.$message.success(res['msg'])
}).catch(() => {
vm.$log.error('err occur')
}).finally(() => { btn.loading = false })
}
}
],
encryptedFields: ['VAULT_AZURE_CLIENT_SECRET'],
fields: [
[this.$t('AccountStorage'),
[
'VAULT_AZURE_HOST',
'VAULT_AZURE_CLIENT_ID',
'VAULT_AZURE_CLIENT_SECRET',
'VAULT_AZURE_TENANT_ID'
]
]
],
fieldsMeta: {
},
submitMethod() {
return 'patch'
}
}
},
computed: {},
methods: {}
}
</script>
<style scoped>
</style>

View File

@@ -1,63 +0,0 @@
<template>
<div>
<el-button size="mini" type="primary" icon="el-icon-setting" @click="visible=true">{{ $t('Setting') }}</el-button>
<Dialog
v-if="visible"
:destroy-on-close="true"
:show-cancel="false"
:show-confirm="false"
:title="title"
:visible.sync="visible"
width="70%"
@confirm="onConfirm()"
v-on="$listeners"
>
<GenericCreateUpdateForm ref="form" v-bind="iConfig" @submitSuccess="submitSuccess" />
</Dialog>
</div>
</template>
<script>
import Dialog from '@/components/Dialog/index.vue'
import GenericCreateUpdateForm from '@/layout/components/GenericCreateUpdateForm/index.vue'
export default {
name: 'Base',
components: {
Dialog,
GenericCreateUpdateForm
},
props: {
title: {
type: String,
default: ''
},
config: {
type: Object,
default: () => ({})
}
},
data() {
return {
visible: false
}
},
computed: {
iConfig() {
return this.config
}
},
methods: {
onConfirm() {
},
submitSuccess(res) {
this.$emit('input', !!res[this.enableField])
this.visible = false
}
}
}
</script>
<style scoped>
</style>

View File

@@ -1,64 +0,0 @@
<template>
<BaseKV :config="$data" :title="$tc('HashicorpVault')" />
</template>
<script>
import BaseKV from './Base.vue'
export default {
name: 'HashiCorpKV',
components: {
BaseKV
},
data() {
const vm = this
return {
url: `/api/v1/settings/setting/?category=hcp`,
hasDetailInMsg: false,
visible: false,
moreButtons: [
{
title: this.$t('Test'),
loading: false,
callback: function(value, form, btn) {
btn.loading = true
vm.$axios.post(
'/api/v1/settings/vault/hcp/testing/',
value
).then(res => {
vm.$message.success(res['msg'])
}).catch(() => {
vm.$log.error('err occur')
}).finally(() => { btn.loading = false })
}
}
],
encryptedFields: ['VAULT_HCP_TOKEN'],
fields: [
[this.$t('AccountStorage'),
[
'VAULT_HCP_HOST',
'VAULT_HCP_TOKEN',
'VAULT_HCP_MOUNT_POINT'
]
]
],
fieldsMeta: {
VAULT_HCP_MOUNT_POINT: {
helpText: this.$t('VaultHCPMountPoint'),
helpTextAsTip: true
}
},
submitMethod() {
return 'patch'
}
}
},
computed: {},
methods: {}
}
</script>
<style scoped>
</style>

View File

@@ -9,7 +9,7 @@
<script>
import TabPage from '@/layout/components/TabPage/index.vue'
import Announcement from './Announcement.vue'
import Vault from './Vault/index.vue'
import Vault from './Vault.vue'
import Ticket from './Ticket.vue'
import Ops from './Ops.vue'
import Chat from './Chat.vue'

View File

@@ -114,8 +114,8 @@ export default {
}
return [
{
key: this.$t('SerialNumber'),
value: this.licenseData?.serial_no || ''
key: this.$t('SubscriptionID'),
value: this.licenseData.subscription_id
},
{
key: this.$t('Corporation'),
@@ -133,6 +133,10 @@ export default {
key: this.$t('Edition'),
value: this.licenseData.edition
},
{
key: this.$t('SerialNumber'),
value: this.licenseData?.serial_no || ''
},
{
key: this.$t('Comment'),
value: this.licenseData?.remark || ''

View File

@@ -48,6 +48,11 @@ export default {
EMAIL_HOST_PASSWORD: {
component: UpdateToken
},
EMAIL_HOST_USER: {
rules: [
rules.EmailCheck
]
},
EMAIL_CUSTOM_USER_CREATED_BODY: {
el: {
type: 'textarea',

View File

@@ -26,14 +26,11 @@
</span>
</template>
</el-table-column>
<el-table-column :label="$tc('Receivers')">
<template slot-scope="scope">
<el-popover trigger="hover" placement="top" popper-class="black-theme-popover">
<p v-for="item in scope.row.receivers" :key="item.name">{{ item.name }}</p>
<span v-if="!scope.row.children" slot="reference" class="name-wrapper">
{{ scope.row.receivers.map(item => item.name).join(', ') }}
</span>
</el-popover>
<el-table-column :label="$tc('Receivers')" show-overflow-tooltip>
<template v-slot="scope">
<span v-if="!scope.row.children">
{{ scope.row.receivers.map(item => item.name).join(', ') }}
</span>
</template>
</el-table-column>
<el-table-column :label="$tc('Actions')" width="200">
@@ -189,25 +186,4 @@ export default {
margin-bottom: 0;
width: 50%;
}
::v-deep .el-table .cell {
display: flex;
.name-wrapper {
display: inline-block;
max-height: 55px;
max-width: 200px;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
vertical-align: middle;
cursor: pointer;
}
}
::v-deep .black-theme-popover .el-popover__inner {
background-color: #000 !important;
color: #fff !important;
border-color: #000 !important;
}
</style>

View File

@@ -23,13 +23,13 @@ export default {
'SECURITY_WATERMARK_ENABLED',
'SECURITY_SESSION_SHARE',
'SESSION_EXPIRE_AT_BROWSER_CLOSE',
'VIEW_ASSET_ONLINE_SESSION_INFO',
'SECURITY_MAX_IDLE_TIME',
'SECURITY_MAX_SESSION_TIME'
]
]
],
fieldsMeta: {}
fieldsMeta: {
}
}
}
}

View File

@@ -84,14 +84,6 @@ const components_ee = [
{
'title': 'Xrdp',
'components': ['xrdp', 'jms_xrdp']
},
{
'title': 'Nec',
'components': ['nec', 'jms_nec']
},
{
'title': 'Facelive',
'components': ['facelive', 'jms_facelive']
}
]
all_components.push(...components_ce)

View File

@@ -23,7 +23,7 @@ export default {
[
this.$t('Port'),
[
'http_port', 'https_port', 'ssh_port', 'rdp_port','vnc_port'
'http_port', 'https_port', 'ssh_port', 'rdp_port'
]
],
[
@@ -33,7 +33,7 @@ export default {
'redis_port', 'sqlserver_port', 'oracle_port_range'
]
],
[this.$t('Other'), ['is_active', 'comment']]
[this.$t('Other'), ['comment']]
],
fieldsMeta: {
host: {
@@ -41,17 +41,13 @@ export default {
},
oracle_port_range: {
disabled: true
},
is_active: {
disabled: this.$route.params.id === '00000000-0000-0000-0000-000000000001'
}
},
hasDetailInMsg: false
}
},
computed: {},
created() {
},
created() {},
methods: {}
}
</script>

View File

@@ -23,9 +23,9 @@ export default {
min: ['name', 'actions'],
default: [
'name', 'host', 'actions',
'http_port', 'https_port', 'ssh_port', 'rdp_port', 'vnc_port',
'http_port', 'https_port', 'ssh_port', 'rdp_port',
'mysql_port', 'mariadb_port', 'postgresql_port',
'redis_port', 'sqlserver_port', 'oracle_port_range', 'is_active'
'redis_port', 'sqlserver_port', 'oracle_port_range'
]
},
columnsMeta: {

View File

@@ -99,9 +99,7 @@ export default {
chen: 'Chen',
kael: 'Kael',
video_worker: 'Video-Worker',
panda: 'Panda',
nec: 'NEC',
facelive: 'Facelive'
panda: 'Panda'
}
return nameMapper[this.componentMetric.type]
},

View File

@@ -1,26 +1,12 @@
<script>
import Layout from '@/layout'
import store from '@/store'
import { mapGetters } from 'vuex'
import { setPreOrgLocal } from '@/utils/auth'
import { GLOBAL_ORG_ID } from '@/utils/org'
export default {
name: 'Setting',
extends: Layout,
computed: {
...mapGetters(['currentOrg'])
},
beforeRouteEnter(to, from, next) {
next(vm => {
if (vm.currentOrg.id === GLOBAL_ORG_ID) {
const currentUser = vm.$store.state.users.username
setPreOrgLocal(currentUser, vm.currentOrg)
}
store.dispatch('users/enterSettingOrg').then(() => next())
})
store.dispatch('users/enterSettingOrg').then(() => next())
},
beforeRouteLeave(to, from, next) {
store.dispatch('users/leaveSettingOrg').then(() => next())

View File

@@ -25,7 +25,7 @@
<el-form-item style="float: right">
<template v-if="hasActionPerm">
<el-button
:disabled="isDisabled || object.status.value === 'closed'"
:disabled="object.status.value === 'closed'"
size="small"
type="primary"
@click="handleApprove"
@@ -33,7 +33,7 @@
<i class="fa fa-check" /> {{ $t('Accept') }}
</el-button>
<el-button
:disabled="isDisabled || object.status.value === 'closed'"
:disabled="object.status.value === 'closed'"
size="small"
type="warning"
@click="handleReject"
@@ -43,7 +43,7 @@
</template>
<el-button
v-if="isSelfTicket"
:disabled="isDisabled || object.status.value === 'closed'"
:disabled="object.status.value === 'closed'"
size="small"
type="danger"
@click="handleClose"
@@ -92,7 +92,6 @@ export default {
},
data() {
return {
isDisabled: false,
comments: '',
type_api: '',
imageUrl: require('@/assets/img/avatar.png'),
@@ -155,35 +154,17 @@ export default {
this.createComment(function() {
})
const url = `/api/v1/tickets/${this.type_api}/${this.object.id}/approve/`
this.$axios.put(url).then(res => {
this.reloadPage()
}).catch(err => {
this.$message.error(err)
}).finally(() => {
this.isDisabled = false
})
this.$axios.put(url).then(res => this.reloadPage()).catch(err => this.$message.error(err))
},
defaultReject() {
this.createComment(function() {
})
const url = `/api/v1/tickets/${this.type_api}/${this.object.id}/reject/`
this.$axios.put(url).then(res => {
this.reloadPage()
}).catch(err => {
this.$message.error(err)
}).finally(() => {
this.isDisabled = false
})
this.$axios.put(url).then(res => this.reloadPage()).catch(err => this.$message.error(err))
},
defaultClose() {
const url = `/api/v1/tickets/${this.type_api}/${this.object.id}/close/`
this.$axios.put(url).then(res => {
this.reloadPage()
}).catch(err => {
this.$message.error(err)
}).finally(() => {
this.isDisabled = false
})
this.$axios.put(url).then(res => this.reloadPage()).catch(err => this.$message.error(err))
},
createComment(successCallback) {
const commentText = this.form.comments
@@ -204,42 +185,17 @@ export default {
}
})
},
handleAction(actionType) {
if (this.isDisabled) {
return
}
this.isDisabled = true
let handler
switch (actionType) {
case 'approve':
handler = this.approve || this.defaultApprove
break
case 'reject':
handler = this.reject || this.defaultReject
break
case 'close':
handler = this.close || this.defaultClose
break
default:
handler = null
break
}
if (handler) {
handler()
} else {
this.$message.error('No handler for action')
}
},
handleApprove() {
this.handleAction('approve')
const handler = this.approve || this.defaultApprove
handler()
},
handleReject() {
this.handleAction('reject')
const handler = this.reject || this.defaultReject
handler()
},
handleClose() {
this.handleAction('close')
const handler = this.close || this.defaultClose
handler()
},
handleComment() {
this.createComment(

Some files were not shown because too many files have changed in this diff Show More