mirror of
https://github.com/jumpserver/jumpserver.git
synced 2025-12-20 02:54:48 +00:00
Compare commits
31 Commits
rdp-patch-
...
v3.6.3
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
1e050560b1 | ||
|
|
305a426789 | ||
|
|
946a01f826 | ||
|
|
127a5d4157 | ||
|
|
18fb9a67ac | ||
|
|
352b2c2bd4 | ||
|
|
5b498650cb | ||
|
|
bd88e0af68 | ||
|
|
29fdeef45f | ||
|
|
ed5f4a227f | ||
|
|
5cb510a200 | ||
|
|
180cf354ad | ||
|
|
89a5c970e4 | ||
|
|
1d25cad449 | ||
|
|
c3b0798311 | ||
|
|
ff851b4672 | ||
|
|
2bcdcce2d3 | ||
|
|
f5ac941eb3 | ||
|
|
efcbfe63f9 | ||
|
|
41a2e00406 | ||
|
|
738b9efe11 | ||
|
|
119c7a8634 | ||
|
|
77e43c1c5c | ||
|
|
4d0231a9ad | ||
|
|
4562f1fbe8 | ||
|
|
4be70ff3da | ||
|
|
1a742d65f6 | ||
|
|
ba83b64d87 | ||
|
|
f46c9f56e8 | ||
|
|
626ec8f25d | ||
|
|
526c7de598 |
12
.github/ISSUE_TEMPLATE/----.md
vendored
Normal file
12
.github/ISSUE_TEMPLATE/----.md
vendored
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
---
|
||||||
|
name: 需求建议
|
||||||
|
about: 提出针对本项目的想法和建议
|
||||||
|
title: "[Feature] "
|
||||||
|
labels: 类型:需求
|
||||||
|
assignees:
|
||||||
|
- ibuler
|
||||||
|
- baijiangjie
|
||||||
|
- wojiushixiaobai
|
||||||
|
---
|
||||||
|
|
||||||
|
**请描述您的需求或者改进建议.**
|
||||||
72
.github/ISSUE_TEMPLATE/1_bug_report.yml
vendored
72
.github/ISSUE_TEMPLATE/1_bug_report.yml
vendored
@@ -1,72 +0,0 @@
|
|||||||
name: '🐛 Bug Report'
|
|
||||||
description: 'Report an Bug'
|
|
||||||
title: '[Bug] '
|
|
||||||
labels: ['🐛 Bug']
|
|
||||||
assignees:
|
|
||||||
- baijiangjie
|
|
||||||
body:
|
|
||||||
- type: input
|
|
||||||
attributes:
|
|
||||||
label: 'Product Version'
|
|
||||||
description: The versions prior to v2.28 (inclusive) are no longer supported.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: 'Product Edition'
|
|
||||||
options:
|
|
||||||
- label: 'Community Edition'
|
|
||||||
- label: 'Enterprise Edition'
|
|
||||||
- label: 'Enterprise Trial Edition'
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: 'Installation Method'
|
|
||||||
options:
|
|
||||||
- label: 'Online Installation (One-click command installation)'
|
|
||||||
- label: 'Offline Package Installation'
|
|
||||||
- label: 'All-in-One'
|
|
||||||
- label: '1Panel'
|
|
||||||
- label: 'Kubernetes'
|
|
||||||
- label: 'Source Code'
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Environment Information'
|
|
||||||
description: Please provide a clear and concise description outlining your environment information.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '🐛 Bug Description'
|
|
||||||
description:
|
|
||||||
Please provide a clear and concise description of the defect. If the issue is complex, please provide detailed explanations. <br/>
|
|
||||||
Unclear descriptions will not be processed. Please ensure you provide enough detail and information to support replicating and fixing the defect.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Recurrence Steps'
|
|
||||||
description: Please provide a clear and concise description outlining how to reproduce the issue.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Expected Behavior'
|
|
||||||
description: Please provide a clear and concise description of what you expect to happen.
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Additional Information'
|
|
||||||
description: Please add any additional background information about the issue here.
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Attempted Solutions'
|
|
||||||
description: If you have already attempted to solve the issue, please list the solutions you have tried here.
|
|
||||||
72
.github/ISSUE_TEMPLATE/1_bug_report_cn.yml
vendored
72
.github/ISSUE_TEMPLATE/1_bug_report_cn.yml
vendored
@@ -1,72 +0,0 @@
|
|||||||
name: '🐛 反馈缺陷'
|
|
||||||
description: '反馈一个缺陷'
|
|
||||||
title: '[Bug] '
|
|
||||||
labels: ['🐛 Bug']
|
|
||||||
assignees:
|
|
||||||
- baijiangjie
|
|
||||||
body:
|
|
||||||
- type: input
|
|
||||||
attributes:
|
|
||||||
label: '产品版本'
|
|
||||||
description: 不再支持 v2.28(含)之前的版本。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: '版本类型'
|
|
||||||
options:
|
|
||||||
- label: '社区版'
|
|
||||||
- label: '企业版'
|
|
||||||
- label: '企业试用版'
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: '安装方式'
|
|
||||||
options:
|
|
||||||
- label: '在线安装 (一键命令安装)'
|
|
||||||
- label: '离线包安装'
|
|
||||||
- label: 'All-in-One'
|
|
||||||
- label: '1Panel'
|
|
||||||
- label: 'Kubernetes'
|
|
||||||
- label: '源码安装'
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '环境信息'
|
|
||||||
description: 请提供一个清晰且简洁的描述,说明你的环境信息。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '🐛 缺陷描述'
|
|
||||||
description: |
|
|
||||||
请提供一个清晰且简洁的缺陷描述,如果问题比较复杂,也请详细说明。<br/>
|
|
||||||
针对不清晰的描述信息将不予处理。请确保提供足够的细节和信息,以支持对缺陷进行复现和修复。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '复现步骤'
|
|
||||||
description: 请提供一个清晰且简洁的描述,说明如何复现问题。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '期望结果'
|
|
||||||
description: 请提供一个清晰且简洁的描述,说明你期望发生什么。
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '补充信息'
|
|
||||||
description: 在这里添加关于问题的任何其他背景信息。
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '尝试过的解决方案'
|
|
||||||
description: 如果你已经尝试解决问题,请在此列出你尝试过的解决方案。
|
|
||||||
56
.github/ISSUE_TEMPLATE/2_feature_request.yml
vendored
56
.github/ISSUE_TEMPLATE/2_feature_request.yml
vendored
@@ -1,56 +0,0 @@
|
|||||||
name: '⭐️ Feature Request'
|
|
||||||
description: 'Suggest an idea'
|
|
||||||
title: '[Feature] '
|
|
||||||
labels: ['⭐️ Feature Request']
|
|
||||||
assignees:
|
|
||||||
- baijiangjie
|
|
||||||
- ibuler
|
|
||||||
body:
|
|
||||||
- type: input
|
|
||||||
attributes:
|
|
||||||
label: 'Product Version'
|
|
||||||
description: The versions prior to v2.28 (inclusive) are no longer supported.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: 'Product Edition'
|
|
||||||
options:
|
|
||||||
- label: 'Community Edition'
|
|
||||||
- label: 'Enterprise Edition'
|
|
||||||
- label: 'Enterprise Trial Edition'
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: 'Installation Method'
|
|
||||||
options:
|
|
||||||
- label: 'Online Installation (One-click command installation)'
|
|
||||||
- label: 'Offline Package Installation'
|
|
||||||
- label: 'All-in-One'
|
|
||||||
- label: '1Panel'
|
|
||||||
- label: 'Kubernetes'
|
|
||||||
- label: 'Source Code'
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '⭐️ Feature Description'
|
|
||||||
description: |
|
|
||||||
Please add a clear and concise description of the problem you aim to solve with this feature request.<br/>
|
|
||||||
Unclear descriptions will not be processed.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Proposed Solution'
|
|
||||||
description: Please provide a clear and concise description of the solution you desire.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Additional Information'
|
|
||||||
description: Please add any additional background information about the issue here.
|
|
||||||
56
.github/ISSUE_TEMPLATE/2_feature_request_cn.yml
vendored
56
.github/ISSUE_TEMPLATE/2_feature_request_cn.yml
vendored
@@ -1,56 +0,0 @@
|
|||||||
name: '⭐️ 功能需求'
|
|
||||||
description: '提出需求或建议'
|
|
||||||
title: '[Feature] '
|
|
||||||
labels: ['⭐️ Feature Request']
|
|
||||||
assignees:
|
|
||||||
- baijiangjie
|
|
||||||
- ibuler
|
|
||||||
body:
|
|
||||||
- type: input
|
|
||||||
attributes:
|
|
||||||
label: '产品版本'
|
|
||||||
description: 不再支持 v2.28(含)之前的版本。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: '版本类型'
|
|
||||||
options:
|
|
||||||
- label: '社区版'
|
|
||||||
- label: '企业版'
|
|
||||||
- label: '企业试用版'
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: '安装方式'
|
|
||||||
options:
|
|
||||||
- label: '在线安装 (一键命令安装)'
|
|
||||||
- label: '离线包安装'
|
|
||||||
- label: 'All-in-One'
|
|
||||||
- label: '1Panel'
|
|
||||||
- label: 'Kubernetes'
|
|
||||||
- label: '源码安装'
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '⭐️ 需求描述'
|
|
||||||
description: |
|
|
||||||
请添加一个清晰且简洁的问题描述,阐述你希望通过这个功能需求解决的问题。<br/>
|
|
||||||
针对不清晰的描述信息将不予处理。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '解决方案'
|
|
||||||
description: 请清晰且简洁地描述你想要的解决方案。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '补充信息'
|
|
||||||
description: 在这里添加关于问题的任何其他背景信息。
|
|
||||||
60
.github/ISSUE_TEMPLATE/3_question.yml
vendored
60
.github/ISSUE_TEMPLATE/3_question.yml
vendored
@@ -1,60 +0,0 @@
|
|||||||
name: '🤔 Question'
|
|
||||||
description: 'Pose a question'
|
|
||||||
title: '[Question] '
|
|
||||||
labels: ['🤔 Question']
|
|
||||||
assignees:
|
|
||||||
- baijiangjie
|
|
||||||
body:
|
|
||||||
- type: input
|
|
||||||
attributes:
|
|
||||||
label: 'Product Version'
|
|
||||||
description: The versions prior to v2.28 (inclusive) are no longer supported.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: 'Product Edition'
|
|
||||||
options:
|
|
||||||
- label: 'Community Edition'
|
|
||||||
- label: 'Enterprise Edition'
|
|
||||||
- label: 'Enterprise Trial Edition'
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: 'Installation Method'
|
|
||||||
options:
|
|
||||||
- label: 'Online Installation (One-click command installation)'
|
|
||||||
- label: 'Offline Package Installation'
|
|
||||||
- label: 'All-in-One'
|
|
||||||
- label: '1Panel'
|
|
||||||
- label: 'Kubernetes'
|
|
||||||
- label: 'Source Code'
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Environment Information'
|
|
||||||
description: Please provide a clear and concise description outlining your environment information.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '🤔 Question Description'
|
|
||||||
description: |
|
|
||||||
Please provide a clear and concise description of the defect. If the issue is complex, please provide detailed explanations. <br/>
|
|
||||||
Unclear descriptions will not be processed.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Expected Behavior'
|
|
||||||
description: Please provide a clear and concise description of what you expect to happen.
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: 'Additional Information'
|
|
||||||
description: Please add any additional background information about the issue here.
|
|
||||||
61
.github/ISSUE_TEMPLATE/3_question_cn.yml
vendored
61
.github/ISSUE_TEMPLATE/3_question_cn.yml
vendored
@@ -1,61 +0,0 @@
|
|||||||
name: '🤔 问题咨询'
|
|
||||||
description: '提出一个问题'
|
|
||||||
title: '[Question] '
|
|
||||||
labels: ['🤔 Question']
|
|
||||||
assignees:
|
|
||||||
- baijiangjie
|
|
||||||
body:
|
|
||||||
- type: input
|
|
||||||
attributes:
|
|
||||||
label: '产品版本'
|
|
||||||
description: 不再支持 v2.28(含)之前的版本。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: '版本类型'
|
|
||||||
options:
|
|
||||||
- label: '社区版'
|
|
||||||
- label: '企业版'
|
|
||||||
- label: '企业试用版'
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
attributes:
|
|
||||||
label: '安装方式'
|
|
||||||
options:
|
|
||||||
- label: '在线安装 (一键命令安装)'
|
|
||||||
- label: '离线包安装'
|
|
||||||
- label: 'All-in-One'
|
|
||||||
- label: '1Panel'
|
|
||||||
- label: 'Kubernetes'
|
|
||||||
- label: '源码安装'
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '环境信息'
|
|
||||||
description: 请在此详细描述你的环境信息,如操作系统、浏览器和部署架构等。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '🤔 问题描述'
|
|
||||||
description: |
|
|
||||||
请提供一个清晰且简洁的问题描述,如果问题比较复杂,也请详细说明。<br/>
|
|
||||||
针对不清晰的描述信息将不予处理。
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '期望结果'
|
|
||||||
description: 请提供一个清晰且简洁的描述,说明你期望发生什么。
|
|
||||||
|
|
||||||
- type: textarea
|
|
||||||
attributes:
|
|
||||||
label: '补充信息'
|
|
||||||
description: 在这里添加关于问题的任何其他背景信息。
|
|
||||||
|
|
||||||
24
.github/ISSUE_TEMPLATE/bug---.md
vendored
Normal file
24
.github/ISSUE_TEMPLATE/bug---.md
vendored
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
---
|
||||||
|
name: Bug 提交
|
||||||
|
about: 提交产品缺陷帮助我们更好的改进
|
||||||
|
title: "[Bug] "
|
||||||
|
labels: 类型:bug
|
||||||
|
assignees:
|
||||||
|
- wojiushixiaobai
|
||||||
|
- baijiangjie
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**JumpServer 版本( v2.28 之前的版本不再支持 )**
|
||||||
|
|
||||||
|
|
||||||
|
**浏览器版本**
|
||||||
|
|
||||||
|
|
||||||
|
**Bug 描述**
|
||||||
|
|
||||||
|
|
||||||
|
**Bug 重现步骤(有截图更好)**
|
||||||
|
1.
|
||||||
|
2.
|
||||||
|
3.
|
||||||
12
.github/ISSUE_TEMPLATE/question.md
vendored
Normal file
12
.github/ISSUE_TEMPLATE/question.md
vendored
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
---
|
||||||
|
name: 问题咨询
|
||||||
|
about: 提出针对本项目安装部署、使用及其他方面的相关问题
|
||||||
|
title: "[Question] "
|
||||||
|
labels: 类型:提问
|
||||||
|
assignees:
|
||||||
|
- wojiushixiaobai
|
||||||
|
- baijiangjie
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**请描述您的问题.**
|
||||||
4
.github/workflows/issue-close-require.yml
vendored
4
.github/workflows/issue-close-require.yml
vendored
@@ -12,9 +12,7 @@ jobs:
|
|||||||
uses: actions-cool/issues-helper@v2
|
uses: actions-cool/issues-helper@v2
|
||||||
with:
|
with:
|
||||||
actions: 'close-issues'
|
actions: 'close-issues'
|
||||||
labels: '⏳ Pending feedback'
|
labels: '状态:待反馈'
|
||||||
inactive-day: 30
|
inactive-day: 30
|
||||||
body: |
|
body: |
|
||||||
You haven't provided feedback for over 30 days.
|
|
||||||
We will close this issue. If you have any further needs, you can reopen it or submit a new issue.
|
|
||||||
您超过 30 天未反馈信息,我们将关闭该 issue,如有需求您可以重新打开或者提交新的 issue。
|
您超过 30 天未反馈信息,我们将关闭该 issue,如有需求您可以重新打开或者提交新的 issue。
|
||||||
|
|||||||
2
.github/workflows/issue-close.yml
vendored
2
.github/workflows/issue-close.yml
vendored
@@ -13,4 +13,4 @@ jobs:
|
|||||||
if: ${{ !github.event.issue.pull_request }}
|
if: ${{ !github.event.issue.pull_request }}
|
||||||
with:
|
with:
|
||||||
actions: 'remove-labels'
|
actions: 'remove-labels'
|
||||||
labels: '🔔 Pending processing,⏳ Pending feedback'
|
labels: '状态:待处理,状态:待反馈'
|
||||||
8
.github/workflows/issue-comment.yml
vendored
8
.github/workflows/issue-comment.yml
vendored
@@ -13,13 +13,13 @@ jobs:
|
|||||||
uses: actions-cool/issues-helper@v2
|
uses: actions-cool/issues-helper@v2
|
||||||
with:
|
with:
|
||||||
actions: 'add-labels'
|
actions: 'add-labels'
|
||||||
labels: '🔔 Pending processing'
|
labels: '状态:待处理'
|
||||||
|
|
||||||
- name: Remove require reply label
|
- name: Remove require reply label
|
||||||
uses: actions-cool/issues-helper@v2
|
uses: actions-cool/issues-helper@v2
|
||||||
with:
|
with:
|
||||||
actions: 'remove-labels'
|
actions: 'remove-labels'
|
||||||
labels: '⏳ Pending feedback'
|
labels: '状态:待反馈'
|
||||||
|
|
||||||
add-label-if-is-member:
|
add-label-if-is-member:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
@@ -55,11 +55,11 @@ jobs:
|
|||||||
uses: actions-cool/issues-helper@v2
|
uses: actions-cool/issues-helper@v2
|
||||||
with:
|
with:
|
||||||
actions: 'add-labels'
|
actions: 'add-labels'
|
||||||
labels: '⏳ Pending feedback'
|
labels: '状态:待反馈'
|
||||||
|
|
||||||
- name: Remove require handle label
|
- name: Remove require handle label
|
||||||
if: contains(steps.member_names.outputs.data, github.event.comment.user.login)
|
if: contains(steps.member_names.outputs.data, github.event.comment.user.login)
|
||||||
uses: actions-cool/issues-helper@v2
|
uses: actions-cool/issues-helper@v2
|
||||||
with:
|
with:
|
||||||
actions: 'remove-labels'
|
actions: 'remove-labels'
|
||||||
labels: '🔔 Pending processing'
|
labels: '状态:待处理'
|
||||||
|
|||||||
2
.github/workflows/issue-open.yml
vendored
2
.github/workflows/issue-open.yml
vendored
@@ -13,4 +13,4 @@ jobs:
|
|||||||
if: ${{ !github.event.issue.pull_request }}
|
if: ${{ !github.event.issue.pull_request }}
|
||||||
with:
|
with:
|
||||||
actions: 'add-labels'
|
actions: 'add-labels'
|
||||||
labels: '🔔 Pending processing'
|
labels: '状态:待处理'
|
||||||
47
.github/workflows/jms-build-test.yml
vendored
47
.github/workflows/jms-build-test.yml
vendored
@@ -1,32 +1,26 @@
|
|||||||
name: "Run Build Test"
|
name: "Run Build Test"
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
paths:
|
branches:
|
||||||
- 'Dockerfile'
|
- pr@*
|
||||||
- 'Dockerfile-*'
|
- repr@*
|
||||||
- 'pyproject.toml'
|
|
||||||
- 'poetry.lock'
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
- uses: docker/setup-qemu-action@v3
|
|
||||||
- uses: docker/setup-buildx-action@v3
|
|
||||||
|
|
||||||
- name: Check Dockerfile
|
- uses: docker/setup-qemu-action@v2
|
||||||
run: |
|
|
||||||
test -f Dockerfile-ce || cp -f Dockerfile Dockerfile-ce
|
|
||||||
|
|
||||||
- name: Build CE Image
|
- uses: docker/setup-buildx-action@v2
|
||||||
uses: docker/build-push-action@v5
|
|
||||||
|
- uses: docker/build-push-action@v3
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
push: false
|
push: false
|
||||||
file: Dockerfile-ce
|
tags: jumpserver/core:test
|
||||||
tags: jumpserver/core-ce:test
|
file: Dockerfile
|
||||||
platforms: linux/amd64
|
|
||||||
build-args: |
|
build-args: |
|
||||||
APT_MIRROR=http://deb.debian.org
|
APT_MIRROR=http://deb.debian.org
|
||||||
PIP_MIRROR=https://pypi.org/simple
|
PIP_MIRROR=https://pypi.org/simple
|
||||||
@@ -34,22 +28,9 @@ jobs:
|
|||||||
cache-from: type=gha
|
cache-from: type=gha
|
||||||
cache-to: type=gha,mode=max
|
cache-to: type=gha,mode=max
|
||||||
|
|
||||||
- name: Prepare EE Image
|
- uses: LouisBrunner/checks-action@v1.5.0
|
||||||
run: |
|
if: always()
|
||||||
sed -i 's@^FROM registry.fit2cloud.com@# FROM registry.fit2cloud.com@g' Dockerfile-ee
|
|
||||||
sed -i 's@^COPY --from=build-xpack@# COPY --from=build-xpack@g' Dockerfile-ee
|
|
||||||
|
|
||||||
- name: Build EE Image
|
|
||||||
uses: docker/build-push-action@v5
|
|
||||||
with:
|
with:
|
||||||
context: .
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
push: false
|
name: Check Build
|
||||||
file: Dockerfile-ee
|
conclusion: ${{ job.status }}
|
||||||
tags: jumpserver/core-ee:test
|
|
||||||
platforms: linux/amd64
|
|
||||||
build-args: |
|
|
||||||
APT_MIRROR=http://deb.debian.org
|
|
||||||
PIP_MIRROR=https://pypi.org/simple
|
|
||||||
PIP_JMS_MIRROR=https://pypi.org/simple
|
|
||||||
cache-from: type=gha
|
|
||||||
cache-to: type=gha,mode=max
|
|
||||||
|
|||||||
@@ -10,4 +10,3 @@ jobs:
|
|||||||
- uses: jumpserver/action-generic-handler@master
|
- uses: jumpserver/action-generic-handler@master
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.PRIVATE_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.PRIVATE_TOKEN }}
|
||||||
I18N_TOKEN: ${{ secrets.I18N_TOKEN }}
|
|
||||||
|
|||||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -43,4 +43,3 @@ releashe
|
|||||||
data/*
|
data/*
|
||||||
test.py
|
test.py
|
||||||
.history/
|
.history/
|
||||||
.test/
|
|
||||||
|
|||||||
105
Dockerfile
105
Dockerfile
@@ -1,4 +1,4 @@
|
|||||||
FROM python:3.11-slim-bullseye AS stage-1
|
FROM python:3.11-slim-bullseye as stage-build
|
||||||
ARG TARGETARCH
|
ARG TARGETARCH
|
||||||
|
|
||||||
ARG VERSION
|
ARG VERSION
|
||||||
@@ -6,10 +6,9 @@ ENV VERSION=$VERSION
|
|||||||
|
|
||||||
WORKDIR /opt/jumpserver
|
WORKDIR /opt/jumpserver
|
||||||
ADD . .
|
ADD . .
|
||||||
RUN echo > /opt/jumpserver/config.yml \
|
RUN cd utils && bash -ixeu build.sh
|
||||||
&& cd utils && bash -ixeu build.sh
|
|
||||||
|
|
||||||
FROM python:3.11-slim-bullseye as stage-2
|
FROM python:3.11-slim-bullseye
|
||||||
ARG TARGETARCH
|
ARG TARGETARCH
|
||||||
|
|
||||||
ARG BUILD_DEPENDENCIES=" \
|
ARG BUILD_DEPENDENCIES=" \
|
||||||
@@ -19,11 +18,11 @@ ARG BUILD_DEPENDENCIES=" \
|
|||||||
|
|
||||||
ARG DEPENDENCIES=" \
|
ARG DEPENDENCIES=" \
|
||||||
freetds-dev \
|
freetds-dev \
|
||||||
|
libpq-dev \
|
||||||
libffi-dev \
|
libffi-dev \
|
||||||
libjpeg-dev \
|
libjpeg-dev \
|
||||||
libkrb5-dev \
|
libkrb5-dev \
|
||||||
libldap2-dev \
|
libldap2-dev \
|
||||||
libpq-dev \
|
|
||||||
libsasl2-dev \
|
libsasl2-dev \
|
||||||
libssl-dev \
|
libssl-dev \
|
||||||
libxml2-dev \
|
libxml2-dev \
|
||||||
@@ -37,15 +36,17 @@ ARG TOOLS=" \
|
|||||||
curl \
|
curl \
|
||||||
default-libmysqlclient-dev \
|
default-libmysqlclient-dev \
|
||||||
default-mysql-client \
|
default-mysql-client \
|
||||||
git \
|
locales \
|
||||||
git-lfs \
|
nmap \
|
||||||
unzip \
|
openssh-client \
|
||||||
xz-utils \
|
sshpass \
|
||||||
|
telnet \
|
||||||
|
vim \
|
||||||
wget"
|
wget"
|
||||||
|
|
||||||
ARG APT_MIRROR=http://mirrors.ustc.edu.cn
|
ARG APT_MIRROR=http://mirrors.ustc.edu.cn
|
||||||
RUN --mount=type=cache,target=/var/cache/apt,sharing=locked,id=core-apt \
|
|
||||||
--mount=type=cache,target=/var/lib/apt,sharing=locked,id=core-apt \
|
RUN --mount=type=cache,target=/var/cache/apt,sharing=locked,id=core \
|
||||||
sed -i "s@http://.*.debian.org@${APT_MIRROR}@g" /etc/apt/sources.list \
|
sed -i "s@http://.*.debian.org@${APT_MIRROR}@g" /etc/apt/sources.list \
|
||||||
&& rm -f /etc/apt/apt.conf.d/docker-clean \
|
&& rm -f /etc/apt/apt.conf.d/docker-clean \
|
||||||
&& ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
|
&& ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
|
||||||
@@ -53,84 +54,30 @@ RUN --mount=type=cache,target=/var/cache/apt,sharing=locked,id=core-apt \
|
|||||||
&& apt-get -y install --no-install-recommends ${BUILD_DEPENDENCIES} \
|
&& apt-get -y install --no-install-recommends ${BUILD_DEPENDENCIES} \
|
||||||
&& apt-get -y install --no-install-recommends ${DEPENDENCIES} \
|
&& apt-get -y install --no-install-recommends ${DEPENDENCIES} \
|
||||||
&& apt-get -y install --no-install-recommends ${TOOLS} \
|
&& apt-get -y install --no-install-recommends ${TOOLS} \
|
||||||
&& echo "no" | dpkg-reconfigure dash
|
&& mkdir -p /root/.ssh/ \
|
||||||
|
&& echo "Host *\n\tStrictHostKeyChecking no\n\tUserKnownHostsFile /dev/null\n\tCiphers +aes128-cbc\n\tKexAlgorithms +diffie-hellman-group1-sha1\n\tHostKeyAlgorithms +ssh-rsa" > /root/.ssh/config \
|
||||||
|
&& echo "set mouse-=a" > ~/.vimrc \
|
||||||
|
&& echo "no" | dpkg-reconfigure dash \
|
||||||
|
&& echo "zh_CN.UTF-8" | dpkg-reconfigure locales \
|
||||||
|
&& sed -i "s@# export @export @g" ~/.bashrc \
|
||||||
|
&& sed -i "s@# alias @alias @g" ~/.bashrc \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
COPY --from=stage-build /opt/jumpserver/release/jumpserver /opt/jumpserver
|
||||||
WORKDIR /opt/jumpserver
|
WORKDIR /opt/jumpserver
|
||||||
|
|
||||||
ARG PIP_MIRROR=https://pypi.tuna.tsinghua.edu.cn/simple
|
ARG PIP_MIRROR=https://pypi.tuna.tsinghua.edu.cn/simple
|
||||||
RUN --mount=type=cache,target=/root/.cache \
|
RUN --mount=type=cache,target=/root/.cache \
|
||||||
--mount=type=bind,source=poetry.lock,target=/opt/jumpserver/poetry.lock \
|
|
||||||
--mount=type=bind,source=pyproject.toml,target=/opt/jumpserver/pyproject.toml \
|
|
||||||
set -ex \
|
set -ex \
|
||||||
&& python3 -m venv /opt/py3 \
|
&& echo > /opt/jumpserver/config.yml \
|
||||||
&& pip install poetry -i ${PIP_MIRROR} \
|
&& pip install poetry -i ${PIP_MIRROR} \
|
||||||
&& poetry config virtualenvs.create false \
|
&& poetry config virtualenvs.create false \
|
||||||
&& . /opt/py3/bin/activate \
|
&& poetry install --only=main
|
||||||
&& poetry install
|
|
||||||
|
|
||||||
FROM python:3.11-slim-bullseye
|
|
||||||
ARG TARGETARCH
|
|
||||||
ENV LANG=zh_CN.UTF-8 \
|
|
||||||
PATH=/opt/py3/bin:$PATH
|
|
||||||
|
|
||||||
ARG DEPENDENCIES=" \
|
|
||||||
libjpeg-dev \
|
|
||||||
libpq-dev \
|
|
||||||
libx11-dev \
|
|
||||||
freerdp2-dev \
|
|
||||||
libxmlsec1-openssl"
|
|
||||||
|
|
||||||
ARG TOOLS=" \
|
|
||||||
ca-certificates \
|
|
||||||
curl \
|
|
||||||
default-libmysqlclient-dev \
|
|
||||||
default-mysql-client \
|
|
||||||
iputils-ping \
|
|
||||||
locales \
|
|
||||||
netcat-openbsd \
|
|
||||||
nmap \
|
|
||||||
openssh-client \
|
|
||||||
patch \
|
|
||||||
sshpass \
|
|
||||||
telnet \
|
|
||||||
vim \
|
|
||||||
bubblewrap \
|
|
||||||
wget"
|
|
||||||
|
|
||||||
ARG APT_MIRROR=http://mirrors.ustc.edu.cn
|
|
||||||
RUN --mount=type=cache,target=/var/cache/apt,sharing=locked,id=core-apt \
|
|
||||||
--mount=type=cache,target=/var/lib/apt,sharing=locked,id=core-apt \
|
|
||||||
sed -i "s@http://.*.debian.org@${APT_MIRROR}@g" /etc/apt/sources.list \
|
|
||||||
&& rm -f /etc/apt/apt.conf.d/docker-clean \
|
|
||||||
&& ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
|
|
||||||
&& apt-get update \
|
|
||||||
&& apt-get -y install --no-install-recommends ${DEPENDENCIES} \
|
|
||||||
&& apt-get -y install --no-install-recommends ${TOOLS} \
|
|
||||||
&& mkdir -p /root/.ssh/ \
|
|
||||||
&& echo "Host *\n\tStrictHostKeyChecking no\n\tUserKnownHostsFile /dev/null\n\tCiphers +aes128-cbc\n\tKexAlgorithms +diffie-hellman-group1-sha1\n\tHostKeyAlgorithms +ssh-rsa" > /root/.ssh/config \
|
|
||||||
&& echo "no" | dpkg-reconfigure dash \
|
|
||||||
&& echo "zh_CN.UTF-8" | dpkg-reconfigure locales \
|
|
||||||
&& sed -i "s@# export @export @g" ~/.bashrc \
|
|
||||||
&& sed -i "s@# alias @alias @g" ~/.bashrc
|
|
||||||
|
|
||||||
ARG RECEPTOR_VERSION=v1.4.5
|
|
||||||
RUN set -ex \
|
|
||||||
&& wget -O /opt/receptor.tar.gz https://github.com/ansible/receptor/releases/download/${RECEPTOR_VERSION}/receptor_${RECEPTOR_VERSION/v/}_linux_${TARGETARCH}.tar.gz \
|
|
||||||
&& tar -xf /opt/receptor.tar.gz -C /usr/local/bin/ \
|
|
||||||
&& chown root:root /usr/local/bin/receptor \
|
|
||||||
&& chmod 755 /usr/local/bin/receptor \
|
|
||||||
&& rm -f /opt/receptor.tar.gz
|
|
||||||
|
|
||||||
COPY --from=stage-2 /opt/py3 /opt/py3
|
|
||||||
COPY --from=stage-1 /opt/jumpserver/release/jumpserver /opt/jumpserver
|
|
||||||
COPY --from=stage-1 /opt/jumpserver/release/jumpserver/apps/libs/ansible/ansible.cfg /etc/ansible/
|
|
||||||
|
|
||||||
WORKDIR /opt/jumpserver
|
|
||||||
|
|
||||||
ARG VERSION
|
|
||||||
ENV VERSION=$VERSION
|
|
||||||
|
|
||||||
VOLUME /opt/jumpserver/data
|
VOLUME /opt/jumpserver/data
|
||||||
|
VOLUME /opt/jumpserver/logs
|
||||||
|
|
||||||
|
ENV LANG=zh_CN.UTF-8
|
||||||
|
|
||||||
EXPOSE 8080
|
EXPOSE 8080
|
||||||
|
|
||||||
|
|||||||
137
Dockerfile-ce
137
Dockerfile-ce
@@ -1,137 +0,0 @@
|
|||||||
FROM python:3.11-slim-bullseye AS stage-1
|
|
||||||
ARG TARGETARCH
|
|
||||||
|
|
||||||
ARG VERSION
|
|
||||||
ENV VERSION=$VERSION
|
|
||||||
|
|
||||||
WORKDIR /opt/jumpserver
|
|
||||||
ADD . .
|
|
||||||
RUN echo > /opt/jumpserver/config.yml \
|
|
||||||
&& cd utils && bash -ixeu build.sh
|
|
||||||
|
|
||||||
FROM python:3.11-slim-bullseye as stage-2
|
|
||||||
ARG TARGETARCH
|
|
||||||
|
|
||||||
ARG BUILD_DEPENDENCIES=" \
|
|
||||||
g++ \
|
|
||||||
make \
|
|
||||||
pkg-config"
|
|
||||||
|
|
||||||
ARG DEPENDENCIES=" \
|
|
||||||
freetds-dev \
|
|
||||||
libffi-dev \
|
|
||||||
libjpeg-dev \
|
|
||||||
libkrb5-dev \
|
|
||||||
libldap2-dev \
|
|
||||||
libpq-dev \
|
|
||||||
libsasl2-dev \
|
|
||||||
libssl-dev \
|
|
||||||
libxml2-dev \
|
|
||||||
libxmlsec1-dev \
|
|
||||||
libxmlsec1-openssl \
|
|
||||||
freerdp2-dev \
|
|
||||||
libaio-dev"
|
|
||||||
|
|
||||||
ARG TOOLS=" \
|
|
||||||
ca-certificates \
|
|
||||||
curl \
|
|
||||||
default-libmysqlclient-dev \
|
|
||||||
default-mysql-client \
|
|
||||||
git \
|
|
||||||
git-lfs \
|
|
||||||
unzip \
|
|
||||||
xz-utils \
|
|
||||||
wget"
|
|
||||||
|
|
||||||
ARG APT_MIRROR=http://mirrors.ustc.edu.cn
|
|
||||||
RUN --mount=type=cache,target=/var/cache/apt,sharing=locked,id=core-apt \
|
|
||||||
--mount=type=cache,target=/var/lib/apt,sharing=locked,id=core-apt \
|
|
||||||
sed -i "s@http://.*.debian.org@${APT_MIRROR}@g" /etc/apt/sources.list \
|
|
||||||
&& rm -f /etc/apt/apt.conf.d/docker-clean \
|
|
||||||
&& ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
|
|
||||||
&& apt-get update \
|
|
||||||
&& apt-get -y install --no-install-recommends ${BUILD_DEPENDENCIES} \
|
|
||||||
&& apt-get -y install --no-install-recommends ${DEPENDENCIES} \
|
|
||||||
&& apt-get -y install --no-install-recommends ${TOOLS} \
|
|
||||||
&& echo "no" | dpkg-reconfigure dash
|
|
||||||
|
|
||||||
WORKDIR /opt/jumpserver
|
|
||||||
|
|
||||||
ARG PIP_MIRROR=https://pypi.tuna.tsinghua.edu.cn/simple
|
|
||||||
RUN --mount=type=cache,target=/root/.cache \
|
|
||||||
--mount=type=bind,source=poetry.lock,target=/opt/jumpserver/poetry.lock \
|
|
||||||
--mount=type=bind,source=pyproject.toml,target=/opt/jumpserver/pyproject.toml \
|
|
||||||
set -ex \
|
|
||||||
&& python3 -m venv /opt/py3 \
|
|
||||||
&& pip install poetry -i ${PIP_MIRROR} \
|
|
||||||
&& poetry config virtualenvs.create false \
|
|
||||||
&& . /opt/py3/bin/activate \
|
|
||||||
&& poetry install
|
|
||||||
|
|
||||||
FROM python:3.11-slim-bullseye
|
|
||||||
ARG TARGETARCH
|
|
||||||
ENV LANG=zh_CN.UTF-8 \
|
|
||||||
PATH=/opt/py3/bin:$PATH
|
|
||||||
|
|
||||||
ARG DEPENDENCIES=" \
|
|
||||||
libjpeg-dev \
|
|
||||||
libpq-dev \
|
|
||||||
libx11-dev \
|
|
||||||
freerdp2-dev \
|
|
||||||
libxmlsec1-openssl"
|
|
||||||
|
|
||||||
ARG TOOLS=" \
|
|
||||||
ca-certificates \
|
|
||||||
curl \
|
|
||||||
default-libmysqlclient-dev \
|
|
||||||
default-mysql-client \
|
|
||||||
iputils-ping \
|
|
||||||
locales \
|
|
||||||
netcat-openbsd \
|
|
||||||
nmap \
|
|
||||||
openssh-client \
|
|
||||||
patch \
|
|
||||||
sshpass \
|
|
||||||
telnet \
|
|
||||||
vim \
|
|
||||||
bubblewrap \
|
|
||||||
wget"
|
|
||||||
|
|
||||||
ARG APT_MIRROR=http://mirrors.ustc.edu.cn
|
|
||||||
RUN --mount=type=cache,target=/var/cache/apt,sharing=locked,id=core-apt \
|
|
||||||
--mount=type=cache,target=/var/lib/apt,sharing=locked,id=core-apt \
|
|
||||||
sed -i "s@http://.*.debian.org@${APT_MIRROR}@g" /etc/apt/sources.list \
|
|
||||||
&& rm -f /etc/apt/apt.conf.d/docker-clean \
|
|
||||||
&& ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
|
|
||||||
&& apt-get update \
|
|
||||||
&& apt-get -y install --no-install-recommends ${DEPENDENCIES} \
|
|
||||||
&& apt-get -y install --no-install-recommends ${TOOLS} \
|
|
||||||
&& mkdir -p /root/.ssh/ \
|
|
||||||
&& echo "Host *\n\tStrictHostKeyChecking no\n\tUserKnownHostsFile /dev/null\n\tCiphers +aes128-cbc\n\tKexAlgorithms +diffie-hellman-group1-sha1\n\tHostKeyAlgorithms +ssh-rsa" > /root/.ssh/config \
|
|
||||||
&& echo "no" | dpkg-reconfigure dash \
|
|
||||||
&& echo "zh_CN.UTF-8" | dpkg-reconfigure locales \
|
|
||||||
&& sed -i "s@# export @export @g" ~/.bashrc \
|
|
||||||
&& sed -i "s@# alias @alias @g" ~/.bashrc
|
|
||||||
|
|
||||||
ARG RECEPTOR_VERSION=v1.4.5
|
|
||||||
RUN set -ex \
|
|
||||||
&& wget -O /opt/receptor.tar.gz https://github.com/ansible/receptor/releases/download/${RECEPTOR_VERSION}/receptor_${RECEPTOR_VERSION/v/}_linux_${TARGETARCH}.tar.gz \
|
|
||||||
&& tar -xf /opt/receptor.tar.gz -C /usr/local/bin/ \
|
|
||||||
&& chown root:root /usr/local/bin/receptor \
|
|
||||||
&& chmod 755 /usr/local/bin/receptor \
|
|
||||||
&& rm -f /opt/receptor.tar.gz
|
|
||||||
|
|
||||||
COPY --from=stage-2 /opt/py3 /opt/py3
|
|
||||||
COPY --from=stage-1 /opt/jumpserver/release/jumpserver /opt/jumpserver
|
|
||||||
COPY --from=stage-1 /opt/jumpserver/release/jumpserver/apps/libs/ansible/ansible.cfg /etc/ansible/
|
|
||||||
|
|
||||||
WORKDIR /opt/jumpserver
|
|
||||||
|
|
||||||
ARG VERSION
|
|
||||||
ENV VERSION=$VERSION
|
|
||||||
|
|
||||||
VOLUME /opt/jumpserver/data
|
|
||||||
|
|
||||||
EXPOSE 8080
|
|
||||||
|
|
||||||
ENTRYPOINT ["./entrypoint.sh"]
|
|
||||||
@@ -1,5 +1,9 @@
|
|||||||
ARG VERSION
|
ARG VERSION
|
||||||
FROM registry.fit2cloud.com/jumpserver/xpack:${VERSION} AS build-xpack
|
FROM registry.fit2cloud.com/jumpserver/xpack:${VERSION} as build-xpack
|
||||||
FROM registry.fit2cloud.com/jumpserver/core-ce:${VERSION}
|
FROM jumpserver/core:${VERSION}
|
||||||
|
|
||||||
COPY --from=build-xpack /opt/xpack /opt/jumpserver/apps/xpack
|
COPY --from=build-xpack /opt/xpack /opt/jumpserver/apps/xpack
|
||||||
|
|
||||||
|
RUN --mount=type=cache,target=/root/.cache \
|
||||||
|
set -ex \
|
||||||
|
&& poetry install --only=xpack
|
||||||
10
README.md
10
README.md
@@ -12,6 +12,8 @@
|
|||||||
|
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
|
JumpServer <a href="https://github.com/jumpserver/jumpserver/releases/tag/v3.0.0">v3.0</a> 正式发布。
|
||||||
|
<br>
|
||||||
9 年时间,倾情投入,用心做好一款开源堡垒机。
|
9 年时间,倾情投入,用心做好一款开源堡垒机。
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
@@ -61,7 +63,6 @@ JumpServer 堡垒机帮助企业以更安全的方式管控和登录各种类型
|
|||||||
|
|
||||||
## 案例研究
|
## 案例研究
|
||||||
|
|
||||||
- [腾讯音乐娱乐集团:基于JumpServer的安全运维审计解决方案](https://blog.fit2cloud.com/?p=a04cdf0d-6704-4d18-9b40-9180baecd0e2)
|
|
||||||
- [腾讯海外游戏:基于JumpServer构建游戏安全运营能力](https://blog.fit2cloud.com/?p=3704)
|
- [腾讯海外游戏:基于JumpServer构建游戏安全运营能力](https://blog.fit2cloud.com/?p=3704)
|
||||||
- [万华化学:通过JumpServer管理全球化分布式IT资产,并且实现与云管平台的联动](https://blog.fit2cloud.com/?p=3504)
|
- [万华化学:通过JumpServer管理全球化分布式IT资产,并且实现与云管平台的联动](https://blog.fit2cloud.com/?p=3504)
|
||||||
- [雪花啤酒:JumpServer堡垒机使用体会](https://blog.fit2cloud.com/?p=3412)
|
- [雪花啤酒:JumpServer堡垒机使用体会](https://blog.fit2cloud.com/?p=3412)
|
||||||
@@ -94,12 +95,11 @@ JumpServer 堡垒机帮助企业以更安全的方式管控和登录各种类型
|
|||||||
| [KoKo](https://github.com/jumpserver/koko) | <a href="https://github.com/jumpserver/koko/releases"><img alt="Koko release" src="https://img.shields.io/github/release/jumpserver/koko.svg" /></a> | JumpServer 字符协议 Connector 项目 |
|
| [KoKo](https://github.com/jumpserver/koko) | <a href="https://github.com/jumpserver/koko/releases"><img alt="Koko release" src="https://img.shields.io/github/release/jumpserver/koko.svg" /></a> | JumpServer 字符协议 Connector 项目 |
|
||||||
| [Lion](https://github.com/jumpserver/lion-release) | <a href="https://github.com/jumpserver/lion-release/releases"><img alt="Lion release" src="https://img.shields.io/github/release/jumpserver/lion-release.svg" /></a> | JumpServer 图形协议 Connector 项目,依赖 [Apache Guacamole](https://guacamole.apache.org/) |
|
| [Lion](https://github.com/jumpserver/lion-release) | <a href="https://github.com/jumpserver/lion-release/releases"><img alt="Lion release" src="https://img.shields.io/github/release/jumpserver/lion-release.svg" /></a> | JumpServer 图形协议 Connector 项目,依赖 [Apache Guacamole](https://guacamole.apache.org/) |
|
||||||
| [Razor](https://github.com/jumpserver/razor) | <img alt="Chen" src="https://img.shields.io/badge/release-私有发布-red" /> | JumpServer RDP 代理 Connector 项目 |
|
| [Razor](https://github.com/jumpserver/razor) | <img alt="Chen" src="https://img.shields.io/badge/release-私有发布-red" /> | JumpServer RDP 代理 Connector 项目 |
|
||||||
| [Tinker](https://github.com/jumpserver/tinker) | <img alt="Tinker" src="https://img.shields.io/badge/release-私有发布-red" /> | JumpServer 远程应用 Connector 项目 (Windows) |
|
| [Tinker](https://github.com/jumpserver/tinker) | <img alt="Tinker" src="https://img.shields.io/badge/release-私有发布-red" /> | JumpServer 远程应用 Connector 项目 |
|
||||||
| [Panda](https://github.com/jumpserver/Panda) | <img alt="Panda" src="https://img.shields.io/badge/release-私有发布-red" /> | JumpServer 远程应用 Connector 项目 (Linux) |
|
|
||||||
| [Magnus](https://github.com/jumpserver/magnus-release) | <a href="https://github.com/jumpserver/magnus-release/releases"><img alt="Magnus release" src="https://img.shields.io/github/release/jumpserver/magnus-release.svg" /> | JumpServer 数据库代理 Connector 项目 |
|
| [Magnus](https://github.com/jumpserver/magnus-release) | <a href="https://github.com/jumpserver/magnus-release/releases"><img alt="Magnus release" src="https://img.shields.io/github/release/jumpserver/magnus-release.svg" /> | JumpServer 数据库代理 Connector 项目 |
|
||||||
| [Chen](https://github.com/jumpserver/chen-release) | <a href="https://github.com/jumpserver/chen-release/releases"><img alt="Chen release" src="https://img.shields.io/github/release/jumpserver/chen-release.svg" /> | JumpServer Web DB 项目,替代原来的 OmniDB |
|
| [Chen](https://github.com/jumpserver/chen-release) | <a href="https://github.com/jumpserver/chen-release/releases"><img alt="Chen release" src="https://img.shields.io/github/release/jumpserver/chen-release.svg" /> | JumpServer Web DB 项目,替代原来的 OmniDB |
|
||||||
| [Kael](https://github.com/jumpserver/kael) | <a href="https://github.com/jumpserver/kael/releases"><img alt="Kael release" src="https://img.shields.io/github/release/jumpserver/kael.svg" /> | JumpServer 连接 GPT 资产的组件项目 |
|
| [Kael](https://github.com/jumpserver/kael) | <a href="https://github.com/jumpserver/kael/releases"><img alt="Kael release" src="https://img.shields.io/github/release/jumpserver/kael.svg" /> | JumpServer 连接 GPT 资产的组件项目 |
|
||||||
| [Wisp](https://github.com/jumpserver/wisp) | <a href="https://github.com/jumpserver/wisp/releases"><img alt="Magnus release" src="https://img.shields.io/github/release/jumpserver/wisp.svg" /> | JumpServer 各系统终端组件和 Core API 通信的组件项目 |
|
| [Wisp](https://github.com/jumpserver/wisp) | <a href="https://github.com/jumpserver/wisp/releases"><img alt="Magnus release" src="https://img.shields.io/github/release/jumpserver/wisp.svg" /> | JumpServer 各系统终端组件和 Core Api 通信的组件项目 |
|
||||||
| [Clients](https://github.com/jumpserver/clients) | <a href="https://github.com/jumpserver/clients/releases"><img alt="Clients release" src="https://img.shields.io/github/release/jumpserver/clients.svg" /> | JumpServer 客户端 项目 |
|
| [Clients](https://github.com/jumpserver/clients) | <a href="https://github.com/jumpserver/clients/releases"><img alt="Clients release" src="https://img.shields.io/github/release/jumpserver/clients.svg" /> | JumpServer 客户端 项目 |
|
||||||
| [Installer](https://github.com/jumpserver/installer) | <a href="https://github.com/jumpserver/installer/releases"><img alt="Installer release" src="https://img.shields.io/github/release/jumpserver/installer.svg" /> | JumpServer 安装包 项目 |
|
| [Installer](https://github.com/jumpserver/installer) | <a href="https://github.com/jumpserver/installer/releases"><img alt="Installer release" src="https://img.shields.io/github/release/jumpserver/installer.svg" /> | JumpServer 安装包 项目 |
|
||||||
|
|
||||||
@@ -113,7 +113,7 @@ JumpServer是一款安全产品,请参考 [基本安全建议](https://docs.ju
|
|||||||
|
|
||||||
## License & Copyright
|
## License & Copyright
|
||||||
|
|
||||||
Copyright (c) 2014-2024 飞致云 FIT2CLOUD, All rights reserved.
|
Copyright (c) 2014-2023 飞致云 FIT2CLOUD, All rights reserved.
|
||||||
|
|
||||||
Licensed under The GNU General Public License version 3 (GPLv3) (the "License"); you may not use this file except in
|
Licensed under The GNU General Public License version 3 (GPLv3) (the "License"); you may not use this file except in
|
||||||
compliance with the License. You may obtain a copy of the License at
|
compliance with the License. You may obtain a copy of the License at
|
||||||
|
|||||||
@@ -85,7 +85,7 @@ If you find a security problem, please contact us directly:
|
|||||||
- 400-052-0755
|
- 400-052-0755
|
||||||
|
|
||||||
### License & Copyright
|
### License & Copyright
|
||||||
Copyright (c) 2014-2024 FIT2CLOUD Tech, Inc., All rights reserved.
|
Copyright (c) 2014-2022 FIT2CLOUD Tech, Inc., All rights reserved.
|
||||||
|
|
||||||
Licensed under The GNU General Public License version 3 (GPLv3) (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
|
Licensed under The GNU General Public License version 3 (GPLv3) (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
|
||||||
|
|
||||||
|
|||||||
@@ -6,12 +6,11 @@ from rest_framework.status import HTTP_200_OK
|
|||||||
|
|
||||||
from accounts import serializers
|
from accounts import serializers
|
||||||
from accounts.filters import AccountFilterSet
|
from accounts.filters import AccountFilterSet
|
||||||
from accounts.mixins import AccountRecordViewLogMixin
|
|
||||||
from accounts.models import Account
|
from accounts.models import Account
|
||||||
from assets.models import Asset, Node
|
from assets.models import Asset, Node
|
||||||
from authentication.permissions import UserConfirmation, ConfirmType
|
from common.api import ExtraFilterFieldsMixin
|
||||||
from common.api.mixin import ExtraFilterFieldsMixin
|
from common.permissions import UserConfirmation, ConfirmType, IsValidUser
|
||||||
from common.permissions import IsValidUser
|
from common.views.mixins import RecordViewLogMixin
|
||||||
from orgs.mixins.api import OrgBulkModelViewSet
|
from orgs.mixins.api import OrgBulkModelViewSet
|
||||||
from rbac.permissions import RBACPermission
|
from rbac.permissions import RBACPermission
|
||||||
|
|
||||||
@@ -58,19 +57,19 @@ class AccountViewSet(OrgBulkModelViewSet):
|
|||||||
permission_classes=[IsValidUser]
|
permission_classes=[IsValidUser]
|
||||||
)
|
)
|
||||||
def username_suggestions(self, request, *args, **kwargs):
|
def username_suggestions(self, request, *args, **kwargs):
|
||||||
asset_ids = request.data.get('assets', [])
|
asset_ids = request.data.get('assets')
|
||||||
node_ids = request.data.get('nodes', [])
|
node_ids = request.data.get('nodes')
|
||||||
username = request.data.get('username', '')
|
username = request.data.get('username')
|
||||||
|
|
||||||
accounts = Account.objects.all()
|
assets = Asset.objects.all()
|
||||||
|
if asset_ids:
|
||||||
|
assets = assets.filter(id__in=asset_ids)
|
||||||
if node_ids:
|
if node_ids:
|
||||||
nodes = Node.objects.filter(id__in=node_ids)
|
nodes = Node.objects.filter(id__in=node_ids)
|
||||||
node_asset_ids = Node.get_nodes_all_assets(*nodes).values_list('id', flat=True)
|
node_asset_ids = Node.get_nodes_all_assets(*nodes).values_list('id', flat=True)
|
||||||
asset_ids.extend(node_asset_ids)
|
assets = assets.filter(id__in=set(list(asset_ids) + list(node_asset_ids)))
|
||||||
|
|
||||||
if asset_ids:
|
|
||||||
accounts = accounts.filter(asset_id__in=list(set(asset_ids)))
|
|
||||||
|
|
||||||
|
accounts = Account.objects.filter(asset__in=assets)
|
||||||
if username:
|
if username:
|
||||||
accounts = accounts.filter(username__icontains=username)
|
accounts = accounts.filter(username__icontains=username)
|
||||||
usernames = list(accounts.values_list('username', flat=True).distinct()[:10])
|
usernames = list(accounts.values_list('username', flat=True).distinct()[:10])
|
||||||
@@ -87,7 +86,7 @@ class AccountViewSet(OrgBulkModelViewSet):
|
|||||||
return Response(status=HTTP_200_OK)
|
return Response(status=HTTP_200_OK)
|
||||||
|
|
||||||
|
|
||||||
class AccountSecretsViewSet(AccountRecordViewLogMixin, AccountViewSet):
|
class AccountSecretsViewSet(RecordViewLogMixin, AccountViewSet):
|
||||||
"""
|
"""
|
||||||
因为可能要导出所有账号,所以单独建立了一个 viewset
|
因为可能要导出所有账号,所以单独建立了一个 viewset
|
||||||
"""
|
"""
|
||||||
@@ -116,7 +115,7 @@ class AssetAccountBulkCreateApi(CreateAPIView):
|
|||||||
return Response(data=serializer.data, status=HTTP_200_OK)
|
return Response(data=serializer.data, status=HTTP_200_OK)
|
||||||
|
|
||||||
|
|
||||||
class AccountHistoriesSecretAPI(ExtraFilterFieldsMixin, AccountRecordViewLogMixin, ListAPIView):
|
class AccountHistoriesSecretAPI(ExtraFilterFieldsMixin, RecordViewLogMixin, ListAPIView):
|
||||||
model = Account.history.model
|
model = Account.history.model
|
||||||
serializer_class = serializers.AccountHistorySerializer
|
serializer_class = serializers.AccountHistorySerializer
|
||||||
http_method_names = ['get', 'options']
|
http_method_names = ['get', 'options']
|
||||||
@@ -144,3 +143,4 @@ class AccountHistoriesSecretAPI(ExtraFilterFieldsMixin, AccountRecordViewLogMixi
|
|||||||
return histories
|
return histories
|
||||||
histories = histories.exclude(history_id=latest_history.history_id)
|
histories = histories.exclude(history_id=latest_history.history_id)
|
||||||
return histories
|
return histories
|
||||||
|
|
||||||
|
|||||||
@@ -1,13 +1,9 @@
|
|||||||
from django.db.models import Q
|
|
||||||
from rest_framework.generics import CreateAPIView
|
from rest_framework.generics import CreateAPIView
|
||||||
|
from rest_framework.response import Response
|
||||||
|
|
||||||
from accounts import serializers
|
from accounts import serializers
|
||||||
from accounts.models import Account
|
from accounts.tasks import verify_accounts_connectivity_task, push_accounts_to_assets_task
|
||||||
from accounts.permissions import AccountTaskActionPermission
|
from assets.exceptions import NotSupportedTemporarilyError
|
||||||
from accounts.tasks import (
|
|
||||||
remove_accounts_task, verify_accounts_connectivity_task, push_accounts_to_assets_task
|
|
||||||
)
|
|
||||||
from authentication.permissions import UserConfirmation, ConfirmType
|
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
'AccountsTaskCreateAPI',
|
'AccountsTaskCreateAPI',
|
||||||
@@ -16,48 +12,38 @@ __all__ = [
|
|||||||
|
|
||||||
class AccountsTaskCreateAPI(CreateAPIView):
|
class AccountsTaskCreateAPI(CreateAPIView):
|
||||||
serializer_class = serializers.AccountTaskSerializer
|
serializer_class = serializers.AccountTaskSerializer
|
||||||
permission_classes = (AccountTaskActionPermission,)
|
|
||||||
|
|
||||||
def get_permissions(self):
|
def check_permissions(self, request):
|
||||||
act = self.request.data.get('action')
|
act = request.data.get('action')
|
||||||
if act == 'remove':
|
if act == 'push':
|
||||||
self.permission_classes = [
|
code = 'accounts.push_account'
|
||||||
AccountTaskActionPermission,
|
else:
|
||||||
UserConfirmation.require(ConfirmType.PASSWORD)
|
code = 'accounts.verify_account'
|
||||||
]
|
return request.user.has_perm(code)
|
||||||
return super().get_permissions()
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def get_account_ids(data, action):
|
|
||||||
account_type = 'gather_accounts' if action == 'remove' else 'accounts'
|
|
||||||
accounts = data.get(account_type, [])
|
|
||||||
account_ids = [str(a.id) for a in accounts]
|
|
||||||
|
|
||||||
if action == 'remove':
|
|
||||||
return account_ids
|
|
||||||
|
|
||||||
assets = data.get('assets', [])
|
|
||||||
asset_ids = [str(a.id) for a in assets]
|
|
||||||
ids = Account.objects.filter(
|
|
||||||
Q(id__in=account_ids) | Q(asset_id__in=asset_ids)
|
|
||||||
).distinct().values_list('id', flat=True)
|
|
||||||
return [str(_id) for _id in ids]
|
|
||||||
|
|
||||||
def perform_create(self, serializer):
|
def perform_create(self, serializer):
|
||||||
data = serializer.validated_data
|
data = serializer.validated_data
|
||||||
action = data['action']
|
accounts = data.get('accounts', [])
|
||||||
ids = self.get_account_ids(data, action)
|
params = data.get('params')
|
||||||
|
account_ids = [str(a.id) for a in accounts]
|
||||||
|
|
||||||
if action == 'push':
|
if data['action'] == 'push':
|
||||||
task = push_accounts_to_assets_task.delay(ids, data.get('params'))
|
task = push_accounts_to_assets_task.delay(account_ids, params)
|
||||||
elif action == 'remove':
|
|
||||||
task = remove_accounts_task.delay(ids)
|
|
||||||
elif action == 'verify':
|
|
||||||
task = verify_accounts_connectivity_task.delay(ids)
|
|
||||||
else:
|
else:
|
||||||
raise ValueError(f"Invalid action: {action}")
|
account = accounts[0]
|
||||||
|
asset = account.asset
|
||||||
|
if not asset.auto_config['ansible_enabled'] or \
|
||||||
|
not asset.auto_config['ping_enabled']:
|
||||||
|
raise NotSupportedTemporarilyError()
|
||||||
|
task = verify_accounts_connectivity_task.delay(account_ids)
|
||||||
|
|
||||||
data = getattr(serializer, '_data', {})
|
data = getattr(serializer, '_data', {})
|
||||||
data["task"] = task.id
|
data["task"] = task.id
|
||||||
setattr(serializer, '_data', data)
|
setattr(serializer, '_data', data)
|
||||||
return task
|
return task
|
||||||
|
|
||||||
|
def get_exception_handler(self):
|
||||||
|
def handler(e, context):
|
||||||
|
return Response({"error": str(e)}, status=400)
|
||||||
|
|
||||||
|
return handler
|
||||||
|
|||||||
@@ -1,15 +1,13 @@
|
|||||||
from django_filters import rest_framework as drf_filters
|
from django_filters import rest_framework as drf_filters
|
||||||
from rest_framework import status
|
|
||||||
from rest_framework.decorators import action
|
from rest_framework.decorators import action
|
||||||
from rest_framework.response import Response
|
from rest_framework.response import Response
|
||||||
|
|
||||||
from accounts import serializers
|
from accounts import serializers
|
||||||
from accounts.mixins import AccountRecordViewLogMixin
|
|
||||||
from accounts.models import AccountTemplate
|
from accounts.models import AccountTemplate
|
||||||
from accounts.tasks import template_sync_related_accounts
|
|
||||||
from assets.const import Protocol
|
from assets.const import Protocol
|
||||||
from authentication.permissions import UserConfirmation, ConfirmType
|
|
||||||
from common.drf.filters import BaseFilterSet
|
from common.drf.filters import BaseFilterSet
|
||||||
|
from common.permissions import UserConfirmation, ConfirmType
|
||||||
|
from common.views.mixins import RecordViewLogMixin
|
||||||
from orgs.mixins.api import OrgBulkModelViewSet
|
from orgs.mixins.api import OrgBulkModelViewSet
|
||||||
from rbac.permissions import RBACPermission
|
from rbac.permissions import RBACPermission
|
||||||
|
|
||||||
@@ -46,7 +44,6 @@ class AccountTemplateViewSet(OrgBulkModelViewSet):
|
|||||||
}
|
}
|
||||||
rbac_perms = {
|
rbac_perms = {
|
||||||
'su_from_account_templates': 'accounts.view_accounttemplate',
|
'su_from_account_templates': 'accounts.view_accounttemplate',
|
||||||
'sync_related_accounts': 'accounts.change_account',
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@action(methods=['get'], detail=False, url_path='su-from-account-templates')
|
@action(methods=['get'], detail=False, url_path='su-from-account-templates')
|
||||||
@@ -57,15 +54,8 @@ class AccountTemplateViewSet(OrgBulkModelViewSet):
|
|||||||
serializer = self.get_serializer(templates, many=True)
|
serializer = self.get_serializer(templates, many=True)
|
||||||
return Response(data=serializer.data)
|
return Response(data=serializer.data)
|
||||||
|
|
||||||
@action(methods=['patch'], detail=True, url_path='sync-related-accounts')
|
|
||||||
def sync_related_accounts(self, request, *args, **kwargs):
|
|
||||||
instance = self.get_object()
|
|
||||||
user_id = str(request.user.id)
|
|
||||||
task = template_sync_related_accounts.delay(str(instance.id), user_id)
|
|
||||||
return Response({'task': task.id}, status=status.HTTP_200_OK)
|
|
||||||
|
|
||||||
|
class AccountTemplateSecretsViewSet(RecordViewLogMixin, AccountTemplateViewSet):
|
||||||
class AccountTemplateSecretsViewSet(AccountRecordViewLogMixin, AccountTemplateViewSet):
|
|
||||||
serializer_classes = {
|
serializer_classes = {
|
||||||
'default': serializers.AccountTemplateSecretSerializer,
|
'default': serializers.AccountTemplateSecretSerializer,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,8 +18,9 @@ __all__ = [
|
|||||||
|
|
||||||
class AccountBackupPlanViewSet(OrgBulkModelViewSet):
|
class AccountBackupPlanViewSet(OrgBulkModelViewSet):
|
||||||
model = AccountBackupAutomation
|
model = AccountBackupAutomation
|
||||||
filterset_fields = ('name',)
|
filter_fields = ('name',)
|
||||||
search_fields = filterset_fields
|
search_fields = filter_fields
|
||||||
|
ordering = ('name',)
|
||||||
serializer_class = serializers.AccountBackupSerializer
|
serializer_class = serializers.AccountBackupSerializer
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -20,8 +20,8 @@ __all__ = [
|
|||||||
class AutomationAssetsListApi(generics.ListAPIView):
|
class AutomationAssetsListApi(generics.ListAPIView):
|
||||||
model = BaseAutomation
|
model = BaseAutomation
|
||||||
serializer_class = serializers.AutomationAssetsSerializer
|
serializer_class = serializers.AutomationAssetsSerializer
|
||||||
filterset_fields = ("name", "address")
|
filter_fields = ("name", "address")
|
||||||
search_fields = filterset_fields
|
search_fields = filter_fields
|
||||||
|
|
||||||
def get_object(self):
|
def get_object(self):
|
||||||
pk = self.kwargs.get('pk')
|
pk = self.kwargs.get('pk')
|
||||||
|
|||||||
@@ -1,17 +1,13 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
#
|
#
|
||||||
from rest_framework import status, mixins
|
|
||||||
from rest_framework.decorators import action
|
from rest_framework import mixins
|
||||||
from rest_framework.response import Response
|
|
||||||
|
|
||||||
from accounts import serializers
|
from accounts import serializers
|
||||||
from accounts.const import AutomationTypes
|
from accounts.const import AutomationTypes
|
||||||
from accounts.filters import ChangeSecretRecordFilterSet
|
from accounts.models import ChangeSecretAutomation, ChangeSecretRecord, AutomationExecution
|
||||||
from accounts.models import ChangeSecretAutomation, ChangeSecretRecord
|
from common.utils import get_object_or_none
|
||||||
from accounts.tasks import execute_automation_record_task
|
|
||||||
from authentication.permissions import UserConfirmation, ConfirmType
|
|
||||||
from orgs.mixins.api import OrgBulkModelViewSet, OrgGenericViewSet
|
from orgs.mixins.api import OrgBulkModelViewSet, OrgGenericViewSet
|
||||||
from rbac.permissions import RBACPermission
|
|
||||||
from .base import (
|
from .base import (
|
||||||
AutomationAssetsListApi, AutomationRemoveAssetApi, AutomationAddAssetApi,
|
AutomationAssetsListApi, AutomationRemoveAssetApi, AutomationAddAssetApi,
|
||||||
AutomationNodeAddRemoveApi, AutomationExecutionViewSet
|
AutomationNodeAddRemoveApi, AutomationExecutionViewSet
|
||||||
@@ -27,53 +23,28 @@ __all__ = [
|
|||||||
|
|
||||||
class ChangeSecretAutomationViewSet(OrgBulkModelViewSet):
|
class ChangeSecretAutomationViewSet(OrgBulkModelViewSet):
|
||||||
model = ChangeSecretAutomation
|
model = ChangeSecretAutomation
|
||||||
filterset_fields = ('name', 'secret_type', 'secret_strategy')
|
filter_fields = ('name', 'secret_type', 'secret_strategy')
|
||||||
search_fields = filterset_fields
|
search_fields = filter_fields
|
||||||
serializer_class = serializers.ChangeSecretAutomationSerializer
|
serializer_class = serializers.ChangeSecretAutomationSerializer
|
||||||
|
|
||||||
|
|
||||||
class ChangeSecretRecordViewSet(mixins.ListModelMixin, OrgGenericViewSet):
|
class ChangeSecretRecordViewSet(mixins.ListModelMixin, OrgGenericViewSet):
|
||||||
filterset_class = ChangeSecretRecordFilterSet
|
serializer_class = serializers.ChangeSecretRecordSerializer
|
||||||
search_fields = ('asset__address',)
|
filter_fields = ['asset', 'execution_id']
|
||||||
tp = AutomationTypes.change_secret
|
search_fields = ['asset__hostname']
|
||||||
serializer_classes = {
|
|
||||||
'default': serializers.ChangeSecretRecordSerializer,
|
|
||||||
'secret': serializers.ChangeSecretRecordViewSecretSerializer,
|
|
||||||
}
|
|
||||||
rbac_perms = {
|
|
||||||
'execute': 'accounts.add_changesecretexecution',
|
|
||||||
'secret': 'accounts.view_changesecretrecord',
|
|
||||||
}
|
|
||||||
|
|
||||||
def get_permissions(self):
|
|
||||||
if self.action == 'secret':
|
|
||||||
self.permission_classes = [
|
|
||||||
RBACPermission,
|
|
||||||
UserConfirmation.require(ConfirmType.MFA)
|
|
||||||
]
|
|
||||||
return super().get_permissions()
|
|
||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
return ChangeSecretRecord.objects.all()
|
return ChangeSecretRecord.objects.filter(
|
||||||
|
execution__automation__type=AutomationTypes.change_secret
|
||||||
@action(methods=['post'], detail=False, url_path='execute')
|
|
||||||
def execute(self, request, *args, **kwargs):
|
|
||||||
record_ids = request.data.get('record_ids')
|
|
||||||
records = self.get_queryset().filter(id__in=record_ids)
|
|
||||||
execution_count = records.values_list('execution_id', flat=True).distinct().count()
|
|
||||||
if execution_count != 1:
|
|
||||||
return Response(
|
|
||||||
{'detail': 'Only one execution is allowed to execute'},
|
|
||||||
status=status.HTTP_400_BAD_REQUEST
|
|
||||||
)
|
)
|
||||||
task = execute_automation_record_task.delay(record_ids, self.tp)
|
|
||||||
return Response({'task': task.id}, status=status.HTTP_200_OK)
|
|
||||||
|
|
||||||
@action(methods=['get'], detail=True, url_path='secret')
|
def filter_queryset(self, queryset):
|
||||||
def secret(self, request, *args, **kwargs):
|
queryset = super().filter_queryset(queryset)
|
||||||
instance = self.get_object()
|
eid = self.request.query_params.get('execution_id')
|
||||||
serializer = self.get_serializer(instance)
|
execution = get_object_or_none(AutomationExecution, pk=eid)
|
||||||
return Response(serializer.data)
|
if execution:
|
||||||
|
queryset = queryset.filter(execution=execution)
|
||||||
|
return queryset
|
||||||
|
|
||||||
|
|
||||||
class ChangSecretExecutionViewSet(AutomationExecutionViewSet):
|
class ChangSecretExecutionViewSet(AutomationExecutionViewSet):
|
||||||
|
|||||||
@@ -20,8 +20,8 @@ __all__ = [
|
|||||||
|
|
||||||
class GatherAccountsAutomationViewSet(OrgBulkModelViewSet):
|
class GatherAccountsAutomationViewSet(OrgBulkModelViewSet):
|
||||||
model = GatherAccountsAutomation
|
model = GatherAccountsAutomation
|
||||||
filterset_fields = ('name',)
|
filter_fields = ('name',)
|
||||||
search_fields = filterset_fields
|
search_fields = filter_fields
|
||||||
serializer_class = serializers.GatherAccountAutomationSerializer
|
serializer_class = serializers.GatherAccountAutomationSerializer
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -20,8 +20,8 @@ __all__ = [
|
|||||||
|
|
||||||
class PushAccountAutomationViewSet(OrgBulkModelViewSet):
|
class PushAccountAutomationViewSet(OrgBulkModelViewSet):
|
||||||
model = PushAccountAutomation
|
model = PushAccountAutomation
|
||||||
filterset_fields = ('name', 'secret_type', 'secret_strategy')
|
filter_fields = ('name', 'secret_type', 'secret_strategy')
|
||||||
search_fields = filterset_fields
|
search_fields = filter_fields
|
||||||
serializer_class = serializers.PushAccountAutomationSerializer
|
serializer_class = serializers.PushAccountAutomationSerializer
|
||||||
|
|
||||||
|
|
||||||
@@ -42,7 +42,6 @@ class PushAccountExecutionViewSet(AutomationExecutionViewSet):
|
|||||||
|
|
||||||
class PushAccountRecordViewSet(ChangeSecretRecordViewSet):
|
class PushAccountRecordViewSet(ChangeSecretRecordViewSet):
|
||||||
serializer_class = serializers.ChangeSecretRecordSerializer
|
serializer_class = serializers.ChangeSecretRecordSerializer
|
||||||
tp = AutomationTypes.push_account
|
|
||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
return ChangeSecretRecord.objects.filter(
|
return ChangeSecretRecord.objects.filter(
|
||||||
|
|||||||
@@ -3,26 +3,19 @@ import time
|
|||||||
from collections import defaultdict, OrderedDict
|
from collections import defaultdict, OrderedDict
|
||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
from openpyxl import Workbook
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
from xlsxwriter import Workbook
|
|
||||||
|
|
||||||
from accounts.const import AccountBackupType
|
from accounts.notifications import AccountBackupExecutionTaskMsg
|
||||||
from accounts.models.automations.backup_account import AccountBackupAutomation
|
|
||||||
from accounts.notifications import AccountBackupExecutionTaskMsg, AccountBackupByObjStorageExecutionTaskMsg
|
|
||||||
from accounts.serializers import AccountSecretSerializer
|
from accounts.serializers import AccountSecretSerializer
|
||||||
from assets.const import AllTypes
|
from assets.const import AllTypes
|
||||||
from common.utils.file import encrypt_and_compress_zip_file, zip_files
|
from common.utils.file import encrypt_and_compress_zip_file
|
||||||
from common.utils.timezone import local_now_filename, local_now_display
|
from common.utils.timezone import local_now_display
|
||||||
from terminal.models.component.storage import ReplayStorage
|
|
||||||
from users.models import User
|
from users.models import User
|
||||||
|
|
||||||
PATH = os.path.join(os.path.dirname(settings.BASE_DIR), 'tmp')
|
PATH = os.path.join(os.path.dirname(settings.BASE_DIR), 'tmp')
|
||||||
|
|
||||||
|
|
||||||
class RecipientsNotFound(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class BaseAccountHandler:
|
class BaseAccountHandler:
|
||||||
@classmethod
|
@classmethod
|
||||||
def unpack_data(cls, serializer_data, data=None):
|
def unpack_data(cls, serializer_data, data=None):
|
||||||
@@ -74,7 +67,7 @@ class AssetAccountHandler(BaseAccountHandler):
|
|||||||
@staticmethod
|
@staticmethod
|
||||||
def get_filename(plan_name):
|
def get_filename(plan_name):
|
||||||
filename = os.path.join(
|
filename = os.path.join(
|
||||||
PATH, f'{plan_name}-{local_now_filename()}-{time.time()}.xlsx'
|
PATH, f'{plan_name}-{local_now_display()}-{time.time()}.xlsx'
|
||||||
)
|
)
|
||||||
return filename
|
return filename
|
||||||
|
|
||||||
@@ -144,14 +137,13 @@ class AccountBackupHandler:
|
|||||||
|
|
||||||
wb = Workbook(filename)
|
wb = Workbook(filename)
|
||||||
for sheet, data in data_map.items():
|
for sheet, data in data_map.items():
|
||||||
ws = wb.add_worksheet(str(sheet))
|
ws = wb.create_sheet(str(sheet))
|
||||||
for row_index, row_data in enumerate(data):
|
for row in data:
|
||||||
for col_index, col_data in enumerate(row_data):
|
ws.append(row)
|
||||||
ws.write_string(row_index, col_index, col_data)
|
wb.save(filename)
|
||||||
wb.close()
|
|
||||||
files.append(filename)
|
files.append(filename)
|
||||||
timedelta = round((time.time() - time_start), 2)
|
timedelta = round((time.time() - time_start), 2)
|
||||||
print('创建备份文件完成: 用时 {}s'.format(timedelta))
|
print('步骤完成: 用时 {}s'.format(timedelta))
|
||||||
return files
|
return files
|
||||||
|
|
||||||
def send_backup_mail(self, files, recipients):
|
def send_backup_mail(self, files, recipients):
|
||||||
@@ -160,7 +152,7 @@ class AccountBackupHandler:
|
|||||||
recipients = User.objects.filter(id__in=list(recipients))
|
recipients = User.objects.filter(id__in=list(recipients))
|
||||||
print(
|
print(
|
||||||
'\n'
|
'\n'
|
||||||
'\033[32m>>> 开始发送备份邮件\033[0m'
|
'\033[32m>>> 发送备份邮件\033[0m'
|
||||||
''
|
''
|
||||||
)
|
)
|
||||||
plan_name = self.plan_name
|
plan_name = self.plan_name
|
||||||
@@ -168,42 +160,20 @@ class AccountBackupHandler:
|
|||||||
if not user.secret_key:
|
if not user.secret_key:
|
||||||
attachment_list = []
|
attachment_list = []
|
||||||
else:
|
else:
|
||||||
attachment = os.path.join(PATH, f'{plan_name}-{local_now_filename()}-{time.time()}.zip')
|
password = user.secret_key.encode('utf8')
|
||||||
encrypt_and_compress_zip_file(attachment, user.secret_key, files)
|
attachment = os.path.join(PATH, f'{plan_name}-{local_now_display()}-{time.time()}.zip')
|
||||||
|
encrypt_and_compress_zip_file(attachment, password, files)
|
||||||
attachment_list = [attachment, ]
|
attachment_list = [attachment, ]
|
||||||
AccountBackupExecutionTaskMsg(plan_name, user).publish(attachment_list)
|
AccountBackupExecutionTaskMsg(plan_name, user).publish(attachment_list)
|
||||||
print('邮件已发送至{}({})'.format(user, user.email))
|
print('邮件已发送至{}({})'.format(user, user.email))
|
||||||
for file in files:
|
for file in files:
|
||||||
os.remove(file)
|
os.remove(file)
|
||||||
|
|
||||||
def send_backup_obj_storage(self, files, recipients, password):
|
|
||||||
if not files:
|
|
||||||
return
|
|
||||||
recipients = ReplayStorage.objects.filter(id__in=list(recipients))
|
|
||||||
print(
|
|
||||||
'\n'
|
|
||||||
'\033[32m>>> 开始发送备份文件到sftp服务器\033[0m'
|
|
||||||
''
|
|
||||||
)
|
|
||||||
plan_name = self.plan_name
|
|
||||||
for rec in recipients:
|
|
||||||
attachment = os.path.join(PATH, f'{plan_name}-{local_now_filename()}-{time.time()}.zip')
|
|
||||||
if password:
|
|
||||||
print('\033[32m>>> 使用加密密码对文件进行加密中\033[0m')
|
|
||||||
encrypt_and_compress_zip_file(attachment, password, files)
|
|
||||||
else:
|
|
||||||
zip_files(attachment, files)
|
|
||||||
attachment_list = attachment
|
|
||||||
AccountBackupByObjStorageExecutionTaskMsg(plan_name, rec).publish(attachment_list)
|
|
||||||
print('备份文件将发送至{}({})'.format(rec.name, rec.id))
|
|
||||||
for file in files:
|
|
||||||
os.remove(file)
|
|
||||||
|
|
||||||
def step_perform_task_update(self, is_success, reason):
|
def step_perform_task_update(self, is_success, reason):
|
||||||
self.execution.reason = reason[:1024]
|
self.execution.reason = reason[:1024]
|
||||||
self.execution.is_success = is_success
|
self.execution.is_success = is_success
|
||||||
self.execution.save()
|
self.execution.save()
|
||||||
print('\n已完成对任务状态的更新\n')
|
print('已完成对任务状态的更新')
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def step_finished(is_success):
|
def step_finished(is_success):
|
||||||
@@ -216,11 +186,24 @@ class AccountBackupHandler:
|
|||||||
is_success = False
|
is_success = False
|
||||||
error = '-'
|
error = '-'
|
||||||
try:
|
try:
|
||||||
backup_type = self.execution.snapshot.get('backup_type', AccountBackupType.email.value)
|
recipients_part_one = self.execution.snapshot.get('recipients_part_one', [])
|
||||||
if backup_type == AccountBackupType.email.value:
|
recipients_part_two = self.execution.snapshot.get('recipients_part_two', [])
|
||||||
self.backup_by_email()
|
if not recipients_part_one and not recipients_part_two:
|
||||||
elif backup_type == AccountBackupType.object_storage.value:
|
print(
|
||||||
self.backup_by_obj_storage()
|
'\n'
|
||||||
|
'\033[32m>>> 该备份任务未分配收件人\033[0m'
|
||||||
|
''
|
||||||
|
)
|
||||||
|
if recipients_part_one and recipients_part_two:
|
||||||
|
files = self.create_excel(section='front')
|
||||||
|
self.send_backup_mail(files, recipients_part_one)
|
||||||
|
|
||||||
|
files = self.create_excel(section='back')
|
||||||
|
self.send_backup_mail(files, recipients_part_two)
|
||||||
|
else:
|
||||||
|
recipients = recipients_part_one or recipients_part_two
|
||||||
|
files = self.create_excel()
|
||||||
|
self.send_backup_mail(files, recipients)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.is_frozen = True
|
self.is_frozen = True
|
||||||
print('任务执行被异常中断')
|
print('任务执行被异常中断')
|
||||||
@@ -234,52 +217,6 @@ class AccountBackupHandler:
|
|||||||
self.step_perform_task_update(is_success, reason)
|
self.step_perform_task_update(is_success, reason)
|
||||||
self.step_finished(is_success)
|
self.step_finished(is_success)
|
||||||
|
|
||||||
def backup_by_obj_storage(self):
|
|
||||||
object_id = self.execution.snapshot.get('id')
|
|
||||||
zip_encrypt_password = AccountBackupAutomation.objects.get(id=object_id).zip_encrypt_password
|
|
||||||
obj_recipients_part_one = self.execution.snapshot.get('obj_recipients_part_one', [])
|
|
||||||
obj_recipients_part_two = self.execution.snapshot.get('obj_recipients_part_two', [])
|
|
||||||
if not obj_recipients_part_one and not obj_recipients_part_two:
|
|
||||||
print(
|
|
||||||
'\n'
|
|
||||||
'\033[31m>>> 该备份任务未分配sftp服务器\033[0m'
|
|
||||||
''
|
|
||||||
)
|
|
||||||
raise RecipientsNotFound('Not Found Recipients')
|
|
||||||
if obj_recipients_part_one and obj_recipients_part_two:
|
|
||||||
print('\033[32m>>> 账号的密钥将被拆分成前后两部分发送\033[0m')
|
|
||||||
files = self.create_excel(section='front')
|
|
||||||
self.send_backup_obj_storage(files, obj_recipients_part_one, zip_encrypt_password)
|
|
||||||
|
|
||||||
files = self.create_excel(section='back')
|
|
||||||
self.send_backup_obj_storage(files, obj_recipients_part_two, zip_encrypt_password)
|
|
||||||
else:
|
|
||||||
recipients = obj_recipients_part_one or obj_recipients_part_two
|
|
||||||
files = self.create_excel()
|
|
||||||
self.send_backup_obj_storage(files, recipients, zip_encrypt_password)
|
|
||||||
|
|
||||||
def backup_by_email(self):
|
|
||||||
recipients_part_one = self.execution.snapshot.get('recipients_part_one', [])
|
|
||||||
recipients_part_two = self.execution.snapshot.get('recipients_part_two', [])
|
|
||||||
if not recipients_part_one and not recipients_part_two:
|
|
||||||
print(
|
|
||||||
'\n'
|
|
||||||
'\033[31m>>> 该备份任务未分配收件人\033[0m'
|
|
||||||
''
|
|
||||||
)
|
|
||||||
raise RecipientsNotFound('Not Found Recipients')
|
|
||||||
if recipients_part_one and recipients_part_two:
|
|
||||||
print('\033[32m>>> 账号的密钥将被拆分成前后两部分发送\033[0m')
|
|
||||||
files = self.create_excel(section='front')
|
|
||||||
self.send_backup_mail(files, recipients_part_one)
|
|
||||||
|
|
||||||
files = self.create_excel(section='back')
|
|
||||||
self.send_backup_mail(files, recipients_part_two)
|
|
||||||
else:
|
|
||||||
recipients = recipients_part_one or recipients_part_two
|
|
||||||
files = self.create_excel()
|
|
||||||
self.send_backup_mail(files, recipients)
|
|
||||||
|
|
||||||
def run(self):
|
def run(self):
|
||||||
print('任务开始: {}'.format(local_now_display()))
|
print('任务开始: {}'.format(local_now_display()))
|
||||||
time_start = time.time()
|
time_start = time.time()
|
||||||
@@ -292,4 +229,4 @@ class AccountBackupHandler:
|
|||||||
finally:
|
finally:
|
||||||
print('\n任务结束: {}'.format(local_now_display()))
|
print('\n任务结束: {}'.format(local_now_display()))
|
||||||
timedelta = round((time.time() - time_start), 2)
|
timedelta = round((time.time() - time_start), 2)
|
||||||
print('用时: {}s'.format(timedelta))
|
print('用时: {}'.format(timedelta))
|
||||||
|
|||||||
@@ -13,15 +13,12 @@
|
|||||||
login_password: "{{ jms_account.secret }}"
|
login_password: "{{ jms_account.secret }}"
|
||||||
login_secret_type: "{{ jms_account.secret_type }}"
|
login_secret_type: "{{ jms_account.secret_type }}"
|
||||||
login_private_key_path: "{{ jms_account.private_key_path }}"
|
login_private_key_path: "{{ jms_account.private_key_path }}"
|
||||||
become: "{{ jms_custom_become | default(False) }}"
|
become: "{{ custom_become | default(False) }}"
|
||||||
become_method: "{{ jms_custom_become_method | default('su') }}"
|
become_method: "{{ custom_become_method | default('su') }}"
|
||||||
become_user: "{{ jms_custom_become_user | default('') }}"
|
become_user: "{{ custom_become_user | default('') }}"
|
||||||
become_password: "{{ jms_custom_become_password | default('') }}"
|
become_password: "{{ custom_become_password | default('') }}"
|
||||||
become_private_key_path: "{{ jms_custom_become_private_key_path | default(None) }}"
|
become_private_key_path: "{{ custom_become_private_key_path | default(None) }}"
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default(None) }}"
|
|
||||||
register: ping_info
|
register: ping_info
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: Change asset password (paramiko)
|
- name: Change asset password (paramiko)
|
||||||
custom_command:
|
custom_command:
|
||||||
@@ -31,11 +28,11 @@
|
|||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_secret_type: "{{ jms_account.secret_type }}"
|
login_secret_type: "{{ jms_account.secret_type }}"
|
||||||
login_private_key_path: "{{ jms_account.private_key_path }}"
|
login_private_key_path: "{{ jms_account.private_key_path }}"
|
||||||
become: "{{ jms_custom_become | default(False) }}"
|
become: "{{ custom_become | default(False) }}"
|
||||||
become_method: "{{ jms_custom_become_method | default('su') }}"
|
become_method: "{{ custom_become_method | default('su') }}"
|
||||||
become_user: "{{ jms_custom_become_user | default('') }}"
|
become_user: "{{ custom_become_user | default('') }}"
|
||||||
become_password: "{{ jms_custom_become_password | default('') }}"
|
become_password: "{{ custom_become_password | default('') }}"
|
||||||
become_private_key_path: "{{ jms_custom_become_private_key_path | default(None) }}"
|
become_private_key_path: "{{ custom_become_private_key_path | default(None) }}"
|
||||||
name: "{{ account.username }}"
|
name: "{{ account.username }}"
|
||||||
password: "{{ account.secret }}"
|
password: "{{ account.secret }}"
|
||||||
commands: "{{ params.commands }}"
|
commands: "{{ params.commands }}"
|
||||||
@@ -43,7 +40,6 @@
|
|||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
when: ping_info is succeeded
|
when: ping_info is succeeded
|
||||||
register: change_info
|
register: change_info
|
||||||
delegate_to: localhost
|
|
||||||
|
|
||||||
- name: Verify password (paramiko)
|
- name: Verify password (paramiko)
|
||||||
ssh_ping:
|
ssh_ping:
|
||||||
@@ -51,11 +47,4 @@
|
|||||||
login_password: "{{ account.secret }}"
|
login_password: "{{ account.secret }}"
|
||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
become: "{{ account.become.ansible_become | default(False) }}"
|
become: false
|
||||||
become_method: su
|
|
||||||
become_user: "{{ account.become.ansible_user | default('') }}"
|
|
||||||
become_password: "{{ account.become.ansible_password | default('') }}"
|
|
||||||
become_private_key_path: "{{ account.become.ansible_ssh_private_key_file | default(None) }}"
|
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default(None) }}"
|
|
||||||
delegate_to: localhost
|
|
||||||
|
|||||||
@@ -6,27 +6,15 @@ category:
|
|||||||
type:
|
type:
|
||||||
- all
|
- all
|
||||||
method: change_secret
|
method: change_secret
|
||||||
protocol: ssh
|
|
||||||
priority: 50
|
|
||||||
params:
|
params:
|
||||||
- name: commands
|
- name: commands
|
||||||
type: list
|
type: list
|
||||||
label: "{{ 'Params commands label' | trans }}"
|
label: '自定义命令'
|
||||||
default: [ '' ]
|
default: [ '' ]
|
||||||
help_text: "{{ 'Params commands help text' | trans }}"
|
help_text: '自定义命令中如需包含账号的 账号、密码、SSH 连接的用户密码 字段,<br />请使用 {username}、{password}、{login_password}格式,执行任务时会进行替换 。<br />比如针对 Cisco 主机进行改密,一般需要配置五条命令:<br />1. enable<br />2. {login_password}<br />3. configure terminal<br />4. username {username} privilege 0 password {password} <br />5. end'
|
||||||
|
|
||||||
i18n:
|
i18n:
|
||||||
SSH account change secret:
|
SSH account change secret:
|
||||||
zh: '使用 SSH 命令行自定义改密'
|
zh: 使用 SSH 命令行自定义改密
|
||||||
ja: 'SSH コマンドライン方式でカスタムパスワード変更'
|
ja: SSH コマンドライン方式でカスタムパスワード変更
|
||||||
en: 'Custom password change by SSH command line'
|
en: Custom password change by SSH command line
|
||||||
|
|
||||||
Params commands help text:
|
|
||||||
zh: '自定义命令中如需包含账号的 账号、密码、SSH 连接的用户密码 字段,<br />请使用 {username}、{password}、{login_password}格式,执行任务时会进行替换 。<br />比如针对 Cisco 主机进行改密,一般需要配置五条命令:<br />1. enable<br />2. {login_password}<br />3. configure terminal<br />4. username {username} privilege 0 password {password} <br />5. end'
|
|
||||||
ja: 'カスタム コマンドに SSH 接続用のアカウント番号、パスワード、ユーザー パスワード フィールドを含める必要がある場合は、<br />{ユーザー名}、{パスワード}、{login_password& を使用してください。 # 125; 形式。タスクの実行時に置き換えられます。 <br />たとえば、Cisco ホストのパスワードを変更するには、通常、次の 5 つのコマンドを設定する必要があります:<br />1.enable<br />2.{login_password}<br />3 .ターミナルの設定<br / >4. ユーザー名 {ユーザー名} 権限 0 パスワード {パスワード} <br />5. 終了'
|
|
||||||
en: 'If the custom command needs to include the account number, password, and user password field for SSH connection,<br />Please use {username}, {password}, {login_password&# 125; format, which will be replaced when executing the task. <br />For example, to change the password of a Cisco host, you generally need to configure five commands:<br />1. enable<br />2. {login_password}<br />3. configure terminal<br / >4. username {username} privilege 0 password {password} <br />5. end'
|
|
||||||
|
|
||||||
Params commands label:
|
|
||||||
zh: '自定义命令'
|
|
||||||
ja: 'カスタムコマンド'
|
|
||||||
en: 'Custom command'
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: mongodb
|
- hosts: mongodb
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test MongoDB connection
|
- name: Test MongoDB connection
|
||||||
@@ -11,9 +11,9 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
ssl: "{{ jms_asset.spec_info.use_ssl | default('') }}"
|
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
||||||
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert | default('') }}"
|
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert }}"
|
||||||
ssl_certfile: "{{ jms_asset.secret_info.client_key | default('') }}"
|
ssl_certfile: "{{ jms_asset.secret_info.client_key }}"
|
||||||
connection_options:
|
connection_options:
|
||||||
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
||||||
register: db_info
|
register: db_info
|
||||||
@@ -31,8 +31,8 @@
|
|||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
||||||
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert | default('') }}"
|
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert }}"
|
||||||
ssl_certfile: "{{ jms_asset.secret_info.client_key | default('') }}"
|
ssl_certfile: "{{ jms_asset.secret_info.client_key }}"
|
||||||
connection_options:
|
connection_options:
|
||||||
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
||||||
db: "{{ jms_asset.spec_info.db_name }}"
|
db: "{{ jms_asset.spec_info.db_name }}"
|
||||||
@@ -49,9 +49,7 @@
|
|||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
||||||
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert | default('') }}"
|
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert }}"
|
||||||
ssl_certfile: "{{ jms_asset.secret_info.client_key | default('') }}"
|
ssl_certfile: "{{ jms_asset.secret_info.client_key }}"
|
||||||
connection_options:
|
connection_options:
|
||||||
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
||||||
register: result
|
|
||||||
failed_when: not result.is_available
|
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
- hosts: mysql
|
- hosts: mysql
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
db_name: "{{ jms_asset.spec_info.db_name }}"
|
db_name: "{{ jms_asset.spec_info.db_name }}"
|
||||||
check_ssl: "{{ jms_asset.spec_info.use_ssl and not jms_asset.spec_info.allow_invalid_cert }}"
|
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test MySQL connection
|
- name: Test MySQL connection
|
||||||
@@ -12,10 +11,6 @@
|
|||||||
login_password: "{{ jms_account.secret }}"
|
login_password: "{{ jms_account.secret }}"
|
||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
check_hostname: "{{ check_ssl if check_ssl else omit }}"
|
|
||||||
ca_cert: "{{ jms_asset.secret_info.ca_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_cert: "{{ jms_asset.secret_info.client_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_key: "{{ jms_asset.secret_info.client_key | default(omit) if check_ssl else omit }}"
|
|
||||||
filter: version
|
filter: version
|
||||||
register: db_info
|
register: db_info
|
||||||
|
|
||||||
@@ -29,15 +24,10 @@
|
|||||||
login_password: "{{ jms_account.secret }}"
|
login_password: "{{ jms_account.secret }}"
|
||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
check_hostname: "{{ check_ssl if check_ssl else omit }}"
|
|
||||||
ca_cert: "{{ jms_asset.secret_info.ca_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_cert: "{{ jms_asset.secret_info.client_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_key: "{{ jms_asset.secret_info.client_key | default(omit) if check_ssl else omit }}"
|
|
||||||
name: "{{ account.username }}"
|
name: "{{ account.username }}"
|
||||||
password: "{{ account.secret }}"
|
password: "{{ account.secret }}"
|
||||||
host: "%"
|
host: "%"
|
||||||
priv: "{{ omit if db_name == '' else db_name + '.*:ALL' }}"
|
priv: "{{ account.username + '.*:USAGE' if db_name == '' else db_name + '.*:ALL' }}"
|
||||||
append_privs: "{{ db_name != '' | bool }}"
|
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
when: db_info is succeeded
|
when: db_info is succeeded
|
||||||
|
|
||||||
@@ -47,8 +37,4 @@
|
|||||||
login_password: "{{ account.secret }}"
|
login_password: "{{ account.secret }}"
|
||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
check_hostname: "{{ check_ssl if check_ssl else omit }}"
|
|
||||||
ca_cert: "{{ jms_asset.secret_info.ca_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_cert: "{{ jms_asset.secret_info.client_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_key: "{{ jms_asset.secret_info.client_key | default(omit) if check_ssl else omit }}"
|
|
||||||
filter: version
|
filter: version
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: oracle
|
- hosts: oracle
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test Oracle connection
|
- name: Test Oracle connection
|
||||||
@@ -39,4 +39,3 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
mode: "{{ account.mode }}"
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: postgre
|
- hosts: postgre
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test PostgreSQL connection
|
- name: Test PostgreSQL connection
|
||||||
@@ -39,5 +39,3 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
db: "{{ jms_asset.spec_info.db_name }}"
|
db: "{{ jms_asset.spec_info.db_name }}"
|
||||||
register: result
|
|
||||||
failed_when: not result.is_available
|
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: sqlserver
|
- hosts: sqlserver
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test SQLServer connection
|
- name: Test SQLServer connection
|
||||||
@@ -40,7 +40,7 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
name: '{{ jms_asset.spec_info.db_name }}'
|
name: '{{ jms_asset.spec_info.db_name }}'
|
||||||
script: "ALTER LOGIN {{ account.username }} WITH PASSWORD = '{{ account.secret }}', DEFAULT_DATABASE = {{ jms_asset.spec_info.db_name }}; select @@version"
|
script: "ALTER LOGIN {{ account.username }} WITH PASSWORD = '{{ account.secret }}'; select @@version"
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
when: user_exist.query_results[0] | length != 0
|
when: user_exist.query_results[0] | length != 0
|
||||||
|
|
||||||
@@ -51,7 +51,7 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
name: '{{ jms_asset.spec_info.db_name }}'
|
name: '{{ jms_asset.spec_info.db_name }}'
|
||||||
script: "CREATE LOGIN {{ account.username }} WITH PASSWORD = '{{ account.secret }}', DEFAULT_DATABASE = {{ jms_asset.spec_info.db_name }}; CREATE USER {{ account.username }} FOR LOGIN {{ account.username }}; select @@version"
|
script: "CREATE LOGIN {{ account.username }} WITH PASSWORD = '{{ account.secret }}'; select @@version"
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
when: user_exist.query_results[0] | length == 0
|
when: user_exist.query_results[0] | length == 0
|
||||||
|
|
||||||
|
|||||||
@@ -35,24 +35,12 @@
|
|||||||
- user_info.failed
|
- user_info.failed
|
||||||
- params.groups
|
- params.groups
|
||||||
|
|
||||||
- name: "Set {{ account.username }} sudo setting"
|
|
||||||
ansible.builtin.lineinfile:
|
|
||||||
dest: /etc/sudoers
|
|
||||||
state: present
|
|
||||||
regexp: "^{{ account.username }} ALL="
|
|
||||||
line: "{{ account.username + ' ALL=(ALL) NOPASSWD: ' + params.sudo }}"
|
|
||||||
validate: visudo -cf %s
|
|
||||||
when:
|
|
||||||
- user_info.failed or params.modify_sudo
|
|
||||||
- params.sudo
|
|
||||||
|
|
||||||
- name: "Change {{ account.username }} password"
|
- name: "Change {{ account.username }} password"
|
||||||
ansible.builtin.user:
|
ansible.builtin.user:
|
||||||
name: "{{ account.username }}"
|
name: "{{ account.username }}"
|
||||||
password: "{{ account.secret | password_hash('des') }}"
|
password: "{{ account.secret | password_hash('des') }}"
|
||||||
update_password: always
|
update_password: always
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
register: change_secret_result
|
|
||||||
when: account.secret_type == "password"
|
when: account.secret_type == "password"
|
||||||
|
|
||||||
- name: remove jumpserver ssh key
|
- name: remove jumpserver ssh key
|
||||||
@@ -69,9 +57,19 @@
|
|||||||
user: "{{ account.username }}"
|
user: "{{ account.username }}"
|
||||||
key: "{{ account.secret }}"
|
key: "{{ account.secret }}"
|
||||||
exclusive: "{{ ssh_params.exclusive }}"
|
exclusive: "{{ ssh_params.exclusive }}"
|
||||||
register: change_secret_result
|
|
||||||
when: account.secret_type == "ssh_key"
|
when: account.secret_type == "ssh_key"
|
||||||
|
|
||||||
|
- name: "Set {{ account.username }} sudo setting"
|
||||||
|
ansible.builtin.lineinfile:
|
||||||
|
dest: /etc/sudoers
|
||||||
|
state: present
|
||||||
|
regexp: "^{{ account.username }} ALL="
|
||||||
|
line: "{{ account.username + ' ALL=(ALL) NOPASSWD: ' + params.sudo }}"
|
||||||
|
validate: visudo -cf %s
|
||||||
|
when:
|
||||||
|
- user_info.failed
|
||||||
|
- params.sudo
|
||||||
|
|
||||||
- name: Refresh connection
|
- name: Refresh connection
|
||||||
ansible.builtin.meta: reset_connection
|
ansible.builtin.meta: reset_connection
|
||||||
|
|
||||||
@@ -82,15 +80,8 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
||||||
become: "{{ account.become.ansible_become | default(False) }}"
|
become: false
|
||||||
become_method: su
|
when: account.secret_type == "password"
|
||||||
become_user: "{{ account.become.ansible_user | default('') }}"
|
|
||||||
become_password: "{{ account.become.ansible_password | default('') }}"
|
|
||||||
become_private_key_path: "{{ account.become.ansible_ssh_private_key_file | default(None) }}"
|
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
|
||||||
when:
|
|
||||||
- account.secret_type == "password"
|
|
||||||
- check_conn_after_change or change_secret_result.failed | default(false)
|
|
||||||
delegate_to: localhost
|
delegate_to: localhost
|
||||||
|
|
||||||
- name: "Verify {{ account.username }} SSH KEY (paramiko)"
|
- name: "Verify {{ account.username }} SSH KEY (paramiko)"
|
||||||
@@ -100,8 +91,6 @@
|
|||||||
login_user: "{{ account.username }}"
|
login_user: "{{ account.username }}"
|
||||||
login_private_key_path: "{{ account.private_key_path }}"
|
login_private_key_path: "{{ account.private_key_path }}"
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
become: false
|
||||||
when:
|
when: account.secret_type == "ssh_key"
|
||||||
- account.secret_type == "ssh_key"
|
|
||||||
- check_conn_after_change or change_secret_result.failed | default(false)
|
|
||||||
delegate_to: localhost
|
delegate_to: localhost
|
||||||
|
|||||||
@@ -5,12 +5,6 @@ type:
|
|||||||
- AIX
|
- AIX
|
||||||
method: change_secret
|
method: change_secret
|
||||||
params:
|
params:
|
||||||
- name: modify_sudo
|
|
||||||
type: bool
|
|
||||||
label: "{{ 'Modify sudo label' | trans }}"
|
|
||||||
default: False
|
|
||||||
help_text: "{{ 'Modify params sudo help text' | trans }}"
|
|
||||||
|
|
||||||
- name: sudo
|
- name: sudo
|
||||||
type: str
|
type: str
|
||||||
label: 'Sudo'
|
label: 'Sudo'
|
||||||
@@ -40,11 +34,6 @@ i18n:
|
|||||||
ja: 'Ansible user モジュールを使用してアカウントのパスワード変更 (DES)'
|
ja: 'Ansible user モジュールを使用してアカウントのパスワード変更 (DES)'
|
||||||
en: 'Using Ansible module user to change account secret (DES)'
|
en: 'Using Ansible module user to change account secret (DES)'
|
||||||
|
|
||||||
Modify params sudo help text:
|
|
||||||
zh: '如果用户存在,可以修改sudo权限'
|
|
||||||
ja: 'ユーザーが存在する場合、sudo権限を変更できます'
|
|
||||||
en: 'If the user exists, sudo permissions can be modified'
|
|
||||||
|
|
||||||
Params sudo help text:
|
Params sudo help text:
|
||||||
zh: '使用逗号分隔多个命令,如: /bin/whoami,/sbin/ifconfig'
|
zh: '使用逗号分隔多个命令,如: /bin/whoami,/sbin/ifconfig'
|
||||||
ja: 'コンマで区切って複数のコマンドを入力してください。例: /bin/whoami,/sbin/ifconfig'
|
ja: 'コンマで区切って複数のコマンドを入力してください。例: /bin/whoami,/sbin/ifconfig'
|
||||||
@@ -60,11 +49,6 @@ i18n:
|
|||||||
ja: 'グループを入力してください。複数のグループはコンマで区切ってください(既存のグループを入力してください)'
|
ja: 'グループを入力してください。複数のグループはコンマで区切ってください(既存のグループを入力してください)'
|
||||||
en: 'Please enter the group. Multiple groups are separated by commas (please enter the existing group)'
|
en: 'Please enter the group. Multiple groups are separated by commas (please enter the existing group)'
|
||||||
|
|
||||||
Modify sudo label:
|
|
||||||
zh: '修改 sudo 权限'
|
|
||||||
ja: 'sudo 権限を変更'
|
|
||||||
en: 'Modify sudo'
|
|
||||||
|
|
||||||
Params home label:
|
Params home label:
|
||||||
zh: '家目录'
|
zh: '家目录'
|
||||||
ja: 'ホームディレクトリ'
|
ja: 'ホームディレクトリ'
|
||||||
|
|||||||
@@ -35,24 +35,12 @@
|
|||||||
- user_info.failed
|
- user_info.failed
|
||||||
- params.groups
|
- params.groups
|
||||||
|
|
||||||
- name: "Set {{ account.username }} sudo setting"
|
|
||||||
ansible.builtin.lineinfile:
|
|
||||||
dest: /etc/sudoers
|
|
||||||
state: present
|
|
||||||
regexp: "^{{ account.username }} ALL="
|
|
||||||
line: "{{ account.username + ' ALL=(ALL) NOPASSWD: ' + params.sudo }}"
|
|
||||||
validate: visudo -cf %s
|
|
||||||
when:
|
|
||||||
- user_info.failed or params.modify_sudo
|
|
||||||
- params.sudo
|
|
||||||
|
|
||||||
- name: "Change {{ account.username }} password"
|
- name: "Change {{ account.username }} password"
|
||||||
ansible.builtin.user:
|
ansible.builtin.user:
|
||||||
name: "{{ account.username }}"
|
name: "{{ account.username }}"
|
||||||
password: "{{ account.secret | password_hash('sha512') }}"
|
password: "{{ account.secret | password_hash('sha512') }}"
|
||||||
update_password: always
|
update_password: always
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
register: change_secret_result
|
|
||||||
when: account.secret_type == "password"
|
when: account.secret_type == "password"
|
||||||
|
|
||||||
- name: remove jumpserver ssh key
|
- name: remove jumpserver ssh key
|
||||||
@@ -69,9 +57,19 @@
|
|||||||
user: "{{ account.username }}"
|
user: "{{ account.username }}"
|
||||||
key: "{{ account.secret }}"
|
key: "{{ account.secret }}"
|
||||||
exclusive: "{{ ssh_params.exclusive }}"
|
exclusive: "{{ ssh_params.exclusive }}"
|
||||||
register: change_secret_result
|
|
||||||
when: account.secret_type == "ssh_key"
|
when: account.secret_type == "ssh_key"
|
||||||
|
|
||||||
|
- name: "Set {{ account.username }} sudo setting"
|
||||||
|
ansible.builtin.lineinfile:
|
||||||
|
dest: /etc/sudoers
|
||||||
|
state: present
|
||||||
|
regexp: "^{{ account.username }} ALL="
|
||||||
|
line: "{{ account.username + ' ALL=(ALL) NOPASSWD: ' + params.sudo }}"
|
||||||
|
validate: visudo -cf %s
|
||||||
|
when:
|
||||||
|
- user_info.failed
|
||||||
|
- params.sudo
|
||||||
|
|
||||||
- name: Refresh connection
|
- name: Refresh connection
|
||||||
ansible.builtin.meta: reset_connection
|
ansible.builtin.meta: reset_connection
|
||||||
|
|
||||||
@@ -82,15 +80,8 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
||||||
become: "{{ account.become.ansible_become | default(False) }}"
|
become: false
|
||||||
become_method: su
|
when: account.secret_type == "password"
|
||||||
become_user: "{{ account.become.ansible_user | default('') }}"
|
|
||||||
become_password: "{{ account.become.ansible_password | default('') }}"
|
|
||||||
become_private_key_path: "{{ account.become.ansible_ssh_private_key_file | default(None) }}"
|
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
|
||||||
when:
|
|
||||||
- account.secret_type == "password"
|
|
||||||
- check_conn_after_change or change_secret_result.failed | default(false)
|
|
||||||
delegate_to: localhost
|
delegate_to: localhost
|
||||||
|
|
||||||
- name: "Verify {{ account.username }} SSH KEY (paramiko)"
|
- name: "Verify {{ account.username }} SSH KEY (paramiko)"
|
||||||
@@ -100,8 +91,6 @@
|
|||||||
login_user: "{{ account.username }}"
|
login_user: "{{ account.username }}"
|
||||||
login_private_key_path: "{{ account.private_key_path }}"
|
login_private_key_path: "{{ account.private_key_path }}"
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
become: false
|
||||||
when:
|
when: account.secret_type == "ssh_key"
|
||||||
- account.secret_type == "ssh_key"
|
|
||||||
- check_conn_after_change or change_secret_result.failed | default(false)
|
|
||||||
delegate_to: localhost
|
delegate_to: localhost
|
||||||
|
|||||||
@@ -6,12 +6,6 @@ type:
|
|||||||
- linux
|
- linux
|
||||||
method: change_secret
|
method: change_secret
|
||||||
params:
|
params:
|
||||||
- name: modify_sudo
|
|
||||||
type: bool
|
|
||||||
label: "{{ 'Modify sudo label' | trans }}"
|
|
||||||
default: False
|
|
||||||
help_text: "{{ 'Modify params sudo help text' | trans }}"
|
|
||||||
|
|
||||||
- name: sudo
|
- name: sudo
|
||||||
type: str
|
type: str
|
||||||
label: 'Sudo'
|
label: 'Sudo'
|
||||||
@@ -42,11 +36,6 @@ i18n:
|
|||||||
ja: 'Ansible user モジュールを使用して アカウントのパスワード変更 (SHA512)'
|
ja: 'Ansible user モジュールを使用して アカウントのパスワード変更 (SHA512)'
|
||||||
en: 'Using Ansible module user to change account secret (SHA512)'
|
en: 'Using Ansible module user to change account secret (SHA512)'
|
||||||
|
|
||||||
Modify params sudo help text:
|
|
||||||
zh: '如果用户存在,可以修改sudo权限'
|
|
||||||
ja: 'ユーザーが存在する場合、sudo権限を変更できます'
|
|
||||||
en: 'If the user exists, sudo permissions can be modified'
|
|
||||||
|
|
||||||
Params sudo help text:
|
Params sudo help text:
|
||||||
zh: '使用逗号分隔多个命令,如: /bin/whoami,/sbin/ifconfig'
|
zh: '使用逗号分隔多个命令,如: /bin/whoami,/sbin/ifconfig'
|
||||||
ja: 'コンマで区切って複数のコマンドを入力してください。例: /bin/whoami,/sbin/ifconfig'
|
ja: 'コンマで区切って複数のコマンドを入力してください。例: /bin/whoami,/sbin/ifconfig'
|
||||||
@@ -62,11 +51,6 @@ i18n:
|
|||||||
ja: 'グループを入力してください。複数のグループはコンマで区切ってください(既存のグループを入力してください)'
|
ja: 'グループを入力してください。複数のグループはコンマで区切ってください(既存のグループを入力してください)'
|
||||||
en: 'Please enter the group. Multiple groups are separated by commas (please enter the existing group)'
|
en: 'Please enter the group. Multiple groups are separated by commas (please enter the existing group)'
|
||||||
|
|
||||||
Modify sudo label:
|
|
||||||
zh: '修改 sudo 权限'
|
|
||||||
ja: 'sudo 権限を変更'
|
|
||||||
en: 'Modify sudo'
|
|
||||||
|
|
||||||
Params home label:
|
Params home label:
|
||||||
zh: '家目录'
|
zh: '家目录'
|
||||||
ja: 'ホームディレクトリ'
|
ja: 'ホームディレクトリ'
|
||||||
|
|||||||
@@ -28,6 +28,4 @@
|
|||||||
vars:
|
vars:
|
||||||
ansible_user: "{{ account.username }}"
|
ansible_user: "{{ account.username }}"
|
||||||
ansible_password: "{{ account.secret }}"
|
ansible_password: "{{ account.secret }}"
|
||||||
when:
|
when: account.secret_type == "password"
|
||||||
- account.secret_type == "password"
|
|
||||||
- check_conn_after_change
|
|
||||||
|
|||||||
@@ -1,37 +0,0 @@
|
|||||||
- hosts: demo
|
|
||||||
gather_facts: no
|
|
||||||
tasks:
|
|
||||||
- name: Test privileged account
|
|
||||||
ansible.windows.win_ping:
|
|
||||||
|
|
||||||
# - name: Print variables
|
|
||||||
# debug:
|
|
||||||
# msg: "Username: {{ account.username }}, Password: {{ account.secret }}"
|
|
||||||
|
|
||||||
- name: Change password
|
|
||||||
ansible.windows.win_user:
|
|
||||||
fullname: "{{ account.username}}"
|
|
||||||
name: "{{ account.username }}"
|
|
||||||
password: "{{ account.secret }}"
|
|
||||||
password_never_expires: yes
|
|
||||||
groups: "{{ params.groups }}"
|
|
||||||
groups_action: add
|
|
||||||
update_password: always
|
|
||||||
ignore_errors: true
|
|
||||||
when: account.secret_type == "password"
|
|
||||||
|
|
||||||
- name: Refresh connection
|
|
||||||
ansible.builtin.meta: reset_connection
|
|
||||||
|
|
||||||
- name: Verify password (pyfreerdp)
|
|
||||||
rdp_ping:
|
|
||||||
login_host: "{{ jms_asset.address }}"
|
|
||||||
login_port: "{{ jms_asset.protocols | selectattr('name', 'equalto', 'rdp') | map(attribute='port') | first }}"
|
|
||||||
login_user: "{{ account.username }}"
|
|
||||||
login_password: "{{ account.secret }}"
|
|
||||||
login_secret_type: "{{ account.secret_type }}"
|
|
||||||
login_private_key_path: "{{ account.private_key_path }}"
|
|
||||||
when:
|
|
||||||
- account.secret_type == "password"
|
|
||||||
- check_conn_after_change
|
|
||||||
delegate_to: localhost
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
id: change_secret_windows_rdp_verify
|
|
||||||
name: "{{ 'Windows account change secret rdp verify' | trans }}"
|
|
||||||
version: 1
|
|
||||||
method: change_secret
|
|
||||||
category: host
|
|
||||||
type:
|
|
||||||
- windows
|
|
||||||
priority: 49
|
|
||||||
params:
|
|
||||||
- name: groups
|
|
||||||
type: str
|
|
||||||
label: '用户组'
|
|
||||||
default: 'Users,Remote Desktop Users'
|
|
||||||
help_text: "{{ 'Params groups help text' | trans }}"
|
|
||||||
|
|
||||||
|
|
||||||
i18n:
|
|
||||||
Windows account change secret rdp verify:
|
|
||||||
zh: '使用 Ansible 模块 win_user 执行 Windows 账号改密 RDP 协议测试最后的可连接性'
|
|
||||||
ja: 'Ansibleモジュールwin_userはWindowsアカウントの改密RDPプロトコルテストの最後の接続性を実行する'
|
|
||||||
en: 'Using the Ansible module win_user performs Windows account encryption RDP protocol testing for final connectivity'
|
|
||||||
|
|
||||||
Params groups help text:
|
|
||||||
zh: '请输入用户组,多个用户组使用逗号分隔(需填写已存在的用户组)'
|
|
||||||
ja: 'グループを入力してください。複数のグループはコンマで区切ってください(既存のグループを入力してください)'
|
|
||||||
en: 'Please enter the group. Multiple groups are separated by commas (please enter the existing group)'
|
|
||||||
|
|
||||||
@@ -1,20 +1,20 @@
|
|||||||
import os
|
import os
|
||||||
import time
|
import time
|
||||||
|
from collections import defaultdict
|
||||||
from copy import deepcopy
|
from copy import deepcopy
|
||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.translation import gettext_lazy as _
|
from openpyxl import Workbook
|
||||||
from xlsxwriter import Workbook
|
|
||||||
|
|
||||||
from accounts.const import AutomationTypes, SecretType, SSHKeyStrategy, SecretStrategy, ChangeSecretRecordStatusChoice
|
from accounts.const import AutomationTypes, SecretType, SSHKeyStrategy, SecretStrategy
|
||||||
from accounts.models import ChangeSecretRecord
|
from accounts.models import ChangeSecretRecord
|
||||||
from accounts.notifications import ChangeSecretExecutionTaskMsg, ChangeSecretFailedMsg
|
from accounts.notifications import ChangeSecretExecutionTaskMsg
|
||||||
from accounts.serializers import ChangeSecretRecordBackUpSerializer
|
from accounts.serializers import ChangeSecretRecordBackUpSerializer
|
||||||
from assets.const import HostTypes
|
from assets.const import HostTypes
|
||||||
from common.utils import get_logger
|
from common.utils import get_logger
|
||||||
from common.utils.file import encrypt_and_compress_zip_file
|
from common.utils.file import encrypt_and_compress_zip_file
|
||||||
from common.utils.timezone import local_now_filename
|
from common.utils.timezone import local_now_display
|
||||||
from users.models import User
|
from users.models import User
|
||||||
from ..base.manager import AccountBasePlaybookManager
|
from ..base.manager import AccountBasePlaybookManager
|
||||||
from ...utils import SecretGenerator
|
from ...utils import SecretGenerator
|
||||||
@@ -27,7 +27,7 @@ class ChangeSecretManager(AccountBasePlaybookManager):
|
|||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
self.record_map = self.execution.snapshot.get('record_map', {})
|
self.method_hosts_mapper = defaultdict(list)
|
||||||
self.secret_type = self.execution.snapshot.get('secret_type')
|
self.secret_type = self.execution.snapshot.get('secret_type')
|
||||||
self.secret_strategy = self.execution.snapshot.get(
|
self.secret_strategy = self.execution.snapshot.get(
|
||||||
'secret_strategy', SecretStrategy.custom
|
'secret_strategy', SecretStrategy.custom
|
||||||
@@ -50,9 +50,7 @@ class ChangeSecretManager(AccountBasePlaybookManager):
|
|||||||
kwargs['exclusive'] = 'yes' if kwargs['strategy'] == SSHKeyStrategy.set else 'no'
|
kwargs['exclusive'] = 'yes' if kwargs['strategy'] == SSHKeyStrategy.set else 'no'
|
||||||
|
|
||||||
if kwargs['strategy'] == SSHKeyStrategy.set_jms:
|
if kwargs['strategy'] == SSHKeyStrategy.set_jms:
|
||||||
username = account.username
|
kwargs['dest'] = '/home/{}/.ssh/authorized_keys'.format(account.username)
|
||||||
path = f'/{username}' if username == "root" else f'/home/{username}'
|
|
||||||
kwargs['dest'] = f'{path}/.ssh/authorized_keys'
|
|
||||||
kwargs['regexp'] = '.*{}$'.format(secret.split()[2].strip())
|
kwargs['regexp'] = '.*{}$'.format(secret.split()[2].strip())
|
||||||
return kwargs
|
return kwargs
|
||||||
|
|
||||||
@@ -98,13 +96,17 @@ class ChangeSecretManager(AccountBasePlaybookManager):
|
|||||||
|
|
||||||
accounts = self.get_accounts(account)
|
accounts = self.get_accounts(account)
|
||||||
if not accounts:
|
if not accounts:
|
||||||
print('没有发现待处理的账号: %s 用户ID: %s 类型: %s' % (
|
print('没有发现待改密账号: %s 用户ID: %s 类型: %s' % (
|
||||||
asset.name, self.account_ids, self.secret_type
|
asset.name, self.account_ids, self.secret_type
|
||||||
))
|
))
|
||||||
return []
|
return []
|
||||||
|
|
||||||
records = []
|
method_attr = getattr(automation, self.method_type() + '_method')
|
||||||
|
method_hosts = self.method_hosts_mapper[method_attr]
|
||||||
|
method_hosts = [h for h in method_hosts if h != host['name']]
|
||||||
inventory_hosts = []
|
inventory_hosts = []
|
||||||
|
records = []
|
||||||
|
|
||||||
if asset.type == HostTypes.WINDOWS and self.secret_type == SecretType.SSH_KEY:
|
if asset.type == HostTypes.WINDOWS and self.secret_type == SecretType.SSH_KEY:
|
||||||
print(f'Windows {asset} does not support ssh key push')
|
print(f'Windows {asset} does not support ssh key push')
|
||||||
return inventory_hosts
|
return inventory_hosts
|
||||||
@@ -114,31 +116,13 @@ class ChangeSecretManager(AccountBasePlaybookManager):
|
|||||||
h = deepcopy(host)
|
h = deepcopy(host)
|
||||||
secret_type = account.secret_type
|
secret_type = account.secret_type
|
||||||
h['name'] += '(' + account.username + ')'
|
h['name'] += '(' + account.username + ')'
|
||||||
if self.secret_type is None:
|
|
||||||
new_secret = account.secret
|
|
||||||
else:
|
|
||||||
new_secret = self.get_secret(secret_type)
|
new_secret = self.get_secret(secret_type)
|
||||||
|
|
||||||
if new_secret is None:
|
|
||||||
print(f'new_secret is None, account: {account}')
|
|
||||||
continue
|
|
||||||
|
|
||||||
asset_account_id = f'{asset.id}-{account.id}'
|
|
||||||
if asset_account_id not in self.record_map:
|
|
||||||
recorder = ChangeSecretRecord(
|
recorder = ChangeSecretRecord(
|
||||||
asset=asset, account=account, execution=self.execution,
|
asset=asset, account=account, execution=self.execution,
|
||||||
old_secret=account.secret, new_secret=new_secret,
|
old_secret=account.secret, new_secret=new_secret,
|
||||||
)
|
)
|
||||||
records.append(recorder)
|
records.append(recorder)
|
||||||
else:
|
|
||||||
record_id = self.record_map[asset_account_id]
|
|
||||||
try:
|
|
||||||
recorder = ChangeSecretRecord.objects.get(id=record_id)
|
|
||||||
new_secret = recorder.new_secret
|
|
||||||
except ChangeSecretRecord.DoesNotExist:
|
|
||||||
print(f"Record {record_id} not found")
|
|
||||||
continue
|
|
||||||
|
|
||||||
self.name_recorder_mapper[h['name']] = recorder
|
self.name_recorder_mapper[h['name']] = recorder
|
||||||
|
|
||||||
private_key_path = None
|
private_key_path = None
|
||||||
@@ -151,13 +135,14 @@ class ChangeSecretManager(AccountBasePlaybookManager):
|
|||||||
'name': account.name,
|
'name': account.name,
|
||||||
'username': account.username,
|
'username': account.username,
|
||||||
'secret_type': secret_type,
|
'secret_type': secret_type,
|
||||||
'secret': account.escape_jinja2_syntax(new_secret),
|
'secret': new_secret,
|
||||||
'private_key_path': private_key_path,
|
'private_key_path': private_key_path
|
||||||
'become': account.get_ansible_become_auth(),
|
|
||||||
}
|
}
|
||||||
if asset.platform.type == 'oracle':
|
if asset.platform.type == 'oracle':
|
||||||
h['account']['mode'] = 'sysdba' if account.privileged else None
|
h['account']['mode'] = 'sysdba' if account.privileged else None
|
||||||
inventory_hosts.append(h)
|
inventory_hosts.append(h)
|
||||||
|
method_hosts.append(h['name'])
|
||||||
|
self.method_hosts_mapper[method_attr] = method_hosts
|
||||||
ChangeSecretRecord.objects.bulk_create(records)
|
ChangeSecretRecord.objects.bulk_create(records)
|
||||||
return inventory_hosts
|
return inventory_hosts
|
||||||
|
|
||||||
@@ -165,46 +150,27 @@ class ChangeSecretManager(AccountBasePlaybookManager):
|
|||||||
recorder = self.name_recorder_mapper.get(host)
|
recorder = self.name_recorder_mapper.get(host)
|
||||||
if not recorder:
|
if not recorder:
|
||||||
return
|
return
|
||||||
recorder.status = ChangeSecretRecordStatusChoice.success.value
|
recorder.status = 'success'
|
||||||
recorder.date_finished = timezone.now()
|
recorder.date_finished = timezone.now()
|
||||||
|
recorder.save()
|
||||||
account = recorder.account
|
account = recorder.account
|
||||||
if not account:
|
if not account:
|
||||||
print("Account not found, deleted ?")
|
print("Account not found, deleted ?")
|
||||||
return
|
return
|
||||||
account.secret = recorder.new_secret
|
account.secret = recorder.new_secret
|
||||||
account.date_updated = timezone.now()
|
account.save(update_fields=['secret'])
|
||||||
|
|
||||||
max_retries = 3
|
|
||||||
retry_count = 0
|
|
||||||
|
|
||||||
while retry_count < max_retries:
|
|
||||||
try:
|
|
||||||
recorder.save()
|
|
||||||
account.save(update_fields=['secret', 'version', 'date_updated'])
|
|
||||||
break
|
|
||||||
except Exception as e:
|
|
||||||
retry_count += 1
|
|
||||||
if retry_count == max_retries:
|
|
||||||
self.on_host_error(host, str(e), result)
|
|
||||||
else:
|
|
||||||
print(f'retry {retry_count} times for {host} recorder save error: {e}')
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
def on_host_error(self, host, error, result):
|
def on_host_error(self, host, error, result):
|
||||||
recorder = self.name_recorder_mapper.get(host)
|
recorder = self.name_recorder_mapper.get(host)
|
||||||
if not recorder:
|
if not recorder:
|
||||||
return
|
return
|
||||||
recorder.status = ChangeSecretRecordStatusChoice.failed.value
|
recorder.status = 'failed'
|
||||||
recorder.date_finished = timezone.now()
|
recorder.date_finished = timezone.now()
|
||||||
recorder.error = error
|
recorder.error = error
|
||||||
try:
|
|
||||||
recorder.save()
|
recorder.save()
|
||||||
except Exception as e:
|
|
||||||
print(f"\033[31m Save {host} recorder error: {e} \033[0m\n")
|
|
||||||
|
|
||||||
def on_runner_failed(self, runner, e):
|
def on_runner_failed(self, runner, e):
|
||||||
logger.error("Account error: ", e)
|
logger.error("Change secret error: ", e)
|
||||||
|
|
||||||
def check_secret(self):
|
def check_secret(self):
|
||||||
if self.secret_strategy == SecretStrategy.custom \
|
if self.secret_strategy == SecretStrategy.custom \
|
||||||
@@ -213,69 +179,35 @@ class ChangeSecretManager(AccountBasePlaybookManager):
|
|||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def get_summary(recorders):
|
|
||||||
total, succeed, failed = 0, 0, 0
|
|
||||||
for recorder in recorders:
|
|
||||||
if recorder.status == ChangeSecretRecordStatusChoice.success.value:
|
|
||||||
succeed += 1
|
|
||||||
else:
|
|
||||||
failed += 1
|
|
||||||
total += 1
|
|
||||||
|
|
||||||
summary = _('Success: %s, Failed: %s, Total: %s') % (succeed, failed, total)
|
|
||||||
return summary
|
|
||||||
|
|
||||||
def run(self, *args, **kwargs):
|
def run(self, *args, **kwargs):
|
||||||
if self.secret_type and not self.check_secret():
|
if not self.check_secret():
|
||||||
return
|
return
|
||||||
super().run(*args, **kwargs)
|
super().run(*args, **kwargs)
|
||||||
recorders = list(self.name_recorder_mapper.values())
|
recorders = self.name_recorder_mapper.values()
|
||||||
summary = self.get_summary(recorders)
|
recorders = list(recorders)
|
||||||
print(summary, end='')
|
self.send_recorder_mail(recorders)
|
||||||
|
|
||||||
if self.record_map:
|
|
||||||
return
|
|
||||||
|
|
||||||
failed_recorders = [
|
|
||||||
r for r in recorders
|
|
||||||
if r.status == ChangeSecretRecordStatusChoice.failed.value
|
|
||||||
]
|
|
||||||
|
|
||||||
|
def send_recorder_mail(self, recorders):
|
||||||
recipients = self.execution.recipients
|
recipients = self.execution.recipients
|
||||||
|
if not recorders or not recipients:
|
||||||
|
return
|
||||||
|
|
||||||
recipients = User.objects.filter(id__in=list(recipients.keys()))
|
recipients = User.objects.filter(id__in=list(recipients.keys()))
|
||||||
if not recipients:
|
|
||||||
return
|
|
||||||
|
|
||||||
if failed_recorders:
|
|
||||||
name = self.execution.snapshot.get('name')
|
|
||||||
execution_id = str(self.execution.id)
|
|
||||||
_ids = [r.id for r in failed_recorders]
|
|
||||||
asset_account_errors = ChangeSecretRecord.objects.filter(
|
|
||||||
id__in=_ids).values_list('asset__name', 'account__username', 'error')
|
|
||||||
|
|
||||||
for user in recipients:
|
|
||||||
ChangeSecretFailedMsg(name, execution_id, user, asset_account_errors).publish()
|
|
||||||
|
|
||||||
if not recorders:
|
|
||||||
return
|
|
||||||
|
|
||||||
self.send_recorder_mail(recipients, recorders, summary)
|
|
||||||
|
|
||||||
def send_recorder_mail(self, recipients, recorders, summary):
|
|
||||||
name = self.execution.snapshot['name']
|
name = self.execution.snapshot['name']
|
||||||
path = os.path.join(os.path.dirname(settings.BASE_DIR), 'tmp')
|
path = os.path.join(os.path.dirname(settings.BASE_DIR), 'tmp')
|
||||||
filename = os.path.join(path, f'{name}-{local_now_filename()}-{time.time()}.xlsx')
|
filename = os.path.join(path, f'{name}-{local_now_display()}-{time.time()}.xlsx')
|
||||||
if not self.create_file(recorders, filename):
|
if not self.create_file(recorders, filename):
|
||||||
return
|
return
|
||||||
|
|
||||||
for user in recipients:
|
for user in recipients:
|
||||||
attachments = []
|
attachments = []
|
||||||
if user.secret_key:
|
if user.secret_key:
|
||||||
attachment = os.path.join(path, f'{name}-{local_now_filename()}-{time.time()}.zip')
|
password = user.secret_key.encode('utf8')
|
||||||
encrypt_and_compress_zip_file(attachment, user.secret_key, [filename])
|
attachment = os.path.join(path, f'{name}-{local_now_display()}-{time.time()}.zip')
|
||||||
|
encrypt_and_compress_zip_file(attachment, password, [filename])
|
||||||
attachments = [attachment]
|
attachments = [attachment]
|
||||||
ChangeSecretExecutionTaskMsg(name, user, summary).publish(attachments)
|
ChangeSecretExecutionTaskMsg(name, user).publish(attachments)
|
||||||
os.remove(filename)
|
os.remove(filename)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@@ -290,9 +222,8 @@ class ChangeSecretManager(AccountBasePlaybookManager):
|
|||||||
|
|
||||||
rows.insert(0, header)
|
rows.insert(0, header)
|
||||||
wb = Workbook(filename)
|
wb = Workbook(filename)
|
||||||
ws = wb.add_worksheet('Sheet1')
|
ws = wb.create_sheet('Sheet1')
|
||||||
for row_index, row_data in enumerate(rows):
|
for row in rows:
|
||||||
for col_index, col_data in enumerate(row_data):
|
ws.append(row)
|
||||||
ws.write_string(row_index, col_index, col_data)
|
wb.save(filename)
|
||||||
wb.close()
|
|
||||||
return True
|
return True
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
from .backup_account.manager import AccountBackupManager
|
|
||||||
from .change_secret.manager import ChangeSecretManager
|
|
||||||
from .gather_accounts.manager import GatherAccountsManager
|
|
||||||
from .push_account.manager import PushAccountManager
|
from .push_account.manager import PushAccountManager
|
||||||
from .remove_account.manager import RemoveAccountManager
|
from .change_secret.manager import ChangeSecretManager
|
||||||
from .verify_account.manager import VerifyAccountManager
|
from .verify_account.manager import VerifyAccountManager
|
||||||
|
from .backup_account.manager import AccountBackupManager
|
||||||
|
from .gather_accounts.manager import GatherAccountsManager
|
||||||
from .verify_gateway_account.manager import VerifyGatewayAccountManager
|
from .verify_gateway_account.manager import VerifyGatewayAccountManager
|
||||||
from ..const import AutomationTypes
|
from ..const import AutomationTypes
|
||||||
|
|
||||||
@@ -13,7 +12,6 @@ class ExecutionManager:
|
|||||||
AutomationTypes.push_account: PushAccountManager,
|
AutomationTypes.push_account: PushAccountManager,
|
||||||
AutomationTypes.change_secret: ChangeSecretManager,
|
AutomationTypes.change_secret: ChangeSecretManager,
|
||||||
AutomationTypes.verify_account: VerifyAccountManager,
|
AutomationTypes.verify_account: VerifyAccountManager,
|
||||||
AutomationTypes.remove_account: RemoveAccountManager,
|
|
||||||
AutomationTypes.gather_accounts: GatherAccountsManager,
|
AutomationTypes.gather_accounts: GatherAccountsManager,
|
||||||
AutomationTypes.verify_gateway_account: VerifyGatewayAccountManager,
|
AutomationTypes.verify_gateway_account: VerifyGatewayAccountManager,
|
||||||
# TODO 后期迁移到自动化策略中
|
# TODO 后期迁移到自动化策略中
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: mongodb
|
- hosts: mongodb
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Get info
|
- name: Get info
|
||||||
@@ -12,8 +12,8 @@
|
|||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
||||||
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert | default('') }}"
|
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert }}"
|
||||||
ssl_certfile: "{{ jms_asset.secret_info.client_key | default('') }}"
|
ssl_certfile: "{{ jms_asset.secret_info.client_key }}"
|
||||||
connection_options:
|
connection_options:
|
||||||
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
||||||
filter: users
|
filter: users
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
- hosts: mysql
|
- hosts: mysql
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
check_ssl: "{{ jms_asset.spec_info.use_ssl and not jms_asset.spec_info.allow_invalid_cert }}"
|
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Get info
|
- name: Get info
|
||||||
@@ -11,10 +10,6 @@
|
|||||||
login_password: "{{ jms_account.secret }}"
|
login_password: "{{ jms_account.secret }}"
|
||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
check_hostname: "{{ check_ssl if check_ssl else omit }}"
|
|
||||||
ca_cert: "{{ jms_asset.secret_info.ca_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_cert: "{{ jms_asset.secret_info.client_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_key: "{{ jms_asset.secret_info.client_key | default(omit) if check_ssl else omit }}"
|
|
||||||
filter: users
|
filter: users
|
||||||
register: db_info
|
register: db_info
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: oralce
|
- hosts: oralce
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Get info
|
- name: Get info
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: postgresql
|
- hosts: postgresql
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Get info
|
- name: Get info
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
- hosts: demo
|
- hosts: demo
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
tasks:
|
tasks:
|
||||||
- name: Gather windows account
|
- name: Gather posix account
|
||||||
ansible.builtin.win_shell: net user
|
ansible.builtin.win_shell: net user
|
||||||
register: result
|
register: result
|
||||||
ignore_errors: true
|
|
||||||
|
|
||||||
- name: Define info by set_fact
|
- name: Define info by set_fact
|
||||||
ansible.builtin.set_fact:
|
ansible.builtin.set_fact:
|
||||||
|
|||||||
@@ -1,14 +1,9 @@
|
|||||||
from collections import defaultdict
|
|
||||||
|
|
||||||
from accounts.const import AutomationTypes
|
from accounts.const import AutomationTypes
|
||||||
from accounts.models import GatheredAccount
|
from accounts.models import GatheredAccount
|
||||||
from assets.models import Asset
|
|
||||||
from common.utils import get_logger
|
from common.utils import get_logger
|
||||||
from orgs.utils import tmp_to_org
|
from orgs.utils import tmp_to_org
|
||||||
from users.models import User
|
|
||||||
from .filter import GatherAccountsFilter
|
from .filter import GatherAccountsFilter
|
||||||
from ..base.manager import AccountBasePlaybookManager
|
from ..base.manager import AccountBasePlaybookManager
|
||||||
from ...notifications import GatherAccountChangeMsg
|
|
||||||
|
|
||||||
logger = get_logger(__name__)
|
logger = get_logger(__name__)
|
||||||
|
|
||||||
@@ -17,9 +12,6 @@ class GatherAccountsManager(AccountBasePlaybookManager):
|
|||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
self.host_asset_mapper = {}
|
self.host_asset_mapper = {}
|
||||||
self.asset_account_info = {}
|
|
||||||
|
|
||||||
self.asset_username_mapper = defaultdict(set)
|
|
||||||
self.is_sync_account = self.execution.snapshot.get('is_sync_account')
|
self.is_sync_account = self.execution.snapshot.get('is_sync_account')
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
@@ -34,11 +26,10 @@ class GatherAccountsManager(AccountBasePlaybookManager):
|
|||||||
def filter_success_result(self, tp, result):
|
def filter_success_result(self, tp, result):
|
||||||
result = GatherAccountsFilter(tp).run(self.method_id_meta_mapper, result)
|
result = GatherAccountsFilter(tp).run(self.method_id_meta_mapper, result)
|
||||||
return result
|
return result
|
||||||
|
@staticmethod
|
||||||
def generate_data(self, asset, result):
|
def generate_data(asset, result):
|
||||||
data = []
|
data = []
|
||||||
for username, info in result.items():
|
for username, info in result.items():
|
||||||
self.asset_username_mapper[str(asset.id)].add(username)
|
|
||||||
d = {'asset': asset, 'username': username, 'present': True}
|
d = {'asset': asset, 'username': username, 'present': True}
|
||||||
if info.get('date'):
|
if info.get('date'):
|
||||||
d['date_last_login'] = info['date']
|
d['date_last_login'] = info['date']
|
||||||
@@ -47,29 +38,8 @@ class GatherAccountsManager(AccountBasePlaybookManager):
|
|||||||
data.append(d)
|
data.append(d)
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def collect_asset_account_info(self, asset, result):
|
def update_or_create_accounts(self, asset, result):
|
||||||
data = self.generate_data(asset, result)
|
data = self.generate_data(asset, result)
|
||||||
self.asset_account_info[asset] = data
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def get_nested_info(data, *keys):
|
|
||||||
for key in keys:
|
|
||||||
data = data.get(key, {})
|
|
||||||
if not data:
|
|
||||||
break
|
|
||||||
return data
|
|
||||||
|
|
||||||
def on_host_success(self, host, result):
|
|
||||||
info = self.get_nested_info(result, 'debug', 'res', 'info')
|
|
||||||
asset = self.host_asset_mapper.get(host)
|
|
||||||
if asset and info:
|
|
||||||
result = self.filter_success_result(asset.type, info)
|
|
||||||
self.collect_asset_account_info(asset, result)
|
|
||||||
else:
|
|
||||||
print(f'\033[31m Not found {host} info \033[0m\n')
|
|
||||||
|
|
||||||
def update_or_create_accounts(self):
|
|
||||||
for asset, data in self.asset_account_info.items():
|
|
||||||
with tmp_to_org(asset.org_id):
|
with tmp_to_org(asset.org_id):
|
||||||
gathered_accounts = []
|
gathered_accounts = []
|
||||||
GatheredAccount.objects.filter(asset=asset, present=True).update(present=False)
|
GatheredAccount.objects.filter(asset=asset, present=True).update(present=False)
|
||||||
@@ -80,60 +50,14 @@ class GatherAccountsManager(AccountBasePlaybookManager):
|
|||||||
)
|
)
|
||||||
gathered_accounts.append(gathered_account)
|
gathered_accounts.append(gathered_account)
|
||||||
if not self.is_sync_account:
|
if not self.is_sync_account:
|
||||||
continue
|
return
|
||||||
GatheredAccount.sync_accounts(gathered_accounts)
|
GatheredAccount.sync_accounts(gathered_accounts)
|
||||||
|
|
||||||
def run(self, *args, **kwargs):
|
def on_host_success(self, host, result):
|
||||||
super().run(*args, **kwargs)
|
info = result.get('debug', {}).get('res', {}).get('info', {})
|
||||||
users, change_info = self.generate_send_users_and_change_info()
|
asset = self.host_asset_mapper.get(host)
|
||||||
self.update_or_create_accounts()
|
if asset and info:
|
||||||
self.send_email_if_need(users, change_info)
|
result = self.filter_success_result(asset.type, info)
|
||||||
|
self.update_or_create_accounts(asset, result)
|
||||||
def generate_send_users_and_change_info(self):
|
else:
|
||||||
recipients = self.execution.recipients
|
logger.error("Not found info".format(host))
|
||||||
if not self.asset_username_mapper or not recipients:
|
|
||||||
return None, None
|
|
||||||
|
|
||||||
users = User.objects.filter(id__in=recipients)
|
|
||||||
if not users:
|
|
||||||
return users, None
|
|
||||||
|
|
||||||
asset_ids = self.asset_username_mapper.keys()
|
|
||||||
assets = Asset.objects.filter(id__in=asset_ids)
|
|
||||||
gather_accounts = GatheredAccount.objects.filter(asset_id__in=asset_ids, present=True)
|
|
||||||
asset_id_map = {str(asset.id): asset for asset in assets}
|
|
||||||
asset_id_username = list(assets.values_list('id', 'accounts__username'))
|
|
||||||
asset_id_username.extend(list(gather_accounts.values_list('asset_id', 'username')))
|
|
||||||
|
|
||||||
system_asset_username_mapper = defaultdict(set)
|
|
||||||
for asset_id, username in asset_id_username:
|
|
||||||
system_asset_username_mapper[str(asset_id)].add(username)
|
|
||||||
|
|
||||||
change_info = {}
|
|
||||||
for asset_id, usernames in self.asset_username_mapper.items():
|
|
||||||
system_usernames = system_asset_username_mapper.get(asset_id)
|
|
||||||
|
|
||||||
if not system_usernames:
|
|
||||||
continue
|
|
||||||
|
|
||||||
add_usernames = usernames - system_usernames
|
|
||||||
remove_usernames = system_usernames - usernames
|
|
||||||
k = f'{asset_id_map[asset_id]}[{asset_id}]'
|
|
||||||
|
|
||||||
if not add_usernames and not remove_usernames:
|
|
||||||
continue
|
|
||||||
|
|
||||||
change_info[k] = {
|
|
||||||
'add_usernames': ', '.join(add_usernames),
|
|
||||||
'remove_usernames': ', '.join(remove_usernames),
|
|
||||||
}
|
|
||||||
|
|
||||||
return users, change_info
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def send_email_if_need(users, change_info):
|
|
||||||
if not users or not change_info:
|
|
||||||
return
|
|
||||||
|
|
||||||
for user in users:
|
|
||||||
GatherAccountChangeMsg(user, change_info).publish_async()
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: mongodb
|
- hosts: mongodb
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test MongoDB connection
|
- name: Test MongoDB connection
|
||||||
@@ -12,8 +12,8 @@
|
|||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
||||||
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert | default('') }}"
|
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert }}"
|
||||||
ssl_certfile: "{{ jms_asset.secret_info.client_key | default('') }}"
|
ssl_certfile: "{{ jms_asset.secret_info.client_key }}"
|
||||||
connection_options:
|
connection_options:
|
||||||
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
||||||
register: db_info
|
register: db_info
|
||||||
@@ -31,8 +31,8 @@
|
|||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
||||||
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert | default('') }}"
|
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert }}"
|
||||||
ssl_certfile: "{{ jms_asset.secret_info.client_key | default('') }}"
|
ssl_certfile: "{{ jms_asset.secret_info.client_key }}"
|
||||||
connection_options:
|
connection_options:
|
||||||
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
||||||
db: "{{ jms_asset.spec_info.db_name }}"
|
db: "{{ jms_asset.spec_info.db_name }}"
|
||||||
@@ -49,9 +49,7 @@
|
|||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
||||||
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert | default('') }}"
|
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert }}"
|
||||||
ssl_certfile: "{{ jms_asset.secret_info.client_key | default('') }}"
|
ssl_certfile: "{{ jms_asset.secret_info.client_key }}"
|
||||||
connection_options:
|
connection_options:
|
||||||
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
||||||
register: result
|
|
||||||
failed_when: not result.is_available
|
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
- hosts: mysql
|
- hosts: mysql
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
db_name: "{{ jms_asset.spec_info.db_name }}"
|
db_name: "{{ jms_asset.spec_info.db_name }}"
|
||||||
check_ssl: "{{ jms_asset.spec_info.use_ssl and not jms_asset.spec_info.allow_invalid_cert }}"
|
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test MySQL connection
|
- name: Test MySQL connection
|
||||||
@@ -12,10 +11,6 @@
|
|||||||
login_password: "{{ jms_account.secret }}"
|
login_password: "{{ jms_account.secret }}"
|
||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
check_hostname: "{{ check_ssl if check_ssl else omit }}"
|
|
||||||
ca_cert: "{{ jms_asset.secret_info.ca_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_cert: "{{ jms_asset.secret_info.client_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_key: "{{ jms_asset.secret_info.client_key | default(omit) if check_ssl else omit }}"
|
|
||||||
filter: version
|
filter: version
|
||||||
register: db_info
|
register: db_info
|
||||||
|
|
||||||
@@ -29,15 +24,10 @@
|
|||||||
login_password: "{{ jms_account.secret }}"
|
login_password: "{{ jms_account.secret }}"
|
||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
check_hostname: "{{ check_ssl if check_ssl else omit }}"
|
|
||||||
ca_cert: "{{ jms_asset.secret_info.ca_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_cert: "{{ jms_asset.secret_info.client_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_key: "{{ jms_asset.secret_info.client_key | default(omit) if check_ssl else omit }}"
|
|
||||||
name: "{{ account.username }}"
|
name: "{{ account.username }}"
|
||||||
password: "{{ account.secret }}"
|
password: "{{ account.secret }}"
|
||||||
host: "%"
|
host: "%"
|
||||||
priv: "{{ omit if db_name == '' else db_name + '.*:ALL' }}"
|
priv: "{{ account.username + '.*:USAGE' if db_name == '' else db_name + '.*:ALL' }}"
|
||||||
append_privs: "{{ db_name != '' | bool }}"
|
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
when: db_info is succeeded
|
when: db_info is succeeded
|
||||||
|
|
||||||
@@ -47,8 +37,4 @@
|
|||||||
login_password: "{{ account.secret }}"
|
login_password: "{{ account.secret }}"
|
||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
check_hostname: "{{ check_ssl if check_ssl else omit }}"
|
|
||||||
ca_cert: "{{ jms_asset.secret_info.ca_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_cert: "{{ jms_asset.secret_info.client_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_key: "{{ jms_asset.secret_info.client_key | default(omit) if check_ssl else omit }}"
|
|
||||||
filter: version
|
filter: version
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: oracle
|
- hosts: oracle
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test Oracle connection
|
- name: Test Oracle connection
|
||||||
@@ -39,4 +39,3 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
mode: "{{ account.mode }}"
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: postgre
|
- hosts: postgre
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test PostgreSQL connection
|
- name: Test PostgreSQL connection
|
||||||
@@ -39,5 +39,6 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
db: "{{ jms_asset.spec_info.db_name }}"
|
db: "{{ jms_asset.spec_info.db_name }}"
|
||||||
register: result
|
when:
|
||||||
failed_when: not result.is_available
|
- result is succeeded
|
||||||
|
- change_info is succeeded
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: sqlserver
|
- hosts: sqlserver
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Test SQLServer connection
|
- name: Test SQLServer connection
|
||||||
@@ -40,7 +40,7 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
name: '{{ jms_asset.spec_info.db_name }}'
|
name: '{{ jms_asset.spec_info.db_name }}'
|
||||||
script: "ALTER LOGIN {{ account.username }} WITH PASSWORD = '{{ account.secret }}', DEFAULT_DATABASE = {{ jms_asset.spec_info.db_name }}; select @@version"
|
script: "ALTER LOGIN {{ account.username }} WITH PASSWORD = '{{ account.secret }}'; select @@version"
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
when: user_exist.query_results[0] | length != 0
|
when: user_exist.query_results[0] | length != 0
|
||||||
register: change_info
|
register: change_info
|
||||||
@@ -52,7 +52,7 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
name: '{{ jms_asset.spec_info.db_name }}'
|
name: '{{ jms_asset.spec_info.db_name }}'
|
||||||
script: "CREATE LOGIN [{{ account.username }}] WITH PASSWORD = '{{ account.secret }}'; CREATE USER [{{ account.username }}] FOR LOGIN [{{ account.username }}]; select @@version"
|
script: "CREATE LOGIN {{ account.username }} WITH PASSWORD = '{{ account.secret }}'; select @@version"
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
when: user_exist.query_results[0] | length == 0
|
when: user_exist.query_results[0] | length == 0
|
||||||
register: change_info
|
register: change_info
|
||||||
|
|||||||
@@ -35,24 +35,12 @@
|
|||||||
- user_info.failed
|
- user_info.failed
|
||||||
- params.groups
|
- params.groups
|
||||||
|
|
||||||
- name: "Set {{ account.username }} sudo setting"
|
|
||||||
ansible.builtin.lineinfile:
|
|
||||||
dest: /etc/sudoers
|
|
||||||
state: present
|
|
||||||
regexp: "^{{ account.username }} ALL="
|
|
||||||
line: "{{ account.username + ' ALL=(ALL) NOPASSWD: ' + params.sudo }}"
|
|
||||||
validate: visudo -cf %s
|
|
||||||
when:
|
|
||||||
- user_info.failed or params.modify_sudo
|
|
||||||
- params.sudo
|
|
||||||
|
|
||||||
- name: "Change {{ account.username }} password"
|
- name: "Change {{ account.username }} password"
|
||||||
ansible.builtin.user:
|
ansible.builtin.user:
|
||||||
name: "{{ account.username }}"
|
name: "{{ account.username }}"
|
||||||
password: "{{ account.secret | password_hash('des') }}"
|
password: "{{ account.secret | password_hash('des') }}"
|
||||||
update_password: always
|
update_password: always
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
register: change_secret_result
|
|
||||||
when: account.secret_type == "password"
|
when: account.secret_type == "password"
|
||||||
|
|
||||||
- name: remove jumpserver ssh key
|
- name: remove jumpserver ssh key
|
||||||
@@ -69,9 +57,19 @@
|
|||||||
user: "{{ account.username }}"
|
user: "{{ account.username }}"
|
||||||
key: "{{ account.secret }}"
|
key: "{{ account.secret }}"
|
||||||
exclusive: "{{ ssh_params.exclusive }}"
|
exclusive: "{{ ssh_params.exclusive }}"
|
||||||
register: change_secret_result
|
|
||||||
when: account.secret_type == "ssh_key"
|
when: account.secret_type == "ssh_key"
|
||||||
|
|
||||||
|
- name: "Set {{ account.username }} sudo setting"
|
||||||
|
ansible.builtin.lineinfile:
|
||||||
|
dest: /etc/sudoers
|
||||||
|
state: present
|
||||||
|
regexp: "^{{ account.username }} ALL="
|
||||||
|
line: "{{ account.username + ' ALL=(ALL) NOPASSWD: ' + params.sudo }}"
|
||||||
|
validate: visudo -cf %s
|
||||||
|
when:
|
||||||
|
- user_info.failed
|
||||||
|
- params.sudo
|
||||||
|
|
||||||
- name: Refresh connection
|
- name: Refresh connection
|
||||||
ansible.builtin.meta: reset_connection
|
ansible.builtin.meta: reset_connection
|
||||||
|
|
||||||
@@ -82,15 +80,8 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
||||||
become: "{{ account.become.ansible_become | default(False) }}"
|
become: false
|
||||||
become_method: su
|
when: account.secret_type == "password"
|
||||||
become_user: "{{ account.become.ansible_user | default('') }}"
|
|
||||||
become_password: "{{ account.become.ansible_password | default('') }}"
|
|
||||||
become_private_key_path: "{{ account.become.ansible_ssh_private_key_file | default(None) }}"
|
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
|
||||||
when:
|
|
||||||
- account.secret_type == "password"
|
|
||||||
- check_conn_after_change or change_secret_result.failed | default(false)
|
|
||||||
delegate_to: localhost
|
delegate_to: localhost
|
||||||
|
|
||||||
- name: "Verify {{ account.username }} SSH KEY (paramiko)"
|
- name: "Verify {{ account.username }} SSH KEY (paramiko)"
|
||||||
@@ -100,9 +91,7 @@
|
|||||||
login_user: "{{ account.username }}"
|
login_user: "{{ account.username }}"
|
||||||
login_private_key_path: "{{ account.private_key_path }}"
|
login_private_key_path: "{{ account.private_key_path }}"
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
become: false
|
||||||
when:
|
when: account.secret_type == "ssh_key"
|
||||||
- account.secret_type == "ssh_key"
|
|
||||||
- check_conn_after_change or change_secret_result.failed | default(false)
|
|
||||||
delegate_to: localhost
|
delegate_to: localhost
|
||||||
|
|
||||||
|
|||||||
@@ -5,17 +5,11 @@ type:
|
|||||||
- AIX
|
- AIX
|
||||||
method: push_account
|
method: push_account
|
||||||
params:
|
params:
|
||||||
- name: modify_sudo
|
|
||||||
type: bool
|
|
||||||
label: "{{ 'Modify sudo label' | trans }}"
|
|
||||||
default: False
|
|
||||||
help_text: "{{ 'Modify params sudo help text' | trans }}"
|
|
||||||
|
|
||||||
- name: sudo
|
- name: sudo
|
||||||
type: str
|
type: str
|
||||||
label: 'Sudo'
|
label: 'Sudo'
|
||||||
default: '/bin/whoami'
|
default: '/bin/whoami'
|
||||||
help_text: "{{ 'Params sudo help text' | trans }}"
|
help_text: '使用逗号分隔多个命令,如: /bin/whoami,/sbin/ifconfig'
|
||||||
|
|
||||||
- name: shell
|
- name: shell
|
||||||
type: str
|
type: str
|
||||||
@@ -24,54 +18,19 @@ params:
|
|||||||
|
|
||||||
- name: home
|
- name: home
|
||||||
type: str
|
type: str
|
||||||
label: "{{ 'Params home label' | trans }}"
|
label: '家目录'
|
||||||
default: ''
|
default: ''
|
||||||
help_text: "{{ 'Params home help text' | trans }}"
|
help_text: '默认家目录 /home/系统用户名: /home/username'
|
||||||
|
|
||||||
- name: groups
|
- name: groups
|
||||||
type: str
|
type: str
|
||||||
label: "{{ 'Params groups label' | trans }}"
|
label: '用户组'
|
||||||
default: ''
|
default: ''
|
||||||
help_text: "{{ 'Params groups help text' | trans }}"
|
help_text: '请输入用户组,多个用户组使用逗号分隔(需填写已存在的用户组)'
|
||||||
|
|
||||||
i18n:
|
i18n:
|
||||||
Aix account push:
|
Aix account push:
|
||||||
zh: '使用 Ansible 模块 user 执行 Aix 账号推送 (DES)'
|
zh: 使用 Ansible 模块 user 执行 Aix 账号推送 (DES)
|
||||||
ja: 'Ansible user モジュールを使用して Aix アカウントをプッシュする (DES)'
|
ja: Ansible user モジュールを使用して Aix アカウントをプッシュする (DES)
|
||||||
en: 'Using Ansible module user to push account (DES)'
|
en: Using Ansible module user to push account (DES)
|
||||||
|
|
||||||
Modify params sudo help text:
|
|
||||||
zh: '如果用户存在,可以修改sudo权限'
|
|
||||||
ja: 'ユーザーが存在する場合、sudo権限を変更できます'
|
|
||||||
en: 'If the user exists, sudo permissions can be modified'
|
|
||||||
|
|
||||||
Params sudo help text:
|
|
||||||
zh: '使用逗号分隔多个命令,如: /bin/whoami,/sbin/ifconfig'
|
|
||||||
ja: 'コンマで区切って複数のコマンドを入力してください。例: /bin/whoami,/sbin/ifconfig'
|
|
||||||
en: 'Use commas to separate multiple commands, such as: /bin/whoami,/sbin/ifconfig'
|
|
||||||
|
|
||||||
Params home help text:
|
|
||||||
zh: '默认家目录 /home/{账号用户名}'
|
|
||||||
ja: 'デフォルトのホームディレクトリ /home/{アカウントユーザ名}'
|
|
||||||
en: 'Default home directory /home/{account username}'
|
|
||||||
|
|
||||||
Params groups help text:
|
|
||||||
zh: '请输入用户组,多个用户组使用逗号分隔(需填写已存在的用户组)'
|
|
||||||
ja: 'グループを入力してください。複数のグループはコンマで区切ってください(既存のグループを入力してください)'
|
|
||||||
en: 'Please enter the group. Multiple groups are separated by commas (please enter the existing group)'
|
|
||||||
|
|
||||||
Modify sudo label:
|
|
||||||
zh: '修改 sudo 权限'
|
|
||||||
ja: 'sudo 権限を変更'
|
|
||||||
en: 'Modify sudo'
|
|
||||||
|
|
||||||
Params home label:
|
|
||||||
zh: '家目录'
|
|
||||||
ja: 'ホームディレクトリ'
|
|
||||||
en: 'Home'
|
|
||||||
|
|
||||||
Params groups label:
|
|
||||||
zh: '用户组'
|
|
||||||
ja: 'グループ'
|
|
||||||
en: 'Groups'
|
|
||||||
|
|
||||||
|
|||||||
@@ -35,24 +35,12 @@
|
|||||||
- user_info.failed
|
- user_info.failed
|
||||||
- params.groups
|
- params.groups
|
||||||
|
|
||||||
- name: "Set {{ account.username }} sudo setting"
|
|
||||||
ansible.builtin.lineinfile:
|
|
||||||
dest: /etc/sudoers
|
|
||||||
state: present
|
|
||||||
regexp: "^{{ account.username }} ALL="
|
|
||||||
line: "{{ account.username + ' ALL=(ALL) NOPASSWD: ' + params.sudo }}"
|
|
||||||
validate: visudo -cf %s
|
|
||||||
when:
|
|
||||||
- user_info.failed or params.modify_sudo
|
|
||||||
- params.sudo
|
|
||||||
|
|
||||||
- name: "Change {{ account.username }} password"
|
- name: "Change {{ account.username }} password"
|
||||||
ansible.builtin.user:
|
ansible.builtin.user:
|
||||||
name: "{{ account.username }}"
|
name: "{{ account.username }}"
|
||||||
password: "{{ account.secret | password_hash('sha512') }}"
|
password: "{{ account.secret | password_hash('sha512') }}"
|
||||||
update_password: always
|
update_password: always
|
||||||
ignore_errors: true
|
ignore_errors: true
|
||||||
register: change_secret_result
|
|
||||||
when: account.secret_type == "password"
|
when: account.secret_type == "password"
|
||||||
|
|
||||||
- name: remove jumpserver ssh key
|
- name: remove jumpserver ssh key
|
||||||
@@ -69,9 +57,19 @@
|
|||||||
user: "{{ account.username }}"
|
user: "{{ account.username }}"
|
||||||
key: "{{ account.secret }}"
|
key: "{{ account.secret }}"
|
||||||
exclusive: "{{ ssh_params.exclusive }}"
|
exclusive: "{{ ssh_params.exclusive }}"
|
||||||
register: change_secret_result
|
|
||||||
when: account.secret_type == "ssh_key"
|
when: account.secret_type == "ssh_key"
|
||||||
|
|
||||||
|
- name: "Set {{ account.username }} sudo setting"
|
||||||
|
ansible.builtin.lineinfile:
|
||||||
|
dest: /etc/sudoers
|
||||||
|
state: present
|
||||||
|
regexp: "^{{ account.username }} ALL="
|
||||||
|
line: "{{ account.username + ' ALL=(ALL) NOPASSWD: ' + params.sudo }}"
|
||||||
|
validate: visudo -cf %s
|
||||||
|
when:
|
||||||
|
- user_info.failed
|
||||||
|
- params.sudo
|
||||||
|
|
||||||
- name: Refresh connection
|
- name: Refresh connection
|
||||||
ansible.builtin.meta: reset_connection
|
ansible.builtin.meta: reset_connection
|
||||||
|
|
||||||
@@ -82,15 +80,8 @@
|
|||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
||||||
become: "{{ account.become.ansible_become | default(False) }}"
|
become: false
|
||||||
become_method: su
|
when: account.secret_type == "password"
|
||||||
become_user: "{{ account.become.ansible_user | default('') }}"
|
|
||||||
become_password: "{{ account.become.ansible_password | default('') }}"
|
|
||||||
become_private_key_path: "{{ account.become.ansible_ssh_private_key_file | default(None) }}"
|
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
|
||||||
when:
|
|
||||||
- account.secret_type == "password"
|
|
||||||
- check_conn_after_change or change_secret_result.failed | default(false)
|
|
||||||
delegate_to: localhost
|
delegate_to: localhost
|
||||||
|
|
||||||
- name: "Verify {{ account.username }} SSH KEY (paramiko)"
|
- name: "Verify {{ account.username }} SSH KEY (paramiko)"
|
||||||
@@ -100,9 +91,7 @@
|
|||||||
login_user: "{{ account.username }}"
|
login_user: "{{ account.username }}"
|
||||||
login_private_key_path: "{{ account.private_key_path }}"
|
login_private_key_path: "{{ account.private_key_path }}"
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default('') }}"
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
become: false
|
||||||
when:
|
when: account.secret_type == "ssh_key"
|
||||||
- account.secret_type == "ssh_key"
|
|
||||||
- check_conn_after_change or change_secret_result.failed | default(false)
|
|
||||||
delegate_to: localhost
|
delegate_to: localhost
|
||||||
|
|
||||||
|
|||||||
@@ -6,17 +6,11 @@ type:
|
|||||||
- linux
|
- linux
|
||||||
method: push_account
|
method: push_account
|
||||||
params:
|
params:
|
||||||
- name: modify_sudo
|
|
||||||
type: bool
|
|
||||||
label: "{{ 'Modify sudo label' | trans }}"
|
|
||||||
default: False
|
|
||||||
help_text: "{{ 'Modify params sudo help text' | trans }}"
|
|
||||||
|
|
||||||
- name: sudo
|
- name: sudo
|
||||||
type: str
|
type: str
|
||||||
label: 'Sudo'
|
label: 'Sudo'
|
||||||
default: '/bin/whoami'
|
default: '/bin/whoami'
|
||||||
help_text: "{{ 'Params sudo help text' | trans }}"
|
help_text: '使用逗号分隔多个命令,如: /bin/whoami,/sbin/ifconfig'
|
||||||
|
|
||||||
- name: shell
|
- name: shell
|
||||||
type: str
|
type: str
|
||||||
@@ -26,53 +20,18 @@ params:
|
|||||||
|
|
||||||
- name: home
|
- name: home
|
||||||
type: str
|
type: str
|
||||||
label: "{{ 'Params home label' | trans }}"
|
label: '家目录'
|
||||||
default: ''
|
default: ''
|
||||||
help_text: "{{ 'Params home help text' | trans }}"
|
help_text: '默认家目录 /home/系统用户名: /home/username'
|
||||||
|
|
||||||
- name: groups
|
- name: groups
|
||||||
type: str
|
type: str
|
||||||
label: "{{ 'Params groups label' | trans }}"
|
label: '用户组'
|
||||||
default: ''
|
default: ''
|
||||||
help_text: "{{ 'Params groups help text' | trans }}"
|
help_text: '请输入用户组,多个用户组使用逗号分隔(需填写已存在的用户组)'
|
||||||
|
|
||||||
i18n:
|
i18n:
|
||||||
Posix account push:
|
Posix account push:
|
||||||
zh: '使用 Ansible 模块 user 执行账号推送 (sha512)'
|
zh: 使用 Ansible 模块 user 执行账号推送 (sha512)
|
||||||
ja: 'Ansible user モジュールを使用してアカウントをプッシュする (sha512)'
|
ja: Ansible user モジュールを使用してアカウントをプッシュする (sha512)
|
||||||
en: 'Using Ansible module user to push account (sha512)'
|
en: Using Ansible module user to push account (sha512)
|
||||||
|
|
||||||
Modify params sudo help text:
|
|
||||||
zh: '如果用户存在,可以修改sudo权限'
|
|
||||||
ja: 'ユーザーが存在する場合、sudo権限を変更できます'
|
|
||||||
en: 'If the user exists, sudo permissions can be modified'
|
|
||||||
|
|
||||||
Params sudo help text:
|
|
||||||
zh: '使用逗号分隔多个命令,如: /bin/whoami,/sbin/ifconfig'
|
|
||||||
ja: 'コンマで区切って複数のコマンドを入力してください。例: /bin/whoami,/sbin/ifconfig'
|
|
||||||
en: 'Use commas to separate multiple commands, such as: /bin/whoami,/sbin/ifconfig'
|
|
||||||
|
|
||||||
Params home help text:
|
|
||||||
zh: '默认家目录 /home/{账号用户名}'
|
|
||||||
ja: 'デフォルトのホームディレクトリ /home/{アカウントユーザ名}'
|
|
||||||
en: 'Default home directory /home/{account username}'
|
|
||||||
|
|
||||||
Params groups help text:
|
|
||||||
zh: '请输入用户组,多个用户组使用逗号分隔(需填写已存在的用户组)'
|
|
||||||
ja: 'グループを入力してください。複数のグループはコンマで区切ってください(既存のグループを入力してください)'
|
|
||||||
en: 'Please enter the group. Multiple groups are separated by commas (please enter the existing group)'
|
|
||||||
|
|
||||||
Modify sudo label:
|
|
||||||
zh: '修改 sudo 权限'
|
|
||||||
ja: 'sudo 権限を変更'
|
|
||||||
en: 'Modify sudo'
|
|
||||||
|
|
||||||
Params home label:
|
|
||||||
zh: '家目录'
|
|
||||||
ja: 'ホームディレクトリ'
|
|
||||||
en: 'Home'
|
|
||||||
|
|
||||||
Params groups label:
|
|
||||||
zh: '用户组'
|
|
||||||
ja: 'グループ'
|
|
||||||
en: 'Groups'
|
|
||||||
|
|||||||
@@ -28,6 +28,4 @@
|
|||||||
vars:
|
vars:
|
||||||
ansible_user: "{{ account.username }}"
|
ansible_user: "{{ account.username }}"
|
||||||
ansible_password: "{{ account.secret }}"
|
ansible_password: "{{ account.secret }}"
|
||||||
when:
|
when: account.secret_type == "password"
|
||||||
- account.secret_type == "password"
|
|
||||||
- check_conn_after_change
|
|
||||||
|
|||||||
@@ -10,15 +10,10 @@ params:
|
|||||||
type: str
|
type: str
|
||||||
label: '用户组'
|
label: '用户组'
|
||||||
default: 'Users,Remote Desktop Users'
|
default: 'Users,Remote Desktop Users'
|
||||||
help_text: "{{ 'Params groups help text' | trans }}"
|
help_text: '请输入用户组,多个用户组使用逗号分隔(需填写已存在的用户组)'
|
||||||
|
|
||||||
i18n:
|
i18n:
|
||||||
Windows account push:
|
Windows account push:
|
||||||
zh: '使用 Ansible 模块 win_user 执行 Windows 账号推送'
|
zh: 使用 Ansible 模块 win_user 执行 Windows 账号推送
|
||||||
ja: 'Ansible win_user モジュールを使用して Windows アカウントをプッシュする'
|
ja: Ansible win_user モジュールを使用して Windows アカウントをプッシュする
|
||||||
en: 'Using Ansible module win_user to push account'
|
en: Using Ansible module win_user to push account
|
||||||
|
|
||||||
Params groups help text:
|
|
||||||
zh: '请输入用户组,多个用户组使用逗号分隔(需填写已存在的用户组)'
|
|
||||||
ja: 'グループを入力してください。複数のグループはコンマで区切ってください(既存のグループを入力してください)'
|
|
||||||
en: 'Please enter the group. Multiple groups are separated by commas (please enter the existing group)'
|
|
||||||
|
|||||||
@@ -1,37 +0,0 @@
|
|||||||
- hosts: demo
|
|
||||||
gather_facts: no
|
|
||||||
tasks:
|
|
||||||
- name: Test privileged account
|
|
||||||
ansible.windows.win_ping:
|
|
||||||
|
|
||||||
# - name: Print variables
|
|
||||||
# debug:
|
|
||||||
# msg: "Username: {{ account.username }}, Password: {{ account.secret }}"
|
|
||||||
|
|
||||||
- name: Push user password
|
|
||||||
ansible.windows.win_user:
|
|
||||||
fullname: "{{ account.username}}"
|
|
||||||
name: "{{ account.username }}"
|
|
||||||
password: "{{ account.secret }}"
|
|
||||||
password_never_expires: yes
|
|
||||||
groups: "{{ params.groups }}"
|
|
||||||
groups_action: add
|
|
||||||
update_password: always
|
|
||||||
ignore_errors: true
|
|
||||||
when: account.secret_type == "password"
|
|
||||||
|
|
||||||
- name: Refresh connection
|
|
||||||
ansible.builtin.meta: reset_connection
|
|
||||||
|
|
||||||
- name: Verify password (pyfreerdp)
|
|
||||||
rdp_ping:
|
|
||||||
login_host: "{{ jms_asset.address }}"
|
|
||||||
login_port: "{{ jms_asset.protocols | selectattr('name', 'equalto', 'rdp') | map(attribute='port') | first }}"
|
|
||||||
login_user: "{{ account.username }}"
|
|
||||||
login_password: "{{ account.secret }}"
|
|
||||||
login_secret_type: "{{ account.secret_type }}"
|
|
||||||
login_private_key_path: "{{ account.private_key_path }}"
|
|
||||||
when:
|
|
||||||
- account.secret_type == "password"
|
|
||||||
- check_conn_after_change
|
|
||||||
delegate_to: localhost
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
id: push_account_windows_rdp_verify
|
|
||||||
name: "{{ 'Windows account push rdp verify' | trans }}"
|
|
||||||
version: 1
|
|
||||||
method: push_account
|
|
||||||
category: host
|
|
||||||
type:
|
|
||||||
- windows
|
|
||||||
priority: 49
|
|
||||||
params:
|
|
||||||
- name: groups
|
|
||||||
type: str
|
|
||||||
label: '用户组'
|
|
||||||
default: 'Users,Remote Desktop Users'
|
|
||||||
help_text: "{{ 'Params groups help text' | trans }}"
|
|
||||||
|
|
||||||
i18n:
|
|
||||||
Windows account push rdp verify:
|
|
||||||
zh: '使用 Ansible 模块 win_user 执行 Windows 账号推送(最后使用 Python 模块 pyfreerdp 验证账号的可连接性)'
|
|
||||||
ja: 'Ansible モジュール win_user を使用して Windows アカウントのプッシュを実行します (最後に Python モジュール pyfreerdp を使用してアカウントの接続性を確認します)'
|
|
||||||
en: 'Use the Ansible module win_user to perform Windows account push (finally use the Python module pyfreerdp to verify the connectability of the account)'
|
|
||||||
|
|
||||||
Params groups help text:
|
|
||||||
zh: '请输入用户组,多个用户组使用逗号分隔(需填写已存在的用户组)'
|
|
||||||
ja: 'グループを入力してください。複数のグループはコンマで区切ってください(既存のグループを入力してください)'
|
|
||||||
en: 'Please enter the group. Multiple groups are separated by commas (please enter the existing group)'
|
|
||||||
@@ -1,4 +1,7 @@
|
|||||||
from accounts.const import AutomationTypes
|
from copy import deepcopy
|
||||||
|
|
||||||
|
from accounts.const import AutomationTypes, SecretType, Connectivity
|
||||||
|
from assets.const import HostTypes
|
||||||
from common.utils import get_logger
|
from common.utils import get_logger
|
||||||
from ..base.manager import AccountBasePlaybookManager
|
from ..base.manager import AccountBasePlaybookManager
|
||||||
from ..change_secret.manager import ChangeSecretManager
|
from ..change_secret.manager import ChangeSecretManager
|
||||||
@@ -7,11 +10,83 @@ logger = get_logger(__name__)
|
|||||||
|
|
||||||
|
|
||||||
class PushAccountManager(ChangeSecretManager, AccountBasePlaybookManager):
|
class PushAccountManager(ChangeSecretManager, AccountBasePlaybookManager):
|
||||||
|
ansible_account_prefer = ''
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def method_type(cls):
|
def method_type(cls):
|
||||||
return AutomationTypes.push_account
|
return AutomationTypes.push_account
|
||||||
|
|
||||||
|
def host_callback(self, host, asset=None, account=None, automation=None, path_dir=None, **kwargs):
|
||||||
|
host = super(ChangeSecretManager, self).host_callback(
|
||||||
|
host, asset=asset, account=account, automation=automation,
|
||||||
|
path_dir=path_dir, **kwargs
|
||||||
|
)
|
||||||
|
if host.get('error'):
|
||||||
|
return host
|
||||||
|
|
||||||
|
accounts = self.get_accounts(account)
|
||||||
|
inventory_hosts = []
|
||||||
|
if asset.type == HostTypes.WINDOWS and self.secret_type == SecretType.SSH_KEY:
|
||||||
|
msg = f'Windows {asset} does not support ssh key push'
|
||||||
|
print(msg)
|
||||||
|
return inventory_hosts
|
||||||
|
|
||||||
|
host['ssh_params'] = {}
|
||||||
|
for account in accounts:
|
||||||
|
h = deepcopy(host)
|
||||||
|
secret_type = account.secret_type
|
||||||
|
h['name'] += '(' + account.username + ')'
|
||||||
|
if self.secret_type is None:
|
||||||
|
new_secret = account.secret
|
||||||
|
else:
|
||||||
|
new_secret = self.get_secret(secret_type)
|
||||||
|
|
||||||
|
self.name_recorder_mapper[h['name']] = {
|
||||||
|
'account': account, 'new_secret': new_secret,
|
||||||
|
}
|
||||||
|
|
||||||
|
private_key_path = None
|
||||||
|
if secret_type == SecretType.SSH_KEY:
|
||||||
|
private_key_path = self.generate_private_key_path(new_secret, path_dir)
|
||||||
|
new_secret = self.generate_public_key(new_secret)
|
||||||
|
|
||||||
|
h['ssh_params'].update(self.get_ssh_params(account, new_secret, secret_type))
|
||||||
|
h['account'] = {
|
||||||
|
'name': account.name,
|
||||||
|
'username': account.username,
|
||||||
|
'secret_type': secret_type,
|
||||||
|
'secret': new_secret,
|
||||||
|
'private_key_path': private_key_path
|
||||||
|
}
|
||||||
|
if asset.platform.type == 'oracle':
|
||||||
|
h['account']['mode'] = 'sysdba' if account.privileged else None
|
||||||
|
inventory_hosts.append(h)
|
||||||
|
return inventory_hosts
|
||||||
|
|
||||||
|
def on_host_success(self, host, result):
|
||||||
|
account_info = self.name_recorder_mapper.get(host)
|
||||||
|
if not account_info:
|
||||||
|
return
|
||||||
|
|
||||||
|
account = account_info['account']
|
||||||
|
new_secret = account_info['new_secret']
|
||||||
|
if not account:
|
||||||
|
return
|
||||||
|
account.secret = new_secret
|
||||||
|
account.save(update_fields=['secret'])
|
||||||
|
account.set_connectivity(Connectivity.OK)
|
||||||
|
|
||||||
|
def on_host_error(self, host, error, result):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def on_runner_failed(self, runner, e):
|
||||||
|
logger.error("Pust account error: ", e)
|
||||||
|
|
||||||
|
def run(self, *args, **kwargs):
|
||||||
|
if self.secret_type and not self.check_secret():
|
||||||
|
return
|
||||||
|
super(ChangeSecretManager, self).run(*args, **kwargs)
|
||||||
|
|
||||||
# @classmethod
|
# @classmethod
|
||||||
# def trigger_by_asset_create(cls, asset):
|
# def trigger_by_asset_create(cls, asset):
|
||||||
# automations = PushAccountAutomation.objects.filter(
|
# automations = PushAccountAutomation.objects.filter(
|
||||||
|
|||||||
@@ -1,21 +0,0 @@
|
|||||||
- hosts: mongodb
|
|
||||||
gather_facts: no
|
|
||||||
vars:
|
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
|
||||||
|
|
||||||
tasks:
|
|
||||||
- name: "Remove account"
|
|
||||||
mongodb_user:
|
|
||||||
login_user: "{{ jms_account.username }}"
|
|
||||||
login_password: "{{ jms_account.secret }}"
|
|
||||||
login_host: "{{ jms_asset.address }}"
|
|
||||||
login_port: "{{ jms_asset.port }}"
|
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
|
||||||
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
|
||||||
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert | default('') }}"
|
|
||||||
ssl_certfile: "{{ jms_asset.secret_info.client_key | default('') }}"
|
|
||||||
connection_options:
|
|
||||||
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
|
||||||
db: "{{ jms_asset.spec_info.db_name }}"
|
|
||||||
name: "{{ account.username }}"
|
|
||||||
state: absent
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
id: remove_account_mongodb
|
|
||||||
name: "{{ 'MongoDB account remove' | trans }}"
|
|
||||||
category: database
|
|
||||||
type:
|
|
||||||
- mongodb
|
|
||||||
method: remove_account
|
|
||||||
|
|
||||||
i18n:
|
|
||||||
MongoDB account remove:
|
|
||||||
zh: 使用 Ansible 模块 mongodb 删除账号
|
|
||||||
ja: Ansible モジュール mongodb を使用してアカウントを削除する
|
|
||||||
en: Delete account using Ansible module mongodb
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
- hosts: mysql
|
|
||||||
gather_facts: no
|
|
||||||
vars:
|
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
|
||||||
|
|
||||||
tasks:
|
|
||||||
- name: "Remove account"
|
|
||||||
community.mysql.mysql_user:
|
|
||||||
login_user: "{{ jms_account.username }}"
|
|
||||||
login_password: "{{ jms_account.secret }}"
|
|
||||||
login_host: "{{ jms_asset.address }}"
|
|
||||||
login_port: "{{ jms_asset.port }}"
|
|
||||||
check_hostname: "{{ check_ssl if check_ssl else omit }}"
|
|
||||||
ca_cert: "{{ jms_asset.secret_info.ca_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_cert: "{{ jms_asset.secret_info.client_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_key: "{{ jms_asset.secret_info.client_key | default(omit) if check_ssl else omit }}"
|
|
||||||
name: "{{ account.username }}"
|
|
||||||
state: absent
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
id: remove_account_mysql
|
|
||||||
name: "{{ 'MySQL account remove' | trans }}"
|
|
||||||
category: database
|
|
||||||
type:
|
|
||||||
- mysql
|
|
||||||
- mariadb
|
|
||||||
method: remove_account
|
|
||||||
|
|
||||||
i18n:
|
|
||||||
MySQL account remove:
|
|
||||||
zh: 使用 Ansible 模块 mysql_user 删除账号
|
|
||||||
ja: Ansible モジュール mysql_user を使用してアカウントを削除します
|
|
||||||
en: Use the Ansible module mysql_user to delete the account
|
|
||||||
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
- hosts: oracle
|
|
||||||
gather_facts: no
|
|
||||||
vars:
|
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
|
||||||
|
|
||||||
tasks:
|
|
||||||
- name: "Remove account"
|
|
||||||
oracle_user:
|
|
||||||
login_user: "{{ jms_account.username }}"
|
|
||||||
login_password: "{{ jms_account.secret }}"
|
|
||||||
login_host: "{{ jms_asset.address }}"
|
|
||||||
login_port: "{{ jms_asset.port }}"
|
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
|
||||||
mode: "{{ jms_account.mode }}"
|
|
||||||
name: "{{ account.username }}"
|
|
||||||
state: absent
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
id: remove_account_oracle
|
|
||||||
name: "{{ 'Oracle account remove' | trans }}"
|
|
||||||
category: database
|
|
||||||
type:
|
|
||||||
- oracle
|
|
||||||
method: remove_account
|
|
||||||
|
|
||||||
i18n:
|
|
||||||
Oracle account remove:
|
|
||||||
zh: 使用 Python 模块 oracledb 删除账号
|
|
||||||
ja: Python モジュール oracledb を使用してアカウントを検証する
|
|
||||||
en: Using Python module oracledb to verify account
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
- hosts: postgresql
|
|
||||||
gather_facts: no
|
|
||||||
vars:
|
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
|
||||||
|
|
||||||
tasks:
|
|
||||||
- name: "Remove account"
|
|
||||||
community.postgresql.postgresql_user:
|
|
||||||
login_user: "{{ jms_account.username }}"
|
|
||||||
login_password: "{{ jms_account.secret }}"
|
|
||||||
login_host: "{{ jms_asset.address }}"
|
|
||||||
login_port: "{{ jms_asset.port }}"
|
|
||||||
db: "{{ jms_asset.spec_info.db_name }}"
|
|
||||||
name: "{{ account.username }}"
|
|
||||||
state: absent
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
id: remove_account_postgresql
|
|
||||||
name: "{{ 'PostgreSQL account remove' | trans }}"
|
|
||||||
category: database
|
|
||||||
type:
|
|
||||||
- postgresql
|
|
||||||
method: remove_account
|
|
||||||
|
|
||||||
i18n:
|
|
||||||
PostgreSQL account remove:
|
|
||||||
zh: 使用 Ansible 模块 postgresql_user 删除账号
|
|
||||||
ja: Ansible モジュール postgresql_user を使用してアカウントを削除します
|
|
||||||
en: Use the Ansible module postgresql_user to delete the account
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
- hosts: sqlserver
|
|
||||||
gather_facts: no
|
|
||||||
vars:
|
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
|
||||||
|
|
||||||
tasks:
|
|
||||||
- name: "Remove account"
|
|
||||||
community.general.mssql_script:
|
|
||||||
login_user: "{{ jms_account.username }}"
|
|
||||||
login_password: "{{ jms_account.secret }}"
|
|
||||||
login_host: "{{ jms_asset.address }}"
|
|
||||||
login_port: "{{ jms_asset.port }}"
|
|
||||||
name: "{{ jms_asset.spec_info.db_name }}"
|
|
||||||
script: "DROP USER {{ account.username }}"
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
id: remove_account_sqlserver
|
|
||||||
name: "{{ 'SQLServer account remove' | trans }}"
|
|
||||||
category: database
|
|
||||||
type:
|
|
||||||
- sqlserver
|
|
||||||
method: remove_account
|
|
||||||
|
|
||||||
i18n:
|
|
||||||
SQLServer account remove:
|
|
||||||
zh: 使用 Ansible 模块 mssql 删除账号
|
|
||||||
ja: Ansible モジュール mssql を使用してアカウントを削除する
|
|
||||||
en: Use Ansible module mssql to delete account
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
- hosts: demo
|
|
||||||
gather_facts: no
|
|
||||||
tasks:
|
|
||||||
- name: "Get user home directory path"
|
|
||||||
ansible.builtin.shell:
|
|
||||||
cmd: "getent passwd {{ account.username }} | cut -d: -f6"
|
|
||||||
register: user_home_dir
|
|
||||||
ignore_errors: yes
|
|
||||||
|
|
||||||
- name: "Check if user home directory exists"
|
|
||||||
ansible.builtin.stat:
|
|
||||||
path: "{{ user_home_dir.stdout }}"
|
|
||||||
register: home_dir
|
|
||||||
when: user_home_dir.stdout != ""
|
|
||||||
ignore_errors: yes
|
|
||||||
|
|
||||||
- name: "Rename user home directory if it exists"
|
|
||||||
ansible.builtin.command:
|
|
||||||
cmd: "mv {{ user_home_dir.stdout }} {{ user_home_dir.stdout }}.bak"
|
|
||||||
when: home_dir.stat | default(false) and user_home_dir.stdout != ""
|
|
||||||
ignore_errors: yes
|
|
||||||
|
|
||||||
- name: "Remove account"
|
|
||||||
ansible.builtin.user:
|
|
||||||
name: "{{ account.username }}"
|
|
||||||
state: absent
|
|
||||||
remove: "{{ home_dir.stat.exists }}"
|
|
||||||
when: home_dir.stat | default(false)
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
id: remove_account_posix
|
|
||||||
name: "{{ 'Posix account remove' | trans }}"
|
|
||||||
category: host
|
|
||||||
type:
|
|
||||||
- linux
|
|
||||||
- unix
|
|
||||||
method: remove_account
|
|
||||||
|
|
||||||
i18n:
|
|
||||||
Posix account remove:
|
|
||||||
zh: 使用 Ansible 模块 user 删除账号
|
|
||||||
ja: Ansible モジュール ユーザーを使用してアカウントを削除します
|
|
||||||
en: Use the Ansible module user to delete the account
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
- hosts: windows
|
|
||||||
gather_facts: no
|
|
||||||
tasks:
|
|
||||||
- name: "Remove account"
|
|
||||||
ansible.windows.win_user:
|
|
||||||
name: "{{ account.username }}"
|
|
||||||
state: absent
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
id: remove_account_windows
|
|
||||||
name: "{{ 'Windows account remove' | trans }}"
|
|
||||||
version: 1
|
|
||||||
method: remove_account
|
|
||||||
category: host
|
|
||||||
type:
|
|
||||||
- windows
|
|
||||||
|
|
||||||
i18n:
|
|
||||||
Windows account remove:
|
|
||||||
zh: 使用 Ansible 模块 win_user 删除账号
|
|
||||||
ja: Ansible モジュール win_user を使用してアカウントを削除する
|
|
||||||
en: Use the Ansible module win_user to delete an account
|
|
||||||
@@ -1,70 +0,0 @@
|
|||||||
import os
|
|
||||||
from copy import deepcopy
|
|
||||||
|
|
||||||
from django.db.models import QuerySet
|
|
||||||
|
|
||||||
from accounts.const import AutomationTypes
|
|
||||||
from accounts.models import Account
|
|
||||||
from common.utils import get_logger
|
|
||||||
from ..base.manager import AccountBasePlaybookManager
|
|
||||||
|
|
||||||
logger = get_logger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class RemoveAccountManager(AccountBasePlaybookManager):
|
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
self.host_account_mapper = {}
|
|
||||||
|
|
||||||
def prepare_runtime_dir(self):
|
|
||||||
path = super().prepare_runtime_dir()
|
|
||||||
ansible_config_path = os.path.join(path, 'ansible.cfg')
|
|
||||||
|
|
||||||
with open(ansible_config_path, 'w') as f:
|
|
||||||
f.write('[ssh_connection]\n')
|
|
||||||
f.write('ssh_args = -o ControlMaster=no -o ControlPersist=no\n')
|
|
||||||
return path
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def method_type(cls):
|
|
||||||
return AutomationTypes.remove_account
|
|
||||||
|
|
||||||
def get_gather_accounts(self, privilege_account, gather_accounts: QuerySet):
|
|
||||||
gather_account_ids = self.execution.snapshot['gather_accounts']
|
|
||||||
gather_accounts = gather_accounts.filter(id__in=gather_account_ids)
|
|
||||||
gather_accounts = gather_accounts.exclude(
|
|
||||||
username__in=[privilege_account.username, 'root', 'Administrator']
|
|
||||||
)
|
|
||||||
return gather_accounts
|
|
||||||
|
|
||||||
def host_callback(self, host, asset=None, account=None, automation=None, path_dir=None, **kwargs):
|
|
||||||
if host.get('error'):
|
|
||||||
return host
|
|
||||||
|
|
||||||
gather_accounts = asset.gatheredaccount_set.all()
|
|
||||||
gather_accounts = self.get_gather_accounts(account, gather_accounts)
|
|
||||||
|
|
||||||
inventory_hosts = []
|
|
||||||
|
|
||||||
for gather_account in gather_accounts:
|
|
||||||
h = deepcopy(host)
|
|
||||||
h['name'] += '(' + gather_account.username + ')'
|
|
||||||
self.host_account_mapper[h['name']] = (asset, gather_account)
|
|
||||||
h['account'] = {'username': gather_account.username}
|
|
||||||
inventory_hosts.append(h)
|
|
||||||
return inventory_hosts
|
|
||||||
|
|
||||||
def on_host_success(self, host, result):
|
|
||||||
tuple_asset_gather_account = self.host_account_mapper.get(host)
|
|
||||||
if not tuple_asset_gather_account:
|
|
||||||
return
|
|
||||||
asset, gather_account = tuple_asset_gather_account
|
|
||||||
try:
|
|
||||||
Account.objects.filter(
|
|
||||||
asset_id=asset.id,
|
|
||||||
username=gather_account.username
|
|
||||||
).delete()
|
|
||||||
gather_account.delete()
|
|
||||||
except Exception as e:
|
|
||||||
print(f'\033[31m Delete account {gather_account.username} failed: {e} \033[0m\n')
|
|
||||||
@@ -3,7 +3,6 @@
|
|||||||
vars:
|
vars:
|
||||||
ansible_shell_type: sh
|
ansible_shell_type: sh
|
||||||
ansible_connection: local
|
ansible_connection: local
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Verify account (pyfreerdp)
|
- name: Verify account (pyfreerdp)
|
||||||
|
|||||||
@@ -5,11 +5,9 @@ category:
|
|||||||
type:
|
type:
|
||||||
- windows
|
- windows
|
||||||
method: verify_account
|
method: verify_account
|
||||||
protocol: rdp
|
|
||||||
priority: 1
|
|
||||||
|
|
||||||
i18n:
|
i18n:
|
||||||
Windows rdp account verify:
|
Windows rdp account verify:
|
||||||
zh: '使用 Python 模块 pyfreerdp 验证账号'
|
zh: 使用 Python 模块 pyfreerdp 验证账号
|
||||||
ja: 'Python モジュール pyfreerdp を使用してアカウントを検証する'
|
ja: Python モジュール pyfreerdp を使用してアカウントを検証する
|
||||||
en: 'Using Python module pyfreerdp to verify account'
|
en: Using Python module pyfreerdp to verify account
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_connection: local
|
ansible_connection: local
|
||||||
ansible_shell_type: sh
|
|
||||||
ansible_become: false
|
ansible_become: false
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
@@ -14,10 +13,8 @@
|
|||||||
login_password: "{{ account.secret }}"
|
login_password: "{{ account.secret }}"
|
||||||
login_secret_type: "{{ account.secret_type }}"
|
login_secret_type: "{{ account.secret_type }}"
|
||||||
login_private_key_path: "{{ account.private_key_path }}"
|
login_private_key_path: "{{ account.private_key_path }}"
|
||||||
become: "{{ account.become.ansible_become | default(False) }}"
|
become: "{{ custom_become | default(False) }}"
|
||||||
become_method: "{{ account.become.ansible_become_method | default('su') }}"
|
become_method: "{{ custom_become_method | default('su') }}"
|
||||||
become_user: "{{ account.become.ansible_user | default('') }}"
|
become_user: "{{ custom_become_user | default('') }}"
|
||||||
become_password: "{{ account.become.ansible_password | default('') }}"
|
become_password: "{{ custom_become_password | default('') }}"
|
||||||
become_private_key_path: "{{ account.become.ansible_ssh_private_key_file | default(None) }}"
|
become_private_key_path: "{{ custom_become_private_key_path | default(None) }}"
|
||||||
old_ssh_version: "{{ jms_asset.old_ssh_version | default(False) }}"
|
|
||||||
gateway_args: "{{ jms_asset.ansible_ssh_common_args | default(None) }}"
|
|
||||||
|
|||||||
@@ -6,11 +6,9 @@ category:
|
|||||||
type:
|
type:
|
||||||
- all
|
- all
|
||||||
method: verify_account
|
method: verify_account
|
||||||
protocol: ssh
|
|
||||||
priority: 50
|
|
||||||
|
|
||||||
i18n:
|
i18n:
|
||||||
SSH account verify:
|
SSH account verify:
|
||||||
zh: '使用 Python 模块 paramiko 验证账号'
|
zh: 使用 Python 模块 paramiko 验证账号
|
||||||
ja: 'Python モジュール paramiko を使用してアカウントを検証する'
|
ja: Python モジュール paramiko を使用してアカウントを検証する
|
||||||
en: 'Using Python module paramiko to verify account'
|
en: Using Python module paramiko to verify account
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: mongodb
|
- hosts: mongdb
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Verify account
|
- name: Verify account
|
||||||
@@ -12,9 +12,7 @@
|
|||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
login_database: "{{ jms_asset.spec_info.db_name }}"
|
login_database: "{{ jms_asset.spec_info.db_name }}"
|
||||||
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
ssl: "{{ jms_asset.spec_info.use_ssl }}"
|
||||||
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert | default('') }}"
|
ssl_ca_certs: "{{ jms_asset.secret_info.ca_cert }}"
|
||||||
ssl_certfile: "{{ jms_asset.secret_info.client_key | default('') }}"
|
ssl_certfile: "{{ jms_asset.secret_info.client_key }}"
|
||||||
connection_options:
|
connection_options:
|
||||||
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert }}"
|
- tlsAllowInvalidHostnames: "{{ jms_asset.spec_info.allow_invalid_cert}}"
|
||||||
register: result
|
|
||||||
failed_when: not result.is_available
|
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
- hosts: mysql
|
- hosts: mysql
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
check_ssl: "{{ jms_asset.spec_info.use_ssl and not jms_asset.spec_info.allow_invalid_cert }}"
|
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Verify account
|
- name: Verify account
|
||||||
@@ -11,8 +10,4 @@
|
|||||||
login_password: "{{ account.secret }}"
|
login_password: "{{ account.secret }}"
|
||||||
login_host: "{{ jms_asset.address }}"
|
login_host: "{{ jms_asset.address }}"
|
||||||
login_port: "{{ jms_asset.port }}"
|
login_port: "{{ jms_asset.port }}"
|
||||||
check_hostname: "{{ check_ssl if check_ssl else omit }}"
|
|
||||||
ca_cert: "{{ jms_asset.secret_info.ca_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_cert: "{{ jms_asset.secret_info.client_cert | default(omit) if check_ssl else omit }}"
|
|
||||||
client_key: "{{ jms_asset.secret_info.client_key | default(omit) if check_ssl else omit }}"
|
|
||||||
filter: version
|
filter: version
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: oracle
|
- hosts: oracle
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Verify account
|
- name: Verify account
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
- hosts: postgresql
|
- hosts: postgresql
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Verify account
|
- name: Verify account
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
- hosts: sqlserver
|
- hosts: sqlserver
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
vars:
|
vars:
|
||||||
ansible_python_interpreter: /opt/py3/bin/python
|
ansible_python_interpreter: /usr/local/bin/python
|
||||||
|
|
||||||
tasks:
|
tasks:
|
||||||
- name: Verify account
|
- name: Verify account
|
||||||
|
|||||||
@@ -1,25 +1,11 @@
|
|||||||
- hosts: demo
|
- hosts: demo
|
||||||
gather_facts: no
|
gather_facts: no
|
||||||
tasks:
|
tasks:
|
||||||
- name: Verify account connectivity(Do not switch)
|
- name: Verify account connectivity
|
||||||
|
become: no
|
||||||
ansible.builtin.ping:
|
ansible.builtin.ping:
|
||||||
vars:
|
vars:
|
||||||
ansible_become: no
|
ansible_become: no
|
||||||
ansible_user: "{{ account.username }}"
|
ansible_user: "{{ account.username }}"
|
||||||
ansible_password: "{{ account.secret }}"
|
ansible_password: "{{ account.secret }}"
|
||||||
ansible_ssh_private_key_file: "{{ account.private_key_path }}"
|
ansible_ssh_private_key_file: "{{ account.private_key_path }}"
|
||||||
ansible_timeout: 30
|
|
||||||
when: not account.become.ansible_become
|
|
||||||
|
|
||||||
- name: Verify account connectivity(Switch)
|
|
||||||
ansible.builtin.ping:
|
|
||||||
vars:
|
|
||||||
ansible_become: yes
|
|
||||||
ansible_user: "{{ account.become.ansible_user }}"
|
|
||||||
ansible_password: "{{ account.become.ansible_password }}"
|
|
||||||
ansible_ssh_private_key_file: "{{ account.become.ansible_ssh_private_key_file }}"
|
|
||||||
ansible_become_method: "{{ account.become.ansible_become_method }}"
|
|
||||||
ansible_become_user: "{{ account.become.ansible_become_user }}"
|
|
||||||
ansible_become_password: "{{ account.become.ansible_become_password }}"
|
|
||||||
ansible_timeout: 30
|
|
||||||
when: account.become.ansible_become
|
|
||||||
|
|||||||
@@ -9,4 +9,3 @@
|
|||||||
vars:
|
vars:
|
||||||
ansible_user: "{{ account.username }}"
|
ansible_user: "{{ account.username }}"
|
||||||
ansible_password: "{{ account.secret }}"
|
ansible_password: "{{ account.secret }}"
|
||||||
ansible_timeout: 30
|
|
||||||
|
|||||||
@@ -42,6 +42,7 @@ class VerifyAccountManager(AccountBasePlaybookManager):
|
|||||||
if host.get('error'):
|
if host.get('error'):
|
||||||
return host
|
return host
|
||||||
|
|
||||||
|
# host['ssh_args'] = '-o ControlMaster=no -o ControlPersist=no'
|
||||||
accounts = asset.accounts.all()
|
accounts = asset.accounts.all()
|
||||||
accounts = self.get_accounts(account, accounts)
|
accounts = self.get_accounts(account, accounts)
|
||||||
inventory_hosts = []
|
inventory_hosts = []
|
||||||
@@ -51,9 +52,6 @@ class VerifyAccountManager(AccountBasePlaybookManager):
|
|||||||
h['name'] += '(' + account.username + ')'
|
h['name'] += '(' + account.username + ')'
|
||||||
self.host_account_mapper[h['name']] = account
|
self.host_account_mapper[h['name']] = account
|
||||||
secret = account.secret
|
secret = account.secret
|
||||||
if secret is None:
|
|
||||||
print(f'account {account.name} secret is None')
|
|
||||||
continue
|
|
||||||
|
|
||||||
private_key_path = None
|
private_key_path = None
|
||||||
if account.secret_type == SecretType.SSH_KEY:
|
if account.secret_type == SecretType.SSH_KEY:
|
||||||
@@ -65,9 +63,8 @@ class VerifyAccountManager(AccountBasePlaybookManager):
|
|||||||
'name': account.name,
|
'name': account.name,
|
||||||
'username': account.username,
|
'username': account.username,
|
||||||
'secret_type': account.secret_type,
|
'secret_type': account.secret_type,
|
||||||
'secret': account.escape_jinja2_syntax(secret),
|
'secret': secret,
|
||||||
'private_key_path': private_key_path,
|
'private_key_path': private_key_path
|
||||||
'become': account.get_ansible_become_auth(),
|
|
||||||
}
|
}
|
||||||
if account.platform.type == 'oracle':
|
if account.platform.type == 'oracle':
|
||||||
h['account']['mode'] = 'sysdba' if account.privileged else None
|
h['account']['mode'] = 'sysdba' if account.privileged else None
|
||||||
@@ -76,14 +73,8 @@ class VerifyAccountManager(AccountBasePlaybookManager):
|
|||||||
|
|
||||||
def on_host_success(self, host, result):
|
def on_host_success(self, host, result):
|
||||||
account = self.host_account_mapper.get(host)
|
account = self.host_account_mapper.get(host)
|
||||||
try:
|
|
||||||
account.set_connectivity(Connectivity.OK)
|
account.set_connectivity(Connectivity.OK)
|
||||||
except Exception as e:
|
|
||||||
print(f'\033[31m Update account {account.name} connectivity failed: {e} \033[0m\n')
|
|
||||||
|
|
||||||
def on_host_error(self, host, error, result):
|
def on_host_error(self, host, error, result):
|
||||||
account = self.host_account_mapper.get(host)
|
account = self.host_account_mapper.get(host)
|
||||||
try:
|
|
||||||
account.set_connectivity(Connectivity.ERR)
|
account.set_connectivity(Connectivity.ERR)
|
||||||
except Exception as e:
|
|
||||||
print(f'\033[31m Update account {account.name} connectivity failed: {e} \033[0m\n')
|
|
||||||
|
|||||||
@@ -15,7 +15,6 @@ class AliasAccount(TextChoices):
|
|||||||
INPUT = '@INPUT', _('Manual input')
|
INPUT = '@INPUT', _('Manual input')
|
||||||
USER = '@USER', _('Dynamic user')
|
USER = '@USER', _('Dynamic user')
|
||||||
ANON = '@ANON', _('Anonymous account')
|
ANON = '@ANON', _('Anonymous account')
|
||||||
SPEC = '@SPEC', _('Specified account')
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def virtual_choices(cls):
|
def virtual_choices(cls):
|
||||||
|
|||||||
@@ -4,19 +4,17 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
from assets.const import Connectivity
|
from assets.const import Connectivity
|
||||||
from common.db.fields import TreeChoices
|
from common.db.fields import TreeChoices
|
||||||
|
|
||||||
|
string_punctuation = '!#$%&()*+,-.:;<=>?@[]^_~'
|
||||||
DEFAULT_PASSWORD_LENGTH = 30
|
DEFAULT_PASSWORD_LENGTH = 30
|
||||||
DEFAULT_PASSWORD_RULES = {
|
DEFAULT_PASSWORD_RULES = {
|
||||||
'length': DEFAULT_PASSWORD_LENGTH,
|
'length': DEFAULT_PASSWORD_LENGTH,
|
||||||
'uppercase': True,
|
'symbol_set': string_punctuation
|
||||||
'lowercase': True,
|
|
||||||
'digit': True,
|
|
||||||
'symbol': True,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
'AutomationTypes', 'SecretStrategy', 'SSHKeyStrategy', 'Connectivity',
|
'AutomationTypes', 'SecretStrategy', 'SSHKeyStrategy', 'Connectivity',
|
||||||
'DEFAULT_PASSWORD_LENGTH', 'DEFAULT_PASSWORD_RULES', 'TriggerChoice',
|
'DEFAULT_PASSWORD_LENGTH', 'DEFAULT_PASSWORD_RULES', 'TriggerChoice',
|
||||||
'PushAccountActionChoice', 'AccountBackupType', 'ChangeSecretRecordStatusChoice',
|
'PushAccountActionChoice',
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
@@ -24,7 +22,6 @@ class AutomationTypes(models.TextChoices):
|
|||||||
push_account = 'push_account', _('Push account')
|
push_account = 'push_account', _('Push account')
|
||||||
change_secret = 'change_secret', _('Change secret')
|
change_secret = 'change_secret', _('Change secret')
|
||||||
verify_account = 'verify_account', _('Verify account')
|
verify_account = 'verify_account', _('Verify account')
|
||||||
remove_account = 'remove_account', _('Remove account')
|
|
||||||
gather_accounts = 'gather_accounts', _('Gather accounts')
|
gather_accounts = 'gather_accounts', _('Gather accounts')
|
||||||
verify_gateway_account = 'verify_gateway_account', _('Verify gateway account')
|
verify_gateway_account = 'verify_gateway_account', _('Verify gateway account')
|
||||||
|
|
||||||
@@ -44,8 +41,8 @@ class AutomationTypes(models.TextChoices):
|
|||||||
|
|
||||||
|
|
||||||
class SecretStrategy(models.TextChoices):
|
class SecretStrategy(models.TextChoices):
|
||||||
custom = 'specific', _('Specific secret')
|
custom = 'specific', _('Specific password')
|
||||||
random = 'random', _('Random generate')
|
random = 'random', _('Random')
|
||||||
|
|
||||||
|
|
||||||
class SSHKeyStrategy(models.TextChoices):
|
class SSHKeyStrategy(models.TextChoices):
|
||||||
@@ -96,16 +93,3 @@ class TriggerChoice(models.TextChoices, TreeChoices):
|
|||||||
class PushAccountActionChoice(models.TextChoices):
|
class PushAccountActionChoice(models.TextChoices):
|
||||||
create_and_push = 'create_and_push', _('Create and push')
|
create_and_push = 'create_and_push', _('Create and push')
|
||||||
only_create = 'only_create', _('Only create')
|
only_create = 'only_create', _('Only create')
|
||||||
|
|
||||||
|
|
||||||
class AccountBackupType(models.TextChoices):
|
|
||||||
"""Backup type"""
|
|
||||||
email = 'email', _('Email')
|
|
||||||
# 目前只支持sftp方式
|
|
||||||
object_storage = 'object_storage', _('SFTP')
|
|
||||||
|
|
||||||
|
|
||||||
class ChangeSecretRecordStatusChoice(models.TextChoices):
|
|
||||||
failed = 'failed', _('Failed')
|
|
||||||
success = 'success', _('Success')
|
|
||||||
pending = 'pending', _('Pending')
|
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ from django_filters import rest_framework as drf_filters
|
|||||||
|
|
||||||
from assets.models import Node
|
from assets.models import Node
|
||||||
from common.drf.filters import BaseFilterSet
|
from common.drf.filters import BaseFilterSet
|
||||||
from .models import Account, GatheredAccount, ChangeSecretRecord
|
from .models import Account, GatheredAccount
|
||||||
|
|
||||||
|
|
||||||
class AccountFilterSet(BaseFilterSet):
|
class AccountFilterSet(BaseFilterSet):
|
||||||
@@ -51,8 +51,6 @@ class AccountFilterSet(BaseFilterSet):
|
|||||||
|
|
||||||
class GatheredAccountFilterSet(BaseFilterSet):
|
class GatheredAccountFilterSet(BaseFilterSet):
|
||||||
node_id = drf_filters.CharFilter(method='filter_nodes')
|
node_id = drf_filters.CharFilter(method='filter_nodes')
|
||||||
asset_id = drf_filters.CharFilter(field_name='asset_id', lookup_expr='exact')
|
|
||||||
asset_name = drf_filters.CharFilter(field_name='asset__name', lookup_expr='icontains')
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def filter_nodes(queryset, name, value):
|
def filter_nodes(queryset, name, value):
|
||||||
@@ -60,14 +58,4 @@ class GatheredAccountFilterSet(BaseFilterSet):
|
|||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = GatheredAccount
|
model = GatheredAccount
|
||||||
fields = ['id', 'username']
|
fields = ['id', 'asset_id', 'username']
|
||||||
|
|
||||||
|
|
||||||
class ChangeSecretRecordFilterSet(BaseFilterSet):
|
|
||||||
asset_name = drf_filters.CharFilter(field_name='asset__name', lookup_expr='icontains')
|
|
||||||
account_username = drf_filters.CharFilter(field_name='account__username', lookup_expr='icontains')
|
|
||||||
execution_id = drf_filters.CharFilter(field_name='execution_id', lookup_expr='exact')
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = ChangeSecretRecord
|
|
||||||
fields = ['id', 'status', 'asset_id', 'execution']
|
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ class Migration(migrations.Migration):
|
|||||||
'verbose_name': 'Automation execution',
|
'verbose_name': 'Automation execution',
|
||||||
'verbose_name_plural': 'Automation executions',
|
'verbose_name_plural': 'Automation executions',
|
||||||
'permissions': [('view_changesecretexecution', 'Can view change secret execution'),
|
'permissions': [('view_changesecretexecution', 'Can view change secret execution'),
|
||||||
('add_changesecretexecution', 'Can add change secret execution'),
|
('add_changesecretexection', 'Can add change secret execution'),
|
||||||
('view_gatheraccountsexecution', 'Can view gather accounts execution'),
|
('view_gatheraccountsexecution', 'Can view gather accounts execution'),
|
||||||
('add_gatheraccountsexecution', 'Can add gather accounts execution')],
|
('add_gatheraccountsexecution', 'Can add gather accounts execution')],
|
||||||
'proxy': True,
|
'proxy': True,
|
||||||
@@ -116,7 +116,7 @@ class Migration(migrations.Migration):
|
|||||||
('new_secret', common.db.fields.EncryptTextField(blank=True, null=True, verbose_name='New secret')),
|
('new_secret', common.db.fields.EncryptTextField(blank=True, null=True, verbose_name='New secret')),
|
||||||
('date_started', models.DateTimeField(blank=True, null=True, verbose_name='Date started')),
|
('date_started', models.DateTimeField(blank=True, null=True, verbose_name='Date started')),
|
||||||
('date_finished', models.DateTimeField(blank=True, null=True, verbose_name='Date finished')),
|
('date_finished', models.DateTimeField(blank=True, null=True, verbose_name='Date finished')),
|
||||||
('status', models.CharField(default='pending', max_length=16, verbose_name='Status')),
|
('status', models.CharField(default='pending', max_length=16)),
|
||||||
('error', models.TextField(blank=True, null=True, verbose_name='Error')),
|
('error', models.TextField(blank=True, null=True, verbose_name='Error')),
|
||||||
('account',
|
('account',
|
||||||
models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to='accounts.account')),
|
models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to='accounts.account')),
|
||||||
@@ -184,7 +184,7 @@ class Migration(migrations.Migration):
|
|||||||
migrations.AlterModelOptions(
|
migrations.AlterModelOptions(
|
||||||
name='automationexecution',
|
name='automationexecution',
|
||||||
options={'permissions': [('view_changesecretexecution', 'Can view change secret execution'),
|
options={'permissions': [('view_changesecretexecution', 'Can view change secret execution'),
|
||||||
('add_changesecretexecution', 'Can add change secret execution'),
|
('add_changesecretexection', 'Can add change secret execution'),
|
||||||
('view_gatheraccountsexecution', 'Can view gather accounts execution'),
|
('view_gatheraccountsexecution', 'Can view gather accounts execution'),
|
||||||
('add_gatheraccountsexecution', 'Can add gather accounts execution'),
|
('add_gatheraccountsexecution', 'Can add gather accounts execution'),
|
||||||
('view_pushaccountexecution', 'Can view push account execution'),
|
('view_pushaccountexecution', 'Can view push account execution'),
|
||||||
|
|||||||
@@ -13,11 +13,11 @@ class Migration(migrations.Migration):
|
|||||||
migrations.AlterField(
|
migrations.AlterField(
|
||||||
model_name='changesecretautomation',
|
model_name='changesecretautomation',
|
||||||
name='secret_strategy',
|
name='secret_strategy',
|
||||||
field=models.CharField(choices=[('specific', 'Specific secret'), ('random', 'Random generate')], default='specific', max_length=16, verbose_name='Secret strategy'),
|
field=models.CharField(choices=[('specific', 'Specific password'), ('random', 'Random')], default='specific', max_length=16, verbose_name='Secret strategy'),
|
||||||
),
|
),
|
||||||
migrations.AlterField(
|
migrations.AlterField(
|
||||||
model_name='pushaccountautomation',
|
model_name='pushaccountautomation',
|
||||||
name='secret_strategy',
|
name='secret_strategy',
|
||||||
field=models.CharField(choices=[('specific', 'Specific secret'), ('random', 'Random generate')], default='specific', max_length=16, verbose_name='Secret strategy'),
|
field=models.CharField(choices=[('specific', 'Specific password'), ('random', 'Random')], default='specific', max_length=16, verbose_name='Secret strategy'),
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ from django.db import migrations
|
|||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
dependencies = [
|
dependencies = [
|
||||||
('accounts', '0006_gatheredaccount'),
|
('accounts', '0006_gatheredaccount'),
|
||||||
]
|
]
|
||||||
@@ -11,13 +12,6 @@ class Migration(migrations.Migration):
|
|||||||
operations = [
|
operations = [
|
||||||
migrations.AlterModelOptions(
|
migrations.AlterModelOptions(
|
||||||
name='account',
|
name='account',
|
||||||
options={'permissions': [
|
options={'permissions': [('view_accountsecret', 'Can view asset account secret'), ('view_historyaccount', 'Can view asset history account'), ('view_historyaccountsecret', 'Can view asset history account secret'), ('verify_account', 'Can verify account'), ('push_account', 'Can push account')], 'verbose_name': 'Account'},
|
||||||
('view_accountsecret', 'Can view asset account secret'),
|
|
||||||
('view_historyaccount', 'Can view asset history account'),
|
|
||||||
('view_historyaccountsecret', 'Can view asset history account secret'),
|
|
||||||
('verify_account', 'Can verify account'),
|
|
||||||
('push_account', 'Can push account'),
|
|
||||||
('remove_account', 'Can remove account'),
|
|
||||||
], 'verbose_name': 'Account'},
|
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user