├── logs
└── .gitkeep
├── common
├── __init__.py
├── utils
│ ├── __init__.py
│ ├── global_info.py
│ ├── timer.py
│ ├── wx_api.py
│ ├── extend_json_encoder.py
│ ├── permission.py
│ ├── aes_decryptor.py
│ ├── const.py
│ └── aliyun_sdk.py
├── middleware
│ ├── __init__.py
│ ├── exception_logging_middleware.py
│ └── check_login_middleware.py
├── templates
│ ├── error.html
│ └── errors
│ │ ├── 400.html
│ │ ├── 403.html
│ │ ├── 500.html
│ │ └── 404.html
├── static
│ ├── bootstrap-editable
│ │ └── img
│ │ │ ├── clear.png
│ │ │ └── loading.gif
│ ├── font-awesome
│ │ └── fonts
│ │ │ ├── FontAwesome.otf
│ │ │ ├── fontawesome-webfont.eot
│ │ │ ├── fontawesome-webfont.ttf
│ │ │ ├── fontawesome-webfont.woff
│ │ │ └── fontawesome-webfont.woff2
│ ├── bootstrap-fileinput
│ │ ├── img
│ │ │ └── loading.gif
│ │ └── js
│ │ │ └── locales
│ │ │ └── zh.js
│ ├── bootstrap
│ │ └── fonts
│ │ │ ├── glyphicons-halflings-regular.eot
│ │ │ ├── glyphicons-halflings-regular.ttf
│ │ │ ├── glyphicons-halflings-regular.woff
│ │ │ └── glyphicons-halflings-regular.woff2
│ ├── ace
│ │ ├── mode-text.js
│ │ ├── snippets
│ │ │ ├── text.js
│ │ │ ├── pgsql.js
│ │ │ ├── mysql.js
│ │ │ ├── sql.js
│ │ │ └── sqlserver.js
│ │ ├── mode-sql.js
│ │ ├── theme-github.js
│ │ ├── theme-textmate.js
│ │ ├── theme-sqlserver.js
│ │ └── theme-mongodb.js
│ ├── dist
│ │ ├── css
│ │ │ └── login.css
│ │ └── js
│ │ │ └── formatter.js
│ ├── datetimepicker
│ │ └── js
│ │ │ └── bootstrap-datetimepicker.zh-CN.js
│ ├── sb-admin-2
│ │ └── js
│ │ │ ├── sb-admin-2.min.js
│ │ │ └── sb-admin-2.js
│ ├── bootstrap-select
│ │ └── js
│ │ │ └── i18n
│ │ │ ├── defaults-zh_CN.min.js
│ │ │ ├── defaults-zh_CN.js.map
│ │ │ └── defaults-zh_CN.js
│ ├── bootstrap-table
│ │ ├── export-libs
│ │ │ ├── FileSaver
│ │ │ │ ├── LICENSE.md
│ │ │ │ └── FileSaver.min.js
│ │ │ ├── html2canvas
│ │ │ │ └── LICENSE
│ │ │ ├── pdfmake
│ │ │ │ └── LICENSE
│ │ │ ├── es6-promise
│ │ │ │ └── LICENSE
│ │ │ ├── jsPDF
│ │ │ │ └── MIT-LICENSE.txt
│ │ │ └── jsPDF-AutoTable
│ │ │ │ └── LICENSE.txt
│ │ ├── js
│ │ │ └── bootstrap-table-zh-CN.min.js
│ │ └── extensions
│ │ │ └── editable
│ │ │ └── bootstrap-table-editable.min.js
│ └── metisMenu
│ │ └── css
│ │ └── metisMenu.min.css
├── storage.py
├── views.py
├── workflow.py
└── config.py
├── sql
├── __init__.py
├── plugins
│ ├── __init__.py
│ ├── sqladvisor.py
│ ├── schemasync.py
│ ├── pt_archiver.py
│ ├── plugin.py
│ └── binglog2sql.py
├── static
│ ├── css
│ │ ├── .gitkeep
│ │ └── basic.css
│ ├── js
│ │ └── .gitkeep
│ └── pics
│ │ └── .gitkeep
├── utils
│ ├── __init__.py
│ ├── tasks.py
│ ├── resource_group.py
│ ├── execute_sql.py
│ └── sql_utils.py
├── templatetags
│ ├── __init__.py
│ └── format_tags.py
├── locale
│ ├── en
│ │ └── LC_MESSAGES
│ │ │ ├── django.mo
│ │ │ ├── djangojs.mo
│ │ │ ├── django.po
│ │ │ └── djangojs.po
│ └── zh_Hans
│ │ └── LC_MESSAGES
│ │ ├── django.mo
│ │ ├── djangojs.mo
│ │ ├── django.po
│ │ └── djangojs.po
├── templates
│ └── dbaprinciples.html
├── completer
│ ├── __init__.py
│ ├── tests.py
│ └── mysql.py
├── fixtures
│ └── auth_group.sql
├── engines
│ └── mongo.py
└── sql_analyze.py
├── CONTRIBUTING.md
├── sql_api
├── __init__.py
├── apps.py
├── urls.py
└── tests.py
├── downloads
├── binlog2sql
│ └── .gitkeep
└── schemasync
│ └── .gitkeep
├── src
├── docker-compose
│ ├── archery
│ │ ├── downloads
│ │ │ ├── log
│ │ │ │ └── .gitkeep
│ │ │ ├── binlog2sql
│ │ │ │ └── .gitkeep
│ │ │ └── schemasync
│ │ │ │ └── .gitkeep
│ │ ├── docs.md
│ │ └── soar.yaml
│ ├── inception
│ │ └── inc.cnf
│ ├── mysql
│ │ └── my.cnf
│ └── docker-compose.yml
├── charts
│ ├── Chart.yaml
│ ├── charts
│ │ ├── inception
│ │ │ ├── Chart.yaml
│ │ │ ├── .helmignore
│ │ │ ├── templates
│ │ │ │ ├── configMap.yaml
│ │ │ │ ├── service.yaml
│ │ │ │ ├── _helpers.tpl
│ │ │ │ ├── ingress.yaml
│ │ │ │ ├── NOTES.txt
│ │ │ │ └── deployment.yaml
│ │ │ └── values.yaml
│ │ └── goinception
│ │ │ ├── Chart.yaml
│ │ │ ├── .helmignore
│ │ │ └── templates
│ │ │ ├── configMap.yaml
│ │ │ ├── service.yaml
│ │ │ ├── _helpers.tpl
│ │ │ ├── ingress.yaml
│ │ │ ├── NOTES.txt
│ │ │ └── deployment.yaml
│ ├── requirements.yaml
│ ├── templates
│ │ ├── configMap.yaml
│ │ ├── service.yaml
│ │ ├── _helpers.tpl
│ │ ├── ingress.yaml
│ │ ├── NOTES.txt
│ │ └── deployment.yaml
│ └── README.md
├── init_sql
│ ├── v1.3.6_v1.3.7.sql
│ ├── v1.6.5_v1.6.6.sql
│ ├── v1.6.0_v1.6.1.sql
│ ├── v1.7.7_v1.7.8.sql
│ ├── v1.6.2_v1.6.3.sql
│ ├── v1.6.6_v1.6.7.sql
│ ├── v1.7.1_v1.7.2.sql
│ ├── v1.3.8_v1.4.0.sql
│ ├── v1.7.2_v1.7.3.sql
│ ├── v1.3.0_v1.3.2.sql
│ ├── v1.7.0_v1.7.1.sql
│ ├── v1.4.2_v1.4.3.sql
│ ├── v1.7.6_v1.7.7.sql
│ ├── del_permissions.sql
│ ├── v1.0_v1.1.0.sql
│ ├── v1.2.0_v1.3.0.sql
│ ├── v1.6.1_v1.6.2.sql
│ ├── v1.6.7_v1.7.0.sql
│ ├── v1.5.3_v1.6.0.sql
│ ├── v1.7.4_v1.7.5.sql
│ └── v1.4.5_v1.5.0.sql
├── docker
│ ├── supervisord.conf
│ ├── Dockerfile
│ ├── startup.sh
│ ├── nginx.conf
│ └── Dockerfile-base
├── plugins
│ └── soar.yaml
└── script
│ └── analysis_slow_query.sh
├── archery
├── __init__.py
├── wsgi.py
└── urls.py
├── .gitattributes
├── debug.sh
├── .coveragerc
├── startup.sh
├── .gitignore
├── .github
├── ISSUE_TEMPLATE
│ ├── feature_request.md
│ ├── custom.md
│ └── bug.md
├── config.yml
├── release-drafter.yml
└── stale.yml
├── manage.py
├── requirements.txt
├── .travis.yml
├── supervisord.conf
├── admin.sh
└── CODE_OF_CONDUCT.md
/logs/.gitkeep:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/common/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/sql/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/CONTRIBUTING.md:
--------------------------------------------------------------------------------
1 |
2 |
--------------------------------------------------------------------------------
/sql/plugins/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/sql/static/css/.gitkeep:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/sql/static/js/.gitkeep:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/sql/utils/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/sql_api/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/common/utils/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/sql/static/pics/.gitkeep:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/sql/templatetags/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/common/middleware/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/downloads/binlog2sql/.gitkeep:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/downloads/schemasync/.gitkeep:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/src/docker-compose/archery/downloads/log/.gitkeep:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/src/docker-compose/archery/downloads/binlog2sql/.gitkeep:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/src/docker-compose/archery/downloads/schemasync/.gitkeep:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/src/docker-compose/archery/docs.md:
--------------------------------------------------------------------------------
1 | # 请替换docs目录下的Markdown文件
2 |
--------------------------------------------------------------------------------
/archery/__init__.py:
--------------------------------------------------------------------------------
1 | version = (1, 7, 8)
2 | display_version = '.'.join(str(i) for i in version)
3 |
--------------------------------------------------------------------------------
/.gitattributes:
--------------------------------------------------------------------------------
1 | *.js linguist-language=python
2 | *.css linguist-language=python
3 | *.html linguist-language=python
4 |
--------------------------------------------------------------------------------
/sql/locale/en/LC_MESSAGES/django.mo:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/sql/locale/en/LC_MESSAGES/django.mo
--------------------------------------------------------------------------------
/sql/locale/en/LC_MESSAGES/djangojs.mo:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/sql/locale/en/LC_MESSAGES/djangojs.mo
--------------------------------------------------------------------------------
/sql_api/apps.py:
--------------------------------------------------------------------------------
1 | from django.apps import AppConfig
2 |
3 |
4 | class SqlApi2Config(AppConfig):
5 | name = 'sql_api'
6 |
--------------------------------------------------------------------------------
/sql/locale/zh_Hans/LC_MESSAGES/django.mo:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/sql/locale/zh_Hans/LC_MESSAGES/django.mo
--------------------------------------------------------------------------------
/common/templates/error.html:
--------------------------------------------------------------------------------
1 | {% extends "base.html" %}
2 |
3 | {% block content %}
4 |
{{errMsg}}
5 | {% endblock content %}
6 |
--------------------------------------------------------------------------------
/sql/locale/zh_Hans/LC_MESSAGES/djangojs.mo:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/sql/locale/zh_Hans/LC_MESSAGES/djangojs.mo
--------------------------------------------------------------------------------
/src/charts/Chart.yaml:
--------------------------------------------------------------------------------
1 | aiVersion: v1
2 | appVersion: "1.0"
3 | description: A Helm chart for Kubernetes
4 | name: archery
5 | version: 0.1.0
6 |
--------------------------------------------------------------------------------
/src/init_sql/v1.3.6_v1.3.7.sql:
--------------------------------------------------------------------------------
1 | -- 修改阿里云配置信息表
2 | alter table aliyun_rds_config add is_enable tinyint not null default 0 comment '是否启用';
3 |
--------------------------------------------------------------------------------
/common/static/bootstrap-editable/img/clear.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/bootstrap-editable/img/clear.png
--------------------------------------------------------------------------------
/common/static/bootstrap-editable/img/loading.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/bootstrap-editable/img/loading.gif
--------------------------------------------------------------------------------
/common/static/font-awesome/fonts/FontAwesome.otf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/font-awesome/fonts/FontAwesome.otf
--------------------------------------------------------------------------------
/common/static/bootstrap-fileinput/img/loading.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/bootstrap-fileinput/img/loading.gif
--------------------------------------------------------------------------------
/common/templates/errors/400.html:
--------------------------------------------------------------------------------
1 | {% extends "base.html" %}
2 |
3 | {% block content %}
4 | Bad Request (400)
5 | {% endblock content %}
6 |
--------------------------------------------------------------------------------
/common/templates/errors/403.html:
--------------------------------------------------------------------------------
1 | {% extends "base.html" %}
2 |
3 | {% block content %}
4 | 403 Forbidden
5 | {% endblock content %}
6 |
--------------------------------------------------------------------------------
/common/templates/errors/500.html:
--------------------------------------------------------------------------------
1 | {% extends "base.html" %}
2 |
3 | {% block content %}
4 | Server Error (500)
5 | {% endblock content %}
6 |
--------------------------------------------------------------------------------
/debug.sh:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 |
3 | nohup python3 manage.py runserver 0.0.0.0:9123 --insecure &
4 |
5 | # 启动Django Q cluster
6 | nohup python3 manage.py qcluster &
7 |
--------------------------------------------------------------------------------
/src/charts/charts/inception/Chart.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: v1
2 | appVersion: "1.0"
3 | description: A Helm chart for Kubernetes
4 | name: inception
5 | version: 0.1.0
6 |
--------------------------------------------------------------------------------
/.coveragerc:
--------------------------------------------------------------------------------
1 | [run]
2 | include =
3 | common*
4 | sql*
5 | sql_api*
6 | omit =
7 | src*
8 | downloads*
9 | sql/migrations/*
10 |
11 |
--------------------------------------------------------------------------------
/common/static/font-awesome/fonts/fontawesome-webfont.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/font-awesome/fonts/fontawesome-webfont.eot
--------------------------------------------------------------------------------
/common/static/font-awesome/fonts/fontawesome-webfont.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/font-awesome/fonts/fontawesome-webfont.ttf
--------------------------------------------------------------------------------
/common/static/font-awesome/fonts/fontawesome-webfont.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/font-awesome/fonts/fontawesome-webfont.woff
--------------------------------------------------------------------------------
/common/static/font-awesome/fonts/fontawesome-webfont.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/font-awesome/fonts/fontawesome-webfont.woff2
--------------------------------------------------------------------------------
/src/charts/charts/goinception/Chart.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: v1
2 | appVersion: "1.0"
3 | description: A Helm chart for Kubernetes
4 | name: goinception
5 | version: 0.1.0
6 |
--------------------------------------------------------------------------------
/startup.sh:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 |
3 | # 收集所有的静态文件到STATIC_ROOT
4 | python3 manage.py collectstatic -v0 --noinput
5 |
6 | # 启动服务
7 | supervisord -c supervisord.conf
8 |
9 |
--------------------------------------------------------------------------------
/common/static/bootstrap/fonts/glyphicons-halflings-regular.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/bootstrap/fonts/glyphicons-halflings-regular.eot
--------------------------------------------------------------------------------
/common/static/bootstrap/fonts/glyphicons-halflings-regular.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/bootstrap/fonts/glyphicons-halflings-regular.ttf
--------------------------------------------------------------------------------
/common/static/bootstrap/fonts/glyphicons-halflings-regular.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/bootstrap/fonts/glyphicons-halflings-regular.woff
--------------------------------------------------------------------------------
/common/static/bootstrap/fonts/glyphicons-halflings-regular.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ctfang/Archery/master/common/static/bootstrap/fonts/glyphicons-halflings-regular.woff2
--------------------------------------------------------------------------------
/src/init_sql/v1.6.5_v1.6.6.sql:
--------------------------------------------------------------------------------
1 | -- 增加查询语句收藏/置顶功能
2 | alter table query_log
3 | add favorite tinyint not null default 0 comment '是否收藏',
4 | add alias varchar(100) not null default '' comment '语句标识/别名';
5 |
--------------------------------------------------------------------------------
/common/templates/errors/404.html:
--------------------------------------------------------------------------------
1 | {% extends "base.html" %}
2 |
3 | {% block content %}
4 | Not Found
5 | The requested resource was not found on this server.
6 | {% endblock content %}
7 |
--------------------------------------------------------------------------------
/sql_api/urls.py:
--------------------------------------------------------------------------------
1 | from django.urls import path
2 | from sql_api import views
3 |
4 | urlpatterns = [
5 | path('info', views.info),
6 | path('debug', views.debug),
7 | path('do_once/mirage', views.mirage)
8 | ]
9 |
--------------------------------------------------------------------------------
/src/init_sql/v1.6.0_v1.6.1.sql:
--------------------------------------------------------------------------------
1 | -- 扩充数据库类型字段
2 | alter table sql_instance modify db_type varchar(20) not null default '' comment '数据库类型';
3 | alter table param_template modify db_type varchar(20) not null default '' comment '数据库类型';
4 |
--------------------------------------------------------------------------------
/src/init_sql/v1.7.7_v1.7.8.sql:
--------------------------------------------------------------------------------
1 | -- 增加工单备注长度
2 | alter table workflow_audit_detail modify remark varchar(1000) NOT NULL DEFAULT '' COMMENT '审核备注';
3 | alter table workflow_log modify operation_info varchar(1000) NOT NULL DEFAULT '' COMMENT '操作信息';
4 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | *.pyc
2 | *.swp
3 | *.lock
4 | *.log
5 | .idea/
6 | .DS_Store
7 | archery/settings.py.github
8 | archery/settings.py.dev
9 | archery/settings_dev.py
10 | sql/migrations/
11 | venv
12 | env
13 | sonar-project.properties
14 | .scannerwork
--------------------------------------------------------------------------------
/src/init_sql/v1.6.2_v1.6.3.sql:
--------------------------------------------------------------------------------
1 | -- 增加数据字典权限
2 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
3 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 数据字典', @content_type_id, 'menu_data_dictionary');
4 |
--------------------------------------------------------------------------------
/src/init_sql/v1.6.6_v1.6.7.sql:
--------------------------------------------------------------------------------
1 | -- 增加实例用户列表字典权限
2 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
3 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 实例用户列表', @content_type_id, 'menu_instance_user');
4 |
--------------------------------------------------------------------------------
/src/init_sql/v1.7.1_v1.7.2.sql:
--------------------------------------------------------------------------------
1 | -- 增加导出数据字典权限
2 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
3 | INSERT IGNORE INTO auth_permission (name, content_type_id, codename) VALUES ('导出数据字典', @content_type_id, 'data_dictionary_export');
4 |
--------------------------------------------------------------------------------
/.github/ISSUE_TEMPLATE/feature_request.md:
--------------------------------------------------------------------------------
1 | ---
2 | name: 功能建议
3 | about: 如果你对Archery有什么功能或者改进建议,请在此反馈,也欢迎直接向我们提交pr
4 | title: '[ 功能建议 ]'
5 | labels: ''
6 | assignees: ''
7 |
8 | ---
9 |
10 | ### 目前遇到的问题/使用障碍
11 |
12 | ### 希望如何解决/实现它
13 |
14 | ### 其他信息
15 | 如果有其他类似的产品功能或者图片信息,可在此提交
16 |
--------------------------------------------------------------------------------
/manage.py:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env python
2 | import os
3 | import sys
4 |
5 | if __name__ == "__main__":
6 | os.environ.setdefault("DJANGO_SETTINGS_MODULE", "archery.settings")
7 |
8 | from django.core.management import execute_from_command_line
9 |
10 | execute_from_command_line(sys.argv)
11 |
--------------------------------------------------------------------------------
/sql/templatetags/format_tags.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | from django import template
3 | from django.utils.safestring import mark_safe
4 |
5 | register = template.Library()
6 |
7 |
8 | # 替换换行符
9 | @register.simple_tag
10 | def format_str(str):
11 | # 换行
12 | return mark_safe(str.replace(',', '
').replace('\n', '
'))
13 |
--------------------------------------------------------------------------------
/common/static/ace/mode-text.js:
--------------------------------------------------------------------------------
1 |
2 | ; (function() {
3 | window.require(["ace/mode/text"], function(m) {
4 | if (typeof module == "object" && typeof exports == "object" && module) {
5 | module.exports = m;
6 | }
7 | });
8 | })();
9 |
--------------------------------------------------------------------------------
/common/middleware/exception_logging_middleware.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | import logging
3 | from django.utils.deprecation import MiddlewareMixin
4 |
5 | logger = logging.getLogger('default')
6 |
7 |
8 | class ExceptionLoggingMiddleware(MiddlewareMixin):
9 | def process_exception(self, request, exception):
10 | import traceback
11 | logger.error(traceback.format_exc())
12 |
--------------------------------------------------------------------------------
/.github/config.yml:
--------------------------------------------------------------------------------
1 | # Configuration for request-info - https://github.com/behaviorbot/request-info
2 |
3 | # *Required* Comment to reply with
4 | requestInfoReplyComment: >
5 | 你好!感谢你反馈的问题/bug,但是你的描述好像是空的,我们需要你完整的信息,这样才能帮你解决问题
6 | 如果不知道怎么写,在新建issue的时候有若干个模板可供选择,祝好!
7 |
8 | # *OPTIONAL* Label to be added to Issues and Pull Requests with insufficient information given
9 | requestInfoLabelToAdd: 请提供更多信息
10 |
--------------------------------------------------------------------------------
/archery/wsgi.py:
--------------------------------------------------------------------------------
1 | """
2 | WSGI config for archery project.
3 |
4 | It exposes the WSGI callable as a module-level variable named ``application``.
5 |
6 | For more information on this file, see
7 | https://docs.djangoproject.com/en/1.8/howto/deployment/wsgi/
8 | """
9 |
10 | import os
11 |
12 | from django.core.wsgi import get_wsgi_application
13 |
14 | os.environ.setdefault("DJANGO_SETTINGS_MODULE", "archery.settings")
15 |
16 | application = get_wsgi_application()
17 |
--------------------------------------------------------------------------------
/src/charts/requirements.yaml:
--------------------------------------------------------------------------------
1 | dependencies:
2 | - name: redis
3 | version: 1.1.15
4 | repository: https://kubernetes.oss-cn-hangzhou.aliyuncs.com/charts
5 | # condition: redis.usePassword="",redis.persistence.enabled=false
6 | - name: mysql
7 | version: 0.3.5
8 | repository: https://kubernetes.oss-cn-hangzhou.aliyuncs.com/charts
9 | # condition: mysql.mysqlRootPassword="archery",mysql.mysqlDatabase="archery",persistence.enabled=false
10 |
--------------------------------------------------------------------------------
/src/charts/charts/inception/.helmignore:
--------------------------------------------------------------------------------
1 | # Patterns to ignore when building packages.
2 | # This supports shell glob matching, relative path matching, and
3 | # negation (prefixed with !). Only one pattern per line.
4 | .DS_Store
5 | # Common VCS dirs
6 | .git/
7 | .gitignore
8 | .bzr/
9 | .bzrignore
10 | .hg/
11 | .hgignore
12 | .svn/
13 | # Common backup files
14 | *.swp
15 | *.bak
16 | *.tmp
17 | *~
18 | # Various IDEs
19 | .project
20 | .idea/
21 | *.tmproj
22 | .vscode/
23 |
--------------------------------------------------------------------------------
/src/init_sql/v1.3.8_v1.4.0.sql:
--------------------------------------------------------------------------------
1 | -- 新增登录失败信息
2 | alter table sql_users
3 | add failed_login_count tinyint not null default 0 comment '失败计数',
4 | add last_login_failed_at timestamp comment '上次失败登录时间';
5 |
6 | -- 修改资源表名
7 | rename table sql_group to resource_group;
8 | rename table sql_group_relations to resource_group_relations;
9 |
10 | -- 使用django_q替换django_apscheduler
11 | drop table django_apscheduler_djangojobexecution;
12 | drop table django_apscheduler_djangojob;
13 |
--------------------------------------------------------------------------------
/src/charts/charts/goinception/.helmignore:
--------------------------------------------------------------------------------
1 | # Patterns to ignore when building packages.
2 | # This supports shell glob matching, relative path matching, and
3 | # negation (prefixed with !). Only one pattern per line.
4 | .DS_Store
5 | # Common VCS dirs
6 | .git/
7 | .gitignore
8 | .bzr/
9 | .bzrignore
10 | .hg/
11 | .hgignore
12 | .svn/
13 | # Common backup files
14 | *.swp
15 | *.bak
16 | *.tmp
17 | *~
18 | # Various IDEs
19 | .project
20 | .idea/
21 | *.tmproj
22 | .vscode/
23 |
--------------------------------------------------------------------------------
/archery/urls.py:
--------------------------------------------------------------------------------
1 | from django.urls import include, path
2 | from django.contrib import admin
3 | from common import views
4 |
5 | urlpatterns = [
6 | path('admin/', admin.site.urls),
7 | path('api/', include(('sql_api.urls', 'sql_api'), namespace="sql_api")),
8 | path('', include(('sql.urls', 'sql'), namespace="sql")),
9 | ]
10 |
11 | handler400 = views.bad_request
12 | handler403 = views.permission_denied
13 | handler404 = views.page_not_found
14 | handler500 = views.server_error
15 |
--------------------------------------------------------------------------------
/common/static/ace/snippets/text.js:
--------------------------------------------------------------------------------
1 | define("ace/snippets/text",["require","exports","module"],function(e,t,n){"use strict";t.snippetText=undefined,t.scope="text"}); (function() {
2 | window.require(["ace/snippets/text"], function(m) {
3 | if (typeof module == "object" && typeof exports == "object" && module) {
4 | module.exports = m;
5 | }
6 | });
7 | })();
8 |
--------------------------------------------------------------------------------
/src/docker/supervisord.conf:
--------------------------------------------------------------------------------
1 | [unix_http_server]
2 | file=/tmp/supervisor.sock
3 |
4 | [supervisord]
5 | logfile=logs/supervisord.log
6 | nodaemon=false
7 |
8 | [supervisorctl]
9 | serverurl=unix:///tmp/supervisor.sock
10 |
11 | [rpcinterface:supervisor]
12 | supervisor.rpcinterface_factory=supervisor.rpcinterface:make_main_rpcinterface
13 |
14 | [program:qcluster]
15 | command=python manage.py qcluster
16 | autorestart=true
17 | stopasgroup=true
18 | killasgroup=true
19 | redirect_stderr=true
20 |
21 |
--------------------------------------------------------------------------------
/common/static/ace/snippets/pgsql.js:
--------------------------------------------------------------------------------
1 | define("ace/snippets/pgsql",["require","exports","module"],function(e,t,n){"use strict";t.snippetText=undefined,t.scope="pgsql"}); (function() {
2 | window.require(["ace/snippets/pgsql"], function(m) {
3 | if (typeof module == "object" && typeof exports == "object" && module) {
4 | module.exports = m;
5 | }
6 | });
7 | })();
8 |
--------------------------------------------------------------------------------
/src/init_sql/v1.7.2_v1.7.3.sql:
--------------------------------------------------------------------------------
1 | -- 修改工具插件的菜单code
2 | UPDATE auth_permission SET codename='menu_tools' WHERE codename='menu_menu_tools';
3 |
4 | -- SQL上线工单增加需求链接
5 | ALTER TABLE sql_workflow ADD demand_url varchar(500) NOT NULL DEFAULT '' COMMENT '需求链接';
6 |
7 | -- 增加事务查看权限
8 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
9 | INSERT IGNORE INTO auth_permission (name, content_type_id, codename) VALUES ('查看事务信息', @content_type_id, 'trx_view');
10 |
11 |
--------------------------------------------------------------------------------
/src/charts/templates/configMap.yaml:
--------------------------------------------------------------------------------
1 | {{- if .Values.configMap.enabled -}}
2 | kind: ConfigMap
3 | apiVersion: v1
4 | metadata:
5 | name: archery-config
6 | labels:
7 | app.kubernetes.io/name: {{ include "archery.name" . }}
8 | helm.sh/chart: {{ include "archery.chart" . }}
9 | app.kubernetes.io/instance: {{ .Release.Name }}
10 | app.kubernetes.io/managed-by: {{ .Release.Service }}
11 | {{- with .Values.configMap.data }}
12 | data:
13 | {{- toYaml . | nindent 2 }}
14 | {{- end -}}
15 | {{- end }}
16 |
--------------------------------------------------------------------------------
/requirements.txt:
--------------------------------------------------------------------------------
1 | Django==2.2.10
2 | mysqlclient==1.4.6
3 | requests==2.22.0
4 | simplejson==3.17.0
5 | mybatis_mapper2sql==0.1.9
6 | django-auth-ldap==2.1.0
7 | python-dateutil==2.8.1
8 | pymongo==3.9.0
9 | psycopg2-binary==2.8.4
10 | mysql-replication==0.21
11 | django-q==1.0.2
12 | django-redis==4.10.0
13 | pyodbc==4.0.27
14 | gunicorn==20.0.4
15 | pyecharts==1.6.0
16 | aliyun-python-sdk-rds==2.1.1
17 | cx-Oracle==7.3.0
18 | supervisor==4.1.0
19 | gevent==1.4.0
20 | phoenixdb==0.7
21 | django-mirage-field==1.0.1
22 | schema-sync==0.9.7
23 |
--------------------------------------------------------------------------------
/src/charts/charts/inception/templates/configMap.yaml:
--------------------------------------------------------------------------------
1 | {{- if .Values.configMap.enabled -}}
2 | kind: ConfigMap
3 | apiVersion: v1
4 | metadata:
5 | name: inception-config
6 | labels:
7 | app.kubernetes.io/name: {{ include "inception.name" . }}
8 | helm.sh/chart: {{ include "inception.chart" . }}
9 | app.kubernetes.io/instance: {{ .Release.Name }}
10 | app.kubernetes.io/managed-by: {{ .Release.Service }}
11 | {{- with .Values.configMap.data }}
12 | data:
13 | {{- toYaml . | nindent 2 }}
14 | {{- end }}
15 | {{- end }}
16 |
--------------------------------------------------------------------------------
/src/charts/charts/goinception/templates/configMap.yaml:
--------------------------------------------------------------------------------
1 | {{- if .Values.configMap.enabled -}}
2 | kind: ConfigMap
3 | apiVersion: v1
4 | metadata:
5 | name: goinception-config
6 | labels:
7 | app.kubernetes.io/name: {{ include "goinception.name" . }}
8 | helm.sh/chart: {{ include "goinception.chart" . }}
9 | app.kubernetes.io/instance: {{ .Release.Name }}
10 | app.kubernetes.io/managed-by: {{ .Release.Service }}
11 | {{- with .Values.configMap.data }}
12 | data:
13 | {{- toYaml . | nindent 2 }}
14 | {{- end }}
15 | {{- end }}
16 |
--------------------------------------------------------------------------------
/common/utils/global_info.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | from sql.utils.workflow_audit import Audit
3 | from archery import display_version
4 |
5 |
6 | def global_info(request):
7 | """存放用户,菜单信息等."""
8 | user = request.user
9 | if user and user.is_authenticated:
10 | # 获取待办数量
11 | try:
12 | todo = Audit.todo(user)
13 | except Exception:
14 | todo = 0
15 | else:
16 | todo = 0
17 |
18 | return {
19 | 'todo': todo,
20 | 'archery_version': display_version
21 | }
22 |
--------------------------------------------------------------------------------
/common/static/dist/css/login.css:
--------------------------------------------------------------------------------
1 | .user-bottom-div {
2 | display: block;
3 | text-align: center;
4 | background-color: #ddd;
5 | position: fixed;
6 | bottom: 0;
7 | width: 100%;
8 | height: 50px;
9 | line-height: 50px;
10 | }
11 |
12 | .lsb-login {
13 | display: flex;
14 | justify-content: center;
15 | align-items: center;
16 | height: 90vh;
17 | }
18 |
19 | .mypanalbox {
20 | background-color: #FFF;
21 | border-radius: 8px;
22 | box-shadow: 3px 3px 3px;
23 | }
24 |
25 | .login-form {
26 | margin: 20px;
27 | }
28 |
29 |
--------------------------------------------------------------------------------
/common/static/ace/snippets/mysql.js:
--------------------------------------------------------------------------------
1 | define("ace/snippets/mysql",["require","exports","module"], function(require, exports, module) {
2 | "use strict";
3 |
4 | exports.snippetText =undefined;
5 | exports.scope = "mysql";
6 |
7 | }); (function() {
8 | window.require(["ace/snippets/mysql"], function(m) {
9 | if (typeof module == "object" && typeof exports == "object" && module) {
10 | module.exports = m;
11 | }
12 | });
13 | })();
14 |
--------------------------------------------------------------------------------
/.github/ISSUE_TEMPLATE/custom.md:
--------------------------------------------------------------------------------
1 | ---
2 | name: 问题咨询
3 | about: 如果在使用过程中遇到问题,在查阅文档后仍无法解决,可以在此反馈
4 | title: '[ 问题咨询 ]'
5 | labels: 'question'
6 | assignees: ''
7 |
8 | ---
9 |
10 | ### 问题描述
11 | 详细描述你的操作步骤和结果,以及你的疑问
12 |
13 | ### 版本信息
14 | - 应用版本/分支:Release v1.4.5
15 | - 部署方式:Docker、手工部署
16 |
17 |
21 |
22 |
26 |
--------------------------------------------------------------------------------
/.travis.yml:
--------------------------------------------------------------------------------
1 | language: python
2 | python:
3 | - "3.6.5"
4 | before_install:
5 | - sudo apt-get install unixodbc unixodbc-dev
6 | services:
7 | - mysql
8 | - redis-server
9 | install:
10 | - pip install codecov coverage flake8 mycli==1.19.0 -r requirements.txt
11 | before_script:
12 | - flake8 . --count --exclude=./.* --select=E9,F63,F7,F82 --show-source --statistics
13 | - mysql -e "CREATE DATABASE archery CHARSET UTF8;"
14 | - python manage.py makemigrations
15 | - python manage.py makemigrations sql
16 | script:
17 | - coverage run manage.py test -v 3
18 |
19 | after_success:
20 | - codecov
21 |
--------------------------------------------------------------------------------
/src/charts/README.md:
--------------------------------------------------------------------------------
1 | 1、setting: 根目录values.yaml下配置,相关configMap的文件,settings.py,soar.yaml,analysis_slow_query.sh等,找出与mysql,redis的数据库连接 charts/goinception,charts/inception目录下的values.yaml配置修改,主要与mysql连接的配置 mysql的存储持久化,请查看values.yaml的方法进行配置
2 |
3 | 2、dependency: cd charts/archeryk8s && helm dependency update
4 |
5 | 3、install: helm install ./ --name archery --namespace=default
6 |
7 | 4、visit:
8 |
9 | i 本机访问 kubectl port-forward pods/archery-xxxxxx 9123:9123
10 | ii 集群外访问 将svc配置为nodePort或loadBalance,或开启ingress
11 |
12 | 默认关闭ingress,如需开启ingress,请在values.yaml设置为true,并修改ingress默认域名配置。
13 |
14 |
15 | 默认用户名:admin
16 | 默认密码:Archery2019
17 |
18 |
--------------------------------------------------------------------------------
/src/init_sql/v1.3.0_v1.3.2.sql:
--------------------------------------------------------------------------------
1 | -- 修改查询日志表
2 | ALTER TABLE query_log
3 | ADD priv_check TINYINT NOT NULL DEFAULT 0
4 | COMMENT '查询权限是否正常校验,1, 正常, 2, 跳过'
5 | AFTER user_display,
6 | ADD hit_rule TINYINT NOT NULL DEFAULT 0
7 | COMMENT '查询是否命中脱敏规则,0,未知, 1, 命中, 2,未命中'
8 | AFTER priv_check,
9 | ADD masking TINYINT NOT NULL DEFAULT 0
10 | COMMENT '查询结果是否正常脱敏,1, 是, 2, 否'
11 | AFTER hit_rule;
12 |
13 | -- 增加菜单权限
14 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
15 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 SchemaSync', @content_type_id, 'menu_schemasync');
16 |
--------------------------------------------------------------------------------
/common/utils/timer.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: timer.py
6 | @time: 2019/05/15
7 | """
8 | import datetime
9 |
10 | __author__ = 'hhyo'
11 |
12 |
13 | class FuncTimer(object):
14 | """
15 | 获取执行时间的上下文管理器
16 | """
17 |
18 | def __init__(self):
19 | self.start = None
20 | self.end = None
21 | self.cost = 0
22 |
23 | def __enter__(self):
24 | self.start = datetime.datetime.now()
25 | return self
26 |
27 | def __exit__(self, exc_type, exc_val, exc_tb):
28 | self.end = datetime.datetime.now()
29 | self.cost = (self.end - self.start).total_seconds()
30 |
--------------------------------------------------------------------------------
/supervisord.conf:
--------------------------------------------------------------------------------
1 | [unix_http_server]
2 | file=supervisor.sock
3 |
4 | [supervisord]
5 | logfile=logs/supervisord.log
6 | nodaemon=false
7 |
8 | [supervisorctl]
9 | serverurl=unix://supervisor.sock
10 |
11 | [rpcinterface:supervisor]
12 | supervisor.rpcinterface_factory=supervisor.rpcinterface:make_main_rpcinterface
13 |
14 | [program:archery]
15 | command=gunicorn -w 4 -k gevent -b 127.0.0.1:8000 --timeout 600 archery.wsgi:application
16 | autorestart=true
17 | stopasgroup=true
18 | killasgroup=true
19 | redirect_stderr=true
20 |
21 | [program:qcluster]
22 | command=python manage.py qcluster
23 | autorestart=true
24 | stopasgroup=true
25 | killasgroup=true
26 | redirect_stderr=true
27 |
28 |
--------------------------------------------------------------------------------
/common/storage.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: storage.py
6 | @time: 2019/06/01
7 | """
8 |
9 | __author__ = 'hhyo'
10 |
11 | from django.contrib.staticfiles.storage import ManifestStaticFilesStorage
12 |
13 |
14 | class ForgivingManifestStaticFilesStorage(ManifestStaticFilesStorage):
15 | manifest_strict = False
16 |
17 | def hashed_name(self, name, content=None, filename=None):
18 | try:
19 | result = super().hashed_name(name, content, filename)
20 | except ValueError:
21 | # When the file is missing, let's forgive and ignore that.
22 | result = name
23 | return result
24 |
--------------------------------------------------------------------------------
/src/charts/templates/service.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: v1
2 | kind: Service
3 | metadata:
4 | name: {{ include "archery.fullname" . }}
5 | labels:
6 | app.kubernetes.io/name: {{ include "archery.name" . }}
7 | helm.sh/chart: {{ include "archery.chart" . }}
8 | app.kubernetes.io/instance: {{ .Release.Name }}
9 | app.kubernetes.io/managed-by: {{ .Release.Service }}
10 | spec:
11 | type: {{ .Values.service.type }}
12 | ports:
13 | - port: {{ .Values.service.port }}
14 | targetPort: {{ .Values.service.targetPort }}
15 | protocol: TCP
16 | name: archery
17 | selector:
18 | app.kubernetes.io/name: {{ include "archery.name" . }}
19 | app.kubernetes.io/instance: {{ .Release.Name }}
20 |
--------------------------------------------------------------------------------
/src/docker/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM hhyo/archery-base:1.2
2 |
3 | WORKDIR /opt/archery
4 |
5 | COPY . /opt/archery/
6 |
7 | #archery
8 | RUN cd /opt \
9 | && yum -y install nginx \
10 | && source /opt/venv4archery/bin/activate \
11 | && pip3 install -r /opt/archery/requirements.txt \
12 | && cp /opt/archery/src/docker/nginx.conf /etc/nginx/ \
13 | && cp /opt/archery/src/docker/supervisord.conf /etc/ \
14 | && mv /opt/sqladvisor /opt/archery/src/plugins/ \
15 | && mv /opt/soar /opt/archery/src/plugins/ \
16 | && mv /opt/tmp_binlog2sql /opt/archery/src/plugins/binlog2sql
17 |
18 | #port
19 | EXPOSE 9123
20 |
21 | #start service
22 | ENTRYPOINT bash /opt/archery/src/docker/startup.sh && bash
23 |
--------------------------------------------------------------------------------
/common/static/datetimepicker/js/bootstrap-datetimepicker.zh-CN.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Simplified Chinese translation for bootstrap-datetimepicker
3 | * Yuan Cheung
4 | */
5 | ;(function($){
6 | $.fn.datetimepicker.dates['zh-CN'] = {
7 | days: ["星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六", "星期日"],
8 | daysShort: ["周日", "周一", "周二", "周三", "周四", "周五", "周六", "周日"],
9 | daysMin: ["日", "一", "二", "三", "四", "五", "六", "日"],
10 | months: ["一月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "十二月"],
11 | monthsShort: ["一月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "十二月"],
12 | today: "今天",
13 | suffix: [],
14 | meridiem: ["上午", "下午"]
15 | };
16 | }(jQuery));
17 |
--------------------------------------------------------------------------------
/.github/release-drafter.yml:
--------------------------------------------------------------------------------
1 | name-template: 'Release v$NEXT_PATCH_VERSION'
2 | tag-template: 'v$NEXT_PATCH_VERSION'
3 | categories:
4 | - title: '变更说明'
5 | labels:
6 | - 'feature'
7 | - 'enhancement'
8 | - title: '修复说明'
9 | labels:
10 | - 'fix'
11 | - 'bugfix'
12 | - 'bug'
13 | - title: '易用性改进'
14 | labels:
15 | - 'ease of use'
16 | exclude-labels:
17 | - 'skip-changelog'
18 | change-template: '- $TITLE @$AUTHOR (#$NUMBER)'
19 | template: |
20 | ## Release date: xxx年xx月xx日
21 |
22 | 代码变更: https://github.com/hhyo/Archery/compare/$PREVIOUS_TAG...v$NEXT_PATCH_VERSION
23 | 贡献者: $CONTRIBUTORS
24 |
25 | $CHANGES
26 |
27 | ## 升级步骤
28 | - https://github.com/hhyo/Archery/wiki/升级
29 |
--------------------------------------------------------------------------------
/src/charts/charts/inception/templates/service.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: v1
2 | kind: Service
3 | metadata:
4 | name: {{ include "inception.fullname" . }}
5 | labels:
6 | app.kubernetes.io/name: {{ include "inception.name" . }}
7 | helm.sh/chart: {{ include "inception.chart" . }}
8 | app.kubernetes.io/instance: {{ .Release.Name }}
9 | app.kubernetes.io/managed-by: {{ .Release.Service }}
10 | spec:
11 | type: {{ .Values.service.type }}
12 | ports:
13 | - port: {{ .Values.service.port }}
14 | targetPort: {{ .Values.service.targetPort }}
15 | protocol: TCP
16 | name: inception
17 | selector:
18 | app.kubernetes.io/name: {{ include "inception.name" . }}
19 | app.kubernetes.io/instance: {{ .Release.Name }}
20 |
--------------------------------------------------------------------------------
/src/charts/charts/goinception/templates/service.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: v1
2 | kind: Service
3 | metadata:
4 | name: {{ include "goinception.fullname" . }}
5 | labels:
6 | app.kubernetes.io/name: {{ include "goinception.name" . }}
7 | helm.sh/chart: {{ include "goinception.chart" . }}
8 | app.kubernetes.io/instance: {{ .Release.Name }}
9 | app.kubernetes.io/managed-by: {{ .Release.Service }}
10 | spec:
11 | type: {{ .Values.service.type }}
12 | ports:
13 | - port: {{ .Values.service.port }}
14 | targetPort: {{ .Values.service.targetPort }}
15 | protocol: TCP
16 | name: goinception
17 | selector:
18 | app.kubernetes.io/name: {{ include "goinception.name" . }}
19 | app.kubernetes.io/instance: {{ .Release.Name }}
20 |
--------------------------------------------------------------------------------
/src/docker/startup.sh:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 |
3 | cd /opt/archery
4 |
5 | echo 切换python运行环境
6 | source /opt/venv4archery/bin/activate
7 | #pip install -r requirements.txt -i https://mirrors.ustc.edu.cn/pypi/web/simple/
8 |
9 | echo 修改重定向端口
10 | if [[ -z $NGINX_PORT ]]; then
11 | sed -i "s/:nginx_port//g" /etc/nginx/nginx.conf
12 | else
13 | sed -i "s/nginx_port/$NGINX_PORT/g" /etc/nginx/nginx.conf
14 | fi
15 |
16 | echo 启动nginx
17 | /usr/sbin/nginx
18 |
19 | echo 收集所有的静态文件到STATIC_ROOT
20 | python3 manage.py collectstatic -v0 --noinput
21 |
22 | echo 启动Django Q cluster
23 | supervisord -c /etc/supervisord.conf
24 |
25 | echo 启动服务
26 | gunicorn -w 4 -k gevent -b 127.0.0.1:8888 --timeout 600 archery.wsgi:application
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
--------------------------------------------------------------------------------
/sql_api/tests.py:
--------------------------------------------------------------------------------
1 | from django.test import TestCase
2 | from django.contrib.auth import get_user_model
3 |
4 | User = get_user_model()
5 |
6 |
7 | class InfoTest(TestCase):
8 | def setUp(self) -> None:
9 | self.superuser = User.objects.create(username='super', is_superuser=True)
10 | self.client.force_login(self.superuser)
11 |
12 | def tearDown(self) -> None:
13 | self.superuser.delete()
14 |
15 | def test_info_api(self):
16 | r = self.client.get('/api/info')
17 | r_json = r.json()
18 | self.assertIsInstance(r_json['archery']['version'], str)
19 |
20 | def test_debug_api(self):
21 | r = self.client.get('/api/debug')
22 | r_json = r.json()
23 | self.assertIsInstance(r_json['archery']['version'], str)
24 |
--------------------------------------------------------------------------------
/src/init_sql/v1.7.0_v1.7.1.sql:
--------------------------------------------------------------------------------
1 | -- 增加资源组粒度的查询权限,v1.7.0的model中遗漏,全新安装的需要补充
2 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
3 | INSERT IGNORE INTO auth_permission (name, content_type_id, codename) VALUES ('可查询所在资源组内的所有实例', @content_type_id, 'query_resource_group_instance');
4 |
5 | -- 记录企业微信userid
6 | alter table sql_users add wx_user_id varchar(64) default null comment '企业微信UserID';
7 |
8 | -- 删除阿里云AK配置表,转移到系统配置中,扩大系统配置项长度
9 | drop table aliyun_access_key;
10 | alter table sql_config modify `item` varchar(200) NOT NULL comment '配置项',
11 | modify `value` varchar(500) NOT NULL DEFAULT '' comment '配置项值';
12 |
13 | -- 使用django-mirage-field加密实例信息,扩大字段长度
14 | alter table sql_instance modify `user` varchar(200) DEFAULT NULL COMMENT '用户名';
15 |
--------------------------------------------------------------------------------
/common/middleware/check_login_middleware.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | import re
3 | from django.http import HttpResponseRedirect
4 | from django.utils.deprecation import MiddlewareMixin
5 |
6 | IGNORE_URL = [
7 | '/login/',
8 | '/authenticate/',
9 | '/signup/',
10 | '/api/info'
11 | ]
12 |
13 | IGNORE_URL_RE = r'/admin/\w*'
14 |
15 |
16 | class CheckLoginMiddleware(MiddlewareMixin):
17 | @staticmethod
18 | def process_request(request):
19 | """
20 | 该函数在每个函数之前检查是否登录,若未登录,则重定向到/login/
21 | """
22 | if not request.user.is_authenticated:
23 | # 以下是不用跳转到login页面的url白名单
24 | if request.path not in IGNORE_URL and re.match(IGNORE_URL_RE, request.path) is None:
25 | return HttpResponseRedirect('/login/')
26 |
--------------------------------------------------------------------------------
/src/init_sql/v1.4.2_v1.4.3.sql:
--------------------------------------------------------------------------------
1 | -- 修改工单状态
2 | UPDATE sql_workflow SET status = 'workflow_finish' WHERE status='已正常结束';
3 | UPDATE sql_workflow SET status = 'workflow_abort' WHERE status='人工终止流程';
4 | UPDATE sql_workflow SET status = 'workflow_manreviewing' WHERE status='等待审核人审核';
5 | UPDATE sql_workflow SET status = 'workflow_review_pass' WHERE status='审核通过';
6 | UPDATE sql_workflow SET status = 'workflow_timingtask' WHERE status='定时执行';
7 | UPDATE sql_workflow SET status = 'workflow_executing' WHERE status='执行中';
8 | UPDATE sql_workflow SET status = 'workflow_autoreviewwrong' WHERE status='自动审核不通过';
9 | UPDATE sql_workflow SET status = 'workflow_exception' WHERE status='执行有异常';
10 |
11 | -- display修改为not null
12 | alter table sql_users modify display varchar(50) not null default '' comment '显示的中文名';
13 |
--------------------------------------------------------------------------------
/src/init_sql/v1.7.6_v1.7.7.sql:
--------------------------------------------------------------------------------
1 | -- 修改多对多的中间表
2 | alter table resource_group_user
3 | rename to sql_users_resource_group,
4 | drop create_time,
5 | change user_id users_id int(11) NOT NULL COMMENT '用户',
6 | change resource_group_id resourcegroup_id int(11) NOT NULL COMMENT '资源组';
7 |
8 | alter table resource_group_instance
9 | rename to sql_instance_resource_group,
10 | drop create_time,
11 | change resource_group_id resourcegroup_id int(11) NOT NULL COMMENT '资源组';
12 |
13 | alter table sql_instance_tag_relations
14 | rename to sql_instance_instance_tag,
15 | drop `active`,
16 | drop create_time,
17 | change instance_tag_id instancetag_id int(11) NOT NULL COMMENT '关联标签ID';
18 |
19 | -- 实例配置表新增默认数据库字段
20 | ALTER TABLE sql_instance ADD `db_name` VARCHAR(64) NOT NULL DEFAULT '' COMMENT '数据库';
21 |
--------------------------------------------------------------------------------
/.github/stale.yml:
--------------------------------------------------------------------------------
1 | # Number of days of inactivity before an issue becomes stale
2 | daysUntilStale: 15
3 | # Number of days of inactivity before a stale issue is closed
4 | daysUntilClose: 3
5 | # Issues with these labels will never be considered stale
6 | exemptLabels:
7 | - pinned
8 | - security
9 | - bug
10 | - enhancement
11 | # Label to use when marking an issue as stale
12 | staleLabel: wontfix
13 | # Comment to post when marking an issue as stale. Set to `false` to disable
14 | markComment: >
15 | 你已经很久没有回复这个issue了,如果没有进一步的信息的话, 会作为不活跃issue关闭, 感谢你对本项目的贡献。
16 |
17 | This issue has been automatically marked as stale because it has not had
18 | recent activity. It will be closed if no further activity occurs. Thank you
19 | for your contributions.
20 | # Comment to post when closing a stale issue. Set to `false` to disable
21 | closeComment: false
22 |
--------------------------------------------------------------------------------
/common/static/sb-admin-2/js/sb-admin-2.min.js:
--------------------------------------------------------------------------------
1 | /*!
2 | * Start Bootstrap - SB Admin 2 v3.3.7+1 (http://startbootstrap.com/template-overviews/sb-admin-2)
3 | * Copyright 2013-2016 Start Bootstrap
4 | * Licensed under MIT (https://github.com/BlackrockDigital/startbootstrap/blob/gh-pages/LICENSE)
5 | */
6 | $(function(){$("#side-menu").metisMenu()}),$(function(){$(window).bind("load resize",function(){var i=50,n=this.window.innerWidth>0?this.window.innerWidth:this.screen.width;n<768?($("div.navbar-collapse").addClass("collapse"),i=100):$("div.navbar-collapse").removeClass("collapse");var e=(this.window.innerHeight>0?this.window.innerHeight:this.screen.height)-1;e-=i,e<1&&(e=1),e>i&&$("#page-wrapper").css("min-height",e+"px")});for(var i=window.location,n=$("ul.nav a").filter(function(){return this.href==i}).addClass("active").parent();;){if(!n.is("li"))break;n=n.parent().addClass("in").parent()}});
--------------------------------------------------------------------------------
/.github/ISSUE_TEMPLATE/bug.md:
--------------------------------------------------------------------------------
1 | ---
2 | name: BUG提交
3 | about: BUG提交,必须使用这个模板,不规范issue将直接关闭
4 | title: "[ bug ]"
5 | labels: ''
6 | assignees: ''
7 |
8 | ---
9 |
10 | 在提交 issue 前,请查阅以下资源,请先进行搜索来保证没有类似的 issue。
11 | [文档](https://github.com/hhyo/archery/wiki) | [FAQ](https://github.com/hhyo/archery/wiki/FAQ)
12 |
13 | ### 重现步骤
14 |
15 |
21 |
22 | ### 期待结果和实际结果
23 | 期待结果:
24 |
25 | 实际结果:
26 |
27 | ### 截图
28 |
29 |
30 |
32 | ### 错误日志
33 |
34 |
35 |
43 | ### 版本信息
44 | 应用版本/分支:
45 |
46 | 部署方式:Docker、手工部署
47 |
48 |
53 |
--------------------------------------------------------------------------------
/src/docker-compose/inception/inc.cnf:
--------------------------------------------------------------------------------
1 | [inception]
2 | general_log=1
3 | general_log_file=inception.log
4 | port=6669
5 | socket=/tmp/inc.socket
6 | character-set-client-handshake=0
7 | character-set-server=utf8
8 | inception_language_code=zh-CN
9 | inception_remote_system_password=123456
10 | inception_remote_system_user=root
11 | inception_remote_backup_port=3306
12 | inception_remote_backup_host=mysql
13 | inception_support_charset=utf8,utf8mb4
14 | inception_enable_nullable=0
15 | inception_check_primary_key=1
16 | inception_check_column_comment=1
17 | inception_check_table_comment=1
18 | inception_osc_on=OFF
19 | inception_osc_bin_dir=/usr/bin
20 | inception_osc_min_table_size=10
21 | inception_osc_chunk_time=0.1
22 | inception_enable_blob_type=1
23 | inception_check_column_default_value=1
24 |
25 | inception_enable_select_star=ON
26 | inception_enable_identifer_keyword=ON
27 | inception_enable_autoincrement_unsigned=ON
28 | inception_check_identifier=OFF
29 |
--------------------------------------------------------------------------------
/common/static/bootstrap-select/js/i18n/defaults-zh_CN.min.js:
--------------------------------------------------------------------------------
1 | /*!
2 | * Bootstrap-select v1.13.9 (https://developer.snapappointments.com/bootstrap-select)
3 | *
4 | * Copyright 2012-2019 SnapAppointments, LLC
5 | * Licensed under MIT (https://github.com/snapappointments/bootstrap-select/blob/master/LICENSE)
6 | */
7 |
8 | !function(e,t){void 0===e&&void 0!==window&&(e=window),"function"==typeof define&&define.amd?define(["jquery"],function(e){return t(e)}):"object"==typeof module&&module.exports?module.exports=t(require("jquery")):t(e.jQuery)}(this,function(e){e.fn.selectpicker.defaults={noneSelectedText:"\u6ca1\u6709\u9009\u4e2d\u4efb\u4f55\u9879",noneResultsText:"\u6ca1\u6709\u627e\u5230\u5339\u914d\u9879",countSelectedText:"\u9009\u4e2d{1}\u4e2d\u7684{0}\u9879",maxOptionsText:["\u8d85\u51fa\u9650\u5236 (\u6700\u591a\u9009\u62e9{n}\u9879)","\u7ec4\u9009\u62e9\u8d85\u51fa\u9650\u5236(\u6700\u591a\u9009\u62e9{n}\u7ec4)"],multipleSeparator:", ",selectAllText:"\u5168\u9009",deselectAllText:"\u53d6\u6d88\u5168\u9009"}});
--------------------------------------------------------------------------------
/common/static/bootstrap-select/js/i18n/defaults-zh_CN.js.map:
--------------------------------------------------------------------------------
1 | {"version":3,"sources":["../../../js/i18n/defaults-zh_CN.js"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;AAAA,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC;AAChB,EAAE,EAAE,EAAE,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;AACjC,IAAI,gBAAgB,CAAC,CAAC,UAAU,CAAC;AACjC,IAAI,eAAe,CAAC,CAAC,UAAU,CAAC;AAChC,IAAI,iBAAiB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC;AACtC,IAAI,cAAc,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC;AAC9D,IAAI,iBAAiB,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC;AAC7B,IAAI,aAAa,CAAC,CAAC,KAAK,CAAC;AACzB,IAAI,eAAe,CAAC,CAAC,MAAM,CAAC;AAC5B,EAAE,EAAE,CAAC;AACL,GAAG,MAAM,EAAE,CAAC","file":"defaults-zh_CN.js","sourcesContent":["(function ($) {\r\n $.fn.selectpicker.defaults = {\r\n noneSelectedText: '没有选中任何项',\r\n noneResultsText: '没有找到匹配项',\r\n countSelectedText: '选中{1}中的{0}项',\r\n maxOptionsText: ['超出限制 (最多选择{n}项)', '组选择超出限制(最多选择{n}组)'],\r\n multipleSeparator: ', ',\r\n selectAllText: '全选',\r\n deselectAllText: '取消全选'\r\n };\r\n})(jQuery);\r\n"]}
--------------------------------------------------------------------------------
/src/docker-compose/mysql/my.cnf:
--------------------------------------------------------------------------------
1 | [mysqld_safe]
2 | socket = /var/run/mysqld/mysqld.sock
3 | nice = 0
4 |
5 | [mysqld]
6 | pid-file = /var/run/mysqld/mysqld.pid
7 | socket = /var/run/mysqld/mysqld.sock
8 | port = 3306
9 | basedir = /usr
10 | datadir = /var/lib/mysql
11 | tmpdir = /tmp
12 | lc-messages-dir = /usr/share/mysql
13 | skip-external-locking
14 | lower_case_table_names=1
15 | default-time_zone = '+8:00'
16 |
17 | innodb_buffer_pool_size = 512M
18 |
19 | server-id = 100
20 | log_bin = /var/log/mysql/mysql-bin.log
21 | expire_logs_days = 1
22 | max_binlog_size = 500M
23 |
24 | character-set-server = utf8mb4
25 | collation-server = utf8mb4_general_ci
26 |
27 | slow_query_log_file = mysql-slow.log
28 | slow_query_log = 1
29 | long_query_time = 1
30 |
31 | [client]
32 | default-character-set=utf8mb4
33 |
34 | [mysqldump]
35 | quick
36 | quote-names
37 | max_allowed_packet = 1024M
38 |
39 |
40 | !includedir /etc/mysql/conf.d/
41 |
--------------------------------------------------------------------------------
/sql/templates/dbaprinciples.html:
--------------------------------------------------------------------------------
1 | {% extends "base.html" %}
2 |
3 | {% block content %}
4 |
8 | {% endblock content %}
9 |
10 | {% block js %}
11 | {% load static %}
12 |
13 |
31 | {% endblock %}
32 |
--------------------------------------------------------------------------------
/sql/plugins/sqladvisor.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: sqladvisor.py
6 | @time: 2019/03/04
7 | """
8 | __author__ = 'hhyo'
9 |
10 | from common.config import SysConfig
11 | from sql.plugins.plugin import Plugin
12 |
13 |
14 | class SQLAdvisor(Plugin):
15 | def __init__(self):
16 | self.path = SysConfig().get('sqladvisor')
17 | self.required_args = ['q']
18 | self.disable_args = []
19 | super(Plugin, self).__init__()
20 |
21 | def generate_args2cmd(self, args, shell):
22 | """
23 | 转换请求参数为命令行
24 | :param args:
25 | :param shell:
26 | :return:
27 | """
28 | if shell:
29 | cmd_args = self.path if self.path else ''
30 | for name, value in args.items():
31 | cmd_args += f' -{name} "{value}"'
32 | else:
33 | cmd_args = [self.path]
34 | for name, value in args.items():
35 | cmd_args.append(f'-{name}')
36 | cmd_args.append(f'{value}')
37 | return cmd_args
38 |
--------------------------------------------------------------------------------
/common/utils/wx_api.py:
--------------------------------------------------------------------------------
1 | # coding: utf-8
2 |
3 | import logging
4 | import requests
5 | from common.config import SysConfig
6 | from django.core.cache import cache
7 |
8 | logger = logging.getLogger('default')
9 |
10 |
11 | def get_wx_access_token():
12 | # 优先获取缓存
13 | try:
14 | token = cache.get('wx_access_token')
15 | except Exception as e:
16 | logger.error(f"获取企业微信token缓存出错:{e}")
17 | token = None
18 | if token:
19 | return token
20 | # 请求企业微信接口获取
21 | sys_config = SysConfig()
22 | corp_id = sys_config.get('wx_corpid')
23 | corp_secret = sys_config.get('wx_app_secret')
24 | url = f"https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid={corp_id}&corpsecret={corp_secret}"
25 | resp = requests.get(url, timeout=3).json()
26 | if resp.get('errcode') == 0:
27 | access_token = resp.get('access_token')
28 | expires_in = resp.get('expires_in')
29 | cache.set('wx_access_token', access_token, timeout=expires_in - 60)
30 | return access_token
31 | else:
32 | logger.error(f"获取企业微信access_token出错:{resp}")
33 | return None
34 |
--------------------------------------------------------------------------------
/src/init_sql/del_permissions.sql:
--------------------------------------------------------------------------------
1 | -- 用于删除非自定义权限,如果不需要使用model权限管理,可以执行该脚本,仅保留自定义权限
2 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
3 |
4 | -- delete auth_group_permissions
5 | delete a
6 | from auth_group_permissions a
7 | join auth_permission b on a.permission_id = b.id
8 | where (b.content_type_id <> @content_type_id or
9 | (b.content_type_id = @content_type_id and
10 | codename in ('add_permission', 'change_permission', 'delete_permission')));
11 |
12 | -- delete sql_users_user_permissions
13 | delete a
14 | from sql_users_user_permissions a
15 | join auth_permission b on a.permission_id = b.id
16 | where (b.content_type_id <> @content_type_id or
17 | (b.content_type_id = @content_type_id and
18 | codename in ('add_permission', 'change_permission', 'delete_permission')));
19 |
20 | -- delete auth_permission
21 | delete
22 | from auth_permission
23 | where (content_type_id <> @content_type_id or
24 | (content_type_id = @content_type_id and
25 | codename in ('add_permission', 'change_permission', 'delete_permission')));
26 |
--------------------------------------------------------------------------------
/common/views.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: views.py
6 | @time: 2019/12/21
7 | """
8 | from django.shortcuts import render
9 |
10 | __author__ = 'hhyo'
11 |
12 | from django.http import (
13 | HttpResponseBadRequest, HttpResponseForbidden, HttpResponseNotFound,
14 | HttpResponseServerError,
15 | )
16 | from django.views.decorators.csrf import requires_csrf_token
17 |
18 |
19 | @requires_csrf_token
20 | def bad_request(request, exception, template_name='errors/400.html'):
21 | return HttpResponseBadRequest(render(request, template_name))
22 |
23 |
24 | @requires_csrf_token
25 | def permission_denied(request, exception, template_name='errors/403.html'):
26 | return HttpResponseForbidden(render(request, template_name))
27 |
28 |
29 | @requires_csrf_token
30 | def page_not_found(request, exception, template_name='errors/404.html'):
31 | return HttpResponseNotFound(render(request, template_name))
32 |
33 |
34 | @requires_csrf_token
35 | def server_error(request, template_name='errors/500.html'):
36 | return HttpResponseServerError(render(request, template_name))
37 |
--------------------------------------------------------------------------------
/common/static/bootstrap-table/export-libs/FileSaver/LICENSE.md:
--------------------------------------------------------------------------------
1 | The MIT License
2 |
3 | Copyright © 2016 [Eli Grey][1].
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
6 |
7 | The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
8 |
9 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
10 |
11 | [1]: http://eligrey.com
12 |
--------------------------------------------------------------------------------
/common/static/bootstrap-table/export-libs/html2canvas/LICENSE:
--------------------------------------------------------------------------------
1 | Copyright (c) 2012 Niklas von Hertzen
2 |
3 | Permission is hereby granted, free of charge, to any person
4 | obtaining a copy of this software and associated documentation
5 | files (the "Software"), to deal in the Software without
6 | restriction, including without limitation the rights to use,
7 | copy, modify, merge, publish, distribute, sublicense, and/or sell
8 | copies of the Software, and to permit persons to whom the
9 | Software is furnished to do so, subject to the following
10 | conditions:
11 |
12 | The above copyright notice and this permission notice shall be
13 | included in all copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
17 | OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
18 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
19 | HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
20 | WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 | FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
22 | OTHER DEALINGS IN THE SOFTWARE.
--------------------------------------------------------------------------------
/common/static/bootstrap-table/export-libs/pdfmake/LICENSE:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2014 bpampuch
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy of
6 | this software and associated documentation files (the "Software"), to deal in
7 | the Software without restriction, including without limitation the rights to
8 | use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
9 | the Software, and to permit persons to whom the Software is furnished to do so,
10 | subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
17 | FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
18 | COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
19 | IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
20 | CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21 |
--------------------------------------------------------------------------------
/src/charts/templates/_helpers.tpl:
--------------------------------------------------------------------------------
1 | {{/* vim: set filetype=mustache: */}}
2 | {{/*
3 | Expand the name of the chart.
4 | */}}
5 | {{- define "archery.name" -}}
6 | {{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}}
7 | {{- end -}}
8 |
9 | {{/*
10 | Create a default fully qualified app name.
11 | We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec).
12 | If release name contains chart name it will be used as a full name.
13 | */}}
14 | {{- define "archery.fullname" -}}
15 | {{- if .Values.fullnameOverride -}}
16 | {{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" -}}
17 | {{- else -}}
18 | {{- $name := default .Chart.Name .Values.nameOverride -}}
19 | {{- if contains $name .Release.Name -}}
20 | {{- .Release.Name | trunc 63 | trimSuffix "-" -}}
21 | {{- else -}}
22 | {{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}}
23 | {{- end -}}
24 | {{- end -}}
25 | {{- end -}}
26 |
27 | {{/*
28 | Create chart name and version as used by the chart label.
29 | */}}
30 | {{- define "archery.chart" -}}
31 | {{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}}
32 | {{- end -}}
33 |
--------------------------------------------------------------------------------
/src/plugins/soar.yaml:
--------------------------------------------------------------------------------
1 | # 是否允许测试环境与线上环境配置相同
2 | allow-online-as-test: false
3 | # 是否清理测试时产生的临时文件
4 | drop-test-temporary: true
5 | # 语法检查小工具
6 | only-syntax-check: false
7 | sampling-data-factor: 100
8 | sampling: true
9 | sampling-statistic-target: 100
10 | profiling: false
11 | trace: false
12 | # 日志级别,[0:Emergency, 1:Alert, 2:Critical, 3:Error, 4:Warning, 5:Notice, 6:Informational, 7:Debug]
13 | log-level: 7
14 | log-output: /opt/archery/logs/soar.log
15 | # 优化建议输出格式
16 | report-type: markdown
17 | ignore-rules:
18 | - ""
19 | # 启发式算法相关配置
20 | max-join-table-count: 5
21 | max-group-by-cols-count: 5
22 | max-distinct-count: 5
23 | max-index-cols-count: 5
24 | max-total-rows: 9999999
25 | spaghetti-query-length: 2048
26 | allow-drop-index: false
27 | # EXPLAIN相关配置
28 | explain-sql-report-type: pretty
29 | explain-type: extended
30 | explain-format: traditional
31 | explain-warn-select-type:
32 | - ""
33 | explain-warn-access-type:
34 | - ALL
35 | explain-max-keys: 3
36 | explain-min-keys: 0
37 | explain-max-rows: 10000
38 | explain-warn-extra:
39 | - ""
40 | explain-max-filtered: 100
41 | explain-warn-scalability:
42 | - O(n)
43 | query: ""
44 | list-heuristic-rules: false
45 | list-test-sqls: false
46 | verbose: true
47 |
--------------------------------------------------------------------------------
/common/static/bootstrap-table/export-libs/es6-promise/LICENSE:
--------------------------------------------------------------------------------
1 | Copyright (c) 2014 Yehuda Katz, Tom Dale, Stefan Penner and contributors
2 |
3 | Permission is hereby granted, free of charge, to any person obtaining a copy of
4 | this software and associated documentation files (the "Software"), to deal in
5 | the Software without restriction, including without limitation the rights to
6 | use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
7 | of the Software, and to permit persons to whom the Software is furnished to do
8 | so, subject to the following conditions:
9 |
10 | The above copyright notice and this permission notice shall be included in all
11 | copies or substantial portions of the Software.
12 |
13 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
19 | SOFTWARE.
20 |
--------------------------------------------------------------------------------
/common/static/bootstrap-table/export-libs/jsPDF/MIT-LICENSE.txt:
--------------------------------------------------------------------------------
1 | Copyright (c) 2010-2016 James Hall, https://github.com/MrRio/jsPDF
2 |
3 | Permission is hereby granted, free of charge, to any person obtaining
4 | a copy of this software and associated documentation files (the
5 | "Software"), to deal in the Software without restriction, including
6 | without limitation the rights to use, copy, modify, merge, publish,
7 | distribute, sublicense, and/or sell copies of the Software, and to
8 | permit persons to whom the Software is furnished to do so, subject to
9 | the following conditions:
10 |
11 | The above copyright notice and this permission notice shall be
12 | included in all copies or substantial portions of the Software.
13 |
14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21 |
--------------------------------------------------------------------------------
/src/charts/charts/inception/templates/_helpers.tpl:
--------------------------------------------------------------------------------
1 | {{/* vim: set filetype=mustache: */}}
2 | {{/*
3 | Expand the name of the chart.
4 | */}}
5 | {{- define "inception.name" -}}
6 | {{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}}
7 | {{- end -}}
8 |
9 | {{/*
10 | Create a default fully qualified app name.
11 | We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec).
12 | If release name contains chart name it will be used as a full name.
13 | */}}
14 | {{- define "inception.fullname" -}}
15 | {{- if .Values.fullnameOverride -}}
16 | {{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" -}}
17 | {{- else -}}
18 | {{- $name := default .Chart.Name .Values.nameOverride -}}
19 | {{- if contains $name .Release.Name -}}
20 | {{- .Release.Name | trunc 63 | trimSuffix "-" -}}
21 | {{- else -}}
22 | {{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}}
23 | {{- end -}}
24 | {{- end -}}
25 | {{- end -}}
26 |
27 | {{/*
28 | Create chart name and version as used by the chart label.
29 | */}}
30 | {{- define "inception.chart" -}}
31 | {{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}}
32 | {{- end -}}
33 |
--------------------------------------------------------------------------------
/src/docker-compose/archery/soar.yaml:
--------------------------------------------------------------------------------
1 | # 是否允许测试环境与线上环境配置相同
2 | allow-online-as-test: false
3 | # 是否清理测试时产生的临时文件
4 | drop-test-temporary: true
5 | # 语法检查小工具
6 | only-syntax-check: false
7 | sampling-data-factor: 100
8 | sampling: false
9 | sampling-statistic-target: 100
10 | profiling: false
11 | trace: false
12 | # 日志级别,[0:Emergency, 1:Alert, 2:Critical, 3:Error, 4:Warning, 5:Notice, 6:Informational, 7:Debug]
13 | log-level: 3
14 | log-output: /opt/archery/logs/soar.log
15 | # 优化建议输出格式
16 | report-type: markdown
17 | ignore-rules:
18 | - ""
19 | # 启发式算法相关配置
20 | max-join-table-count: 5
21 | max-group-by-cols-count: 5
22 | max-distinct-count: 5
23 | max-index-cols-count: 5
24 | max-total-rows: 9999999
25 | spaghetti-query-length: 2048
26 | allow-drop-index: false
27 | # EXPLAIN相关配置
28 | explain-sql-report-type: pretty
29 | explain-type: extended
30 | explain-format: traditional
31 | explain-warn-select-type:
32 | - ""
33 | explain-warn-access-type:
34 | - ALL
35 | explain-max-keys: 3
36 | explain-min-keys: 0
37 | explain-max-rows: 10000
38 | explain-warn-extra:
39 | - ""
40 | explain-max-filtered: 100
41 | explain-warn-scalability:
42 | - O(n)
43 | query: ""
44 | list-heuristic-rules: false
45 | list-test-sqls: false
46 | verbose: true
47 |
--------------------------------------------------------------------------------
/src/charts/charts/goinception/templates/_helpers.tpl:
--------------------------------------------------------------------------------
1 | {{/* vim: set filetype=mustache: */}}
2 | {{/*
3 | Expand the name of the chart.
4 | */}}
5 | {{- define "goinception.name" -}}
6 | {{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}}
7 | {{- end -}}
8 |
9 | {{/*
10 | Create a default fully qualified app name.
11 | We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec).
12 | If release name contains chart name it will be used as a full name.
13 | */}}
14 | {{- define "goinception.fullname" -}}
15 | {{- if .Values.fullnameOverride -}}
16 | {{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" -}}
17 | {{- else -}}
18 | {{- $name := default .Chart.Name .Values.nameOverride -}}
19 | {{- if contains $name .Release.Name -}}
20 | {{- .Release.Name | trunc 63 | trimSuffix "-" -}}
21 | {{- else -}}
22 | {{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}}
23 | {{- end -}}
24 | {{- end -}}
25 | {{- end -}}
26 |
27 | {{/*
28 | Create chart name and version as used by the chart label.
29 | */}}
30 | {{- define "goinception.chart" -}}
31 | {{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}}
32 | {{- end -}}
33 |
--------------------------------------------------------------------------------
/common/static/bootstrap-table/export-libs/jsPDF-AutoTable/LICENSE.txt:
--------------------------------------------------------------------------------
1 | Copyright (c) 2014 Simon Bengtsson, https://github.com/someatoms/jspdf-autotable
2 |
3 | Permission is hereby granted, free of charge, to any person obtaining
4 | a copy of this software and associated documentation files (the
5 | "Software"), to deal in the Software without restriction, including
6 | without limitation the rights to use, copy, modify, merge, publish,
7 | distribute, sublicense, and/or sell copies of the Software, and to
8 | permit persons to whom the Software is furnished to do so, subject to
9 | the following conditions:
10 |
11 | The above copyright notice and this permission notice shall be
12 | included in all copies or substantial portions of the Software.
13 |
14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21 |
--------------------------------------------------------------------------------
/src/init_sql/v1.0_v1.1.0.sql:
--------------------------------------------------------------------------------
1 | -- 表结构变更
2 | ALTER TABLE sql_group
3 | ADD ding_webhook VARCHAR(255) NOT NULL DEFAULT ''
4 | AFTER group_level;
5 | ALTER TABLE sql_workflow
6 | ADD engineer_display VARCHAR(50) NOT NULL DEFAULT ''
7 | AFTER engineer;
8 | ALTER TABLE workflow_audit
9 | ADD create_user_display VARCHAR(50) NOT NULL DEFAULT ''
10 | AFTER create_user;
11 | ALTER TABLE query_privileges_apply
12 | ADD user_display VARCHAR(50) NOT NULL DEFAULT ''
13 | AFTER user_name;
14 | ALTER TABLE query_privileges
15 | ADD user_display VARCHAR(50) NOT NULL DEFAULT ''
16 | AFTER user_name;
17 | ALTER TABLE query_log
18 | ADD user_display VARCHAR(50) NOT NULL DEFAULT ''
19 | AFTER username;
20 |
21 | -- 数据清洗
22 | UPDATE sql_workflow, sql_users
23 | SET engineer_display = display
24 | WHERE engineer = username;
25 | UPDATE workflow_audit, sql_users
26 | SET create_user_display = display
27 | WHERE create_user = username;
28 | UPDATE query_privileges_apply, sql_users
29 | SET user_display = display
30 | WHERE user_name = username;
31 | UPDATE query_privileges, sql_users
32 | SET user_display = display
33 | WHERE user_name = username;
34 | UPDATE query_log, sql_users
35 | SET user_display = display
36 | WHERE query_log.username = sql_users.username;
--------------------------------------------------------------------------------
/src/init_sql/v1.2.0_v1.3.0.sql:
--------------------------------------------------------------------------------
1 | -- 增加工单日志表
2 | CREATE TABLE `workflow_log` (
3 | `id` bigint(20) NOT NULL AUTO_INCREMENT,
4 | `audit_id` bigint(20) NOT NULL DEFAULT '0' COMMENT '工单审批id',
5 | `operation_type` tinyint(4) NOT NULL DEFAULT '0' COMMENT '操作类型,0提交/待审核、1审核通过、2审核不通过、3审核取消/取消执行、4定时执行、5执行工单、6执行结束',
6 | `operation_type_desc` char(10) NOT NULL DEFAULT '' COMMENT '操作类型描述',
7 | `operation_info` varchar(200) NOT NULL DEFAULT '' COMMENT '操作信息',
8 | `operator` varchar(30) NOT NULL DEFAULT '' COMMENT '操作人',
9 | `operator_display` varchar(50) NOT NULL DEFAULT '' COMMENT '操作人中文名',
10 | `operation_time` timestamp(6) NOT NULL DEFAULT CURRENT_TIMESTAMP(6) COMMENT '操作时间',
11 | PRIMARY KEY (`id`),
12 | index idx_audit_id(audit_id)
13 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
14 |
15 | -- 增加菜单权限
16 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
17 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 数据库审核', @content_type_id, 'menu_themis');
18 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 实例管理', @content_type_id, 'menu_instance');
19 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 Binlog2SQL', @content_type_id, 'menu_binlog2sql');
20 |
--------------------------------------------------------------------------------
/src/charts/charts/goinception/templates/ingress.yaml:
--------------------------------------------------------------------------------
1 | {{- if .Values.ingress.enabled -}}
2 | {{- $fullName := include "goinception.fullname" . -}}
3 | {{- $ingressPaths := .Values.ingress.paths -}}
4 | apiVersion: extensions/v1beta1
5 | kind: Ingress
6 | metadata:
7 | name: {{ $fullName }}
8 | labels:
9 | app.kubernetes.io/name: {{ include "goinception.name" . }}
10 | helm.sh/chart: {{ include "goinception.chart" . }}
11 | app.kubernetes.io/instance: {{ .Release.Name }}
12 | app.kubernetes.io/managed-by: {{ .Release.Service }}
13 | {{- with .Values.ingress.annotations }}
14 | annotations:
15 | {{- toYaml . | nindent 4 }}
16 | {{- end }}
17 | spec:
18 | {{- if .Values.ingress.tls }}
19 | tls:
20 | {{- range .Values.ingress.tls }}
21 | - hosts:
22 | {{- range .hosts }}
23 | - {{ . | quote }}
24 | {{- end }}
25 | secretName: {{ .secretName }}
26 | {{- end }}
27 | {{- end }}
28 | rules:
29 | {{- range .Values.ingress.hosts }}
30 | - host: {{ . | quote }}
31 | http:
32 | paths:
33 | {{- range $ingressPaths }}
34 | - path: {{ . }}
35 | backend:
36 | serviceName: {{ $fullName }}
37 | servicePort: http
38 | {{- end }}
39 | {{- end }}
40 | {{- end }}
41 |
--------------------------------------------------------------------------------
/src/charts/charts/inception/templates/ingress.yaml:
--------------------------------------------------------------------------------
1 | {{- if .Values.ingress.enabled -}}
2 | {{- $fullName := include "inception.fullname" . -}}
3 | {{- $ingressPaths := .Values.ingress.paths -}}
4 | apiVersion: extensions/v1beta1
5 | kind: Ingress
6 | metadata:
7 | name: {{ $fullName }}
8 | labels:
9 | app.kubernetes.io/name: {{ include "inception.name" . }}
10 | helm.sh/chart: {{ include "inception.chart" . }}
11 | app.kubernetes.io/instance: {{ .Release.Name }}
12 | app.kubernetes.io/managed-by: {{ .Release.Service }}
13 | {{- with .Values.ingress.annotations }}
14 | annotations:
15 | {{- toYaml . | nindent 4 }}
16 | {{- end }}
17 | spec:
18 | {{- if .Values.ingress.tls }}
19 | tls:
20 | {{- range .Values.ingress.tls }}
21 | - hosts:
22 | {{- range .hosts }}
23 | - {{ . | quote }}
24 | {{- end }}
25 | secretName: {{ .secretName }}
26 | {{- end }}
27 | {{- end }}
28 | rules:
29 | {{- range .Values.ingress.hosts }}
30 | - host: {{ . | quote }}
31 | http:
32 | paths:
33 | {{- range $ingressPaths }}
34 | - path: {{ . }}
35 | backend:
36 | serviceName: {{ $fullName }}
37 | servicePort: http
38 | {{- end }}
39 | {{- end }}
40 | {{- end }}
41 |
--------------------------------------------------------------------------------
/common/static/ace/snippets/sql.js:
--------------------------------------------------------------------------------
1 | define("ace/snippets/sql",["require","exports","module"],function(e,t,n){"use strict";t.snippetText="snippet tbl\n create table ${1:table} (\n ${2:columns}\n );\nsnippet col\n ${1:name} ${2:type} ${3:default ''} ${4:not null}\nsnippet ccol\n ${1:name} varchar2(${2:size}) ${3:default ''} ${4:not null}\nsnippet ncol\n ${1:name} number ${3:default 0} ${4:not null}\nsnippet dcol\n ${1:name} date ${3:default sysdate} ${4:not null}\nsnippet ind\n create index ${3:$1_$2} on ${1:table}(${2:column});\nsnippet uind\n create unique index ${1:name} on ${2:table}(${3:column});\nsnippet tblcom\n comment on table ${1:table} is '${2:comment}';\nsnippet colcom\n comment on column ${1:table}.${2:column} is '${3:comment}';\nsnippet addcol\n alter table ${1:table} add (${2:column} ${3:type});\nsnippet seq\n create sequence ${1:name} start with ${2:1} increment by ${3:1} minvalue ${4:1};\nsnippet s*\n select * from ${1:table}\n",t.scope="sql"}); (function() {
2 | window.require(["ace/snippets/sql"], function(m) {
3 | if (typeof module == "object" && typeof exports == "object" && module) {
4 | module.exports = m;
5 | }
6 | });
7 | })();
8 |
--------------------------------------------------------------------------------
/common/utils/extend_json_encoder.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | import simplejson as json
3 |
4 | from decimal import Decimal
5 | from datetime import datetime, date, timedelta
6 | from functools import singledispatch
7 |
8 |
9 | @singledispatch
10 | def convert(o):
11 | raise TypeError('can not convert type')
12 |
13 |
14 | @convert.register(datetime)
15 | def _(o):
16 | return o.strftime('%Y-%m-%d %H:%M:%S')
17 |
18 |
19 | @convert.register(date)
20 | def _(o):
21 | return o.strftime('%Y-%m-%d')
22 |
23 |
24 | @convert.register(timedelta)
25 | def _(o):
26 | return o.__str__()
27 |
28 |
29 | @convert.register(Decimal)
30 | def _(o):
31 | return float(o)
32 |
33 |
34 | class ExtendJSONEncoder(json.JSONEncoder):
35 | def default(self, obj):
36 | try:
37 | return convert(obj)
38 | except TypeError:
39 | return super(ExtendJSONEncoder, self).default(obj)
40 |
41 |
42 | class ExtendJSONEncoderFTime(json.JSONEncoder):
43 |
44 | def default(self, obj):
45 | try:
46 | if isinstance(obj, datetime):
47 | return obj.isoformat(' ')
48 | else:
49 | return convert(obj)
50 | except TypeError:
51 | return super(ExtendJSONEncoderFTime, self).default(obj)
52 |
--------------------------------------------------------------------------------
/src/charts/templates/ingress.yaml:
--------------------------------------------------------------------------------
1 | {{- if .Values.ingress.enabled -}}
2 | {{- $fullName := include "archery.fullname" . -}}
3 | {{- $ingressPaths := .Values.ingress.paths -}}
4 | {{- $servicePort := .Values.ingress.servicePort -}}
5 | apiVersion: extensions/v1beta1
6 | kind: Ingress
7 | metadata:
8 | name: {{ $fullName }}
9 | labels:
10 | app.kubernetes.io/name: {{ include "archery.name" . }}
11 | helm.sh/chart: {{ include "archery.chart" . }}
12 | app.kubernetes.io/instance: {{ .Release.Name }}
13 | app.kubernetes.io/managed-by: {{ .Release.Service }}
14 | {{- with .Values.ingress.annotations }}
15 | annotations:
16 | {{- toYaml . | nindent 4 }}
17 | {{- end }}
18 | spec:
19 | {{- if .Values.ingress.tls }}
20 | tls:
21 | {{- range .Values.ingress.tls }}
22 | - hosts:
23 | {{- range .hosts }}
24 | - {{ . | quote }}
25 | {{- end }}
26 | secretName: {{ .secretName }}
27 | {{- end }}
28 | {{- end }}
29 | rules:
30 | {{- range .Values.ingress.hosts }}
31 | - host: {{ . | quote }}
32 | http:
33 | paths:
34 | {{- range $ingressPaths }}
35 | - path: {{ . }}
36 | backend:
37 | serviceName: {{ $fullName }}
38 | servicePort: {{ $servicePort }}
39 | {{- end }}
40 | {{- end }}
41 | {{- end }}
42 |
--------------------------------------------------------------------------------
/common/static/bootstrap-select/js/i18n/defaults-zh_CN.js:
--------------------------------------------------------------------------------
1 | /*!
2 | * Bootstrap-select v1.13.9 (https://developer.snapappointments.com/bootstrap-select)
3 | *
4 | * Copyright 2012-2019 SnapAppointments, LLC
5 | * Licensed under MIT (https://github.com/snapappointments/bootstrap-select/blob/master/LICENSE)
6 | */
7 |
8 | (function (root, factory) {
9 | if (root === undefined && window !== undefined) root = window;
10 | if (typeof define === 'function' && define.amd) {
11 | // AMD. Register as an anonymous module unless amdModuleId is set
12 | define(["jquery"], function (a0) {
13 | return (factory(a0));
14 | });
15 | } else if (typeof module === 'object' && module.exports) {
16 | // Node. Does not work with strict CommonJS, but
17 | // only CommonJS-like environments that support module.exports,
18 | // like Node.
19 | module.exports = factory(require("jquery"));
20 | } else {
21 | factory(root["jQuery"]);
22 | }
23 | }(this, function (jQuery) {
24 |
25 | (function ($) {
26 | $.fn.selectpicker.defaults = {
27 | noneSelectedText: '没有选中任何项',
28 | noneResultsText: '没有找到匹配项',
29 | countSelectedText: '选中{1}中的{0}项',
30 | maxOptionsText: ['超出限制 (最多选择{n}项)', '组选择超出限制(最多选择{n}组)'],
31 | multipleSeparator: ', ',
32 | selectAllText: '全选',
33 | deselectAllText: '取消全选'
34 | };
35 | })(jQuery);
36 |
37 |
38 | }));
39 | //# sourceMappingURL=defaults-zh_CN.js.map
--------------------------------------------------------------------------------
/sql/locale/zh_Hans/LC_MESSAGES/django.po:
--------------------------------------------------------------------------------
1 | # SOME DESCRIPTIVE TITLE.
2 | # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
3 | # This file is distributed under the same license as the PACKAGE package.
4 | # FIRST AUTHOR , YEAR.
5 | #
6 | #, fuzzy
7 | # 更改翻译文件后务必使用 python manage.py compilemessages 编译翻译文件
8 | msgid ""
9 | msgstr ""
10 | "Project-Id-Version: PACKAGE VERSION\n"
11 | "Report-Msgid-Bugs-To: \n"
12 | "POT-Creation-Date: 2019-02-14 19:34+0800\n"
13 | "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
14 | "Last-Translator: FULL NAME \n"
15 | "Language-Team: LANGUAGE \n"
16 | "Language: \n"
17 | "MIME-Version: 1.0\n"
18 | "Content-Type: text/plain; charset=UTF-8\n"
19 | "Content-Transfer-Encoding: 8bit\n"
20 |
21 | #: .\sql\models.py:111
22 | msgid "workflow_finish"
23 | msgstr "已正常结束"
24 |
25 | #: .\sql\models.py:112
26 | msgid "workflow_abort"
27 | msgstr "人工终止流程"
28 |
29 | #: .\sql\models.py:113
30 | msgid "workflow_manreviewing"
31 | msgstr "等待审核人审核"
32 |
33 | #: .\sql\models.py:114
34 | msgid "workflow_review_pass"
35 | msgstr "审核通过"
36 |
37 | #: .\sql\models.py:115
38 | msgid "workflow_timingtask"
39 | msgstr "定时执行"
40 |
41 | #: .\sql\models.py:116
42 | msgid "workflow_executing"
43 | msgstr "执行中"
44 |
45 | #: .\sql\models.py:117
46 | msgid "workflow_autoreviewwrong"
47 | msgstr "自动审核不通过"
48 |
49 | #: .\sql\models.py:118
50 | msgid "workflow_exception"
51 | msgstr "执行有异常"
52 |
--------------------------------------------------------------------------------
/sql/locale/zh_Hans/LC_MESSAGES/djangojs.po:
--------------------------------------------------------------------------------
1 | # SOME DESCRIPTIVE TITLE.
2 | # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
3 | # This file is distributed under the same license as the PACKAGE package.
4 | # FIRST AUTHOR , YEAR.
5 | #
6 | #, fuzzy
7 | # 更改翻译文件后务必使用 python manage.py compilemessages 编译翻译文件
8 | msgid ""
9 | msgstr ""
10 | "Project-Id-Version: PACKAGE VERSION\n"
11 | "Report-Msgid-Bugs-To: \n"
12 | "POT-Creation-Date: 2019-02-14 19:34+0800\n"
13 | "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
14 | "Last-Translator: FULL NAME \n"
15 | "Language-Team: LANGUAGE \n"
16 | "Language: \n"
17 | "MIME-Version: 1.0\n"
18 | "Content-Type: text/plain; charset=UTF-8\n"
19 | "Content-Transfer-Encoding: 8bit\n"
20 |
21 | #: .\sql\models.py:111
22 | msgid "workflow_finish"
23 | msgstr "已正常结束"
24 |
25 | #: .\sql\models.py:112
26 | msgid "workflow_abort"
27 | msgstr "人工终止流程"
28 |
29 | #: .\sql\models.py:113
30 | msgid "workflow_manreviewing"
31 | msgstr "等待审核人审核"
32 |
33 | #: .\sql\models.py:114
34 | msgid "workflow_review_pass"
35 | msgstr "审核通过"
36 |
37 | #: .\sql\models.py:115
38 | msgid "workflow_timingtask"
39 | msgstr "定时执行"
40 |
41 | #: .\sql\models.py:116
42 | msgid "workflow_executing"
43 | msgstr "执行中"
44 |
45 | #: .\sql\models.py:117
46 | msgid "workflow_autoreviewwrong"
47 | msgstr "自动审核不通过"
48 |
49 | #: .\sql\models.py:118
50 | msgid "workflow_exception"
51 | msgstr "执行有异常"
52 |
--------------------------------------------------------------------------------
/sql/locale/en/LC_MESSAGES/django.po:
--------------------------------------------------------------------------------
1 | # SOME DESCRIPTIVE TITLE.
2 | # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
3 | # This file is distributed under the same license as the PACKAGE package.
4 | # FIRST AUTHOR , YEAR.
5 | #
6 | #, fuzzy
7 | # 更改翻译文件后务必使用 python manage.py compilemessages 编译翻译文件
8 | msgid ""
9 | msgstr ""
10 | "Project-Id-Version: PACKAGE VERSION\n"
11 | "Report-Msgid-Bugs-To: \n"
12 | "POT-Creation-Date: 2019-02-14 19:47+0800\n"
13 | "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
14 | "Last-Translator: FULL NAME \n"
15 | "Language-Team: LANGUAGE \n"
16 | "Language: \n"
17 | "MIME-Version: 1.0\n"
18 | "Content-Type: text/plain; charset=UTF-8\n"
19 | "Content-Transfer-Encoding: 8bit\n"
20 | "Plural-Forms: nplurals=2; plural=(n != 1);\n"
21 | #: .\sql\models.py:111
22 | msgid "workflow_finish"
23 | msgstr "Workflow finished"
24 |
25 | #: .\sql\models.py:112
26 | msgid "workflow_abort"
27 | msgstr "Aborted"
28 |
29 | #: .\sql\models.py:113
30 | msgid "workflow_manreviewing"
31 | msgstr "Reviewing"
32 |
33 | #: .\sql\models.py:114
34 | msgid "workflow_review_pass"
35 | msgstr "Review passed"
36 |
37 | #: .\sql\models.py:115
38 | msgid "workflow_timingtask"
39 | msgstr "Scheduled"
40 |
41 | #: .\sql\models.py:116
42 | msgid "workflow_executing"
43 | msgstr "Executing"
44 |
45 | #: .\sql\models.py:117
46 | msgid "workflow_autoreviewwrong"
47 | msgstr "Rejected"
48 |
49 | #: .\sql\models.py:118
50 | msgid "workflow_exception"
51 | msgstr "Exceptions during execution"
52 |
--------------------------------------------------------------------------------
/sql/locale/en/LC_MESSAGES/djangojs.po:
--------------------------------------------------------------------------------
1 | # SOME DESCRIPTIVE TITLE.
2 | # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
3 | # This file is distributed under the same license as the PACKAGE package.
4 | # FIRST AUTHOR , YEAR.
5 | #
6 | #, fuzzy
7 | # 更改翻译文件后务必使用 python manage.py compilemessages 编译翻译文件
8 | msgid ""
9 | msgstr ""
10 | "Project-Id-Version: PACKAGE VERSION\n"
11 | "Report-Msgid-Bugs-To: \n"
12 | "POT-Creation-Date: 2019-02-14 19:47+0800\n"
13 | "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
14 | "Last-Translator: FULL NAME \n"
15 | "Language-Team: LANGUAGE \n"
16 | "Language: \n"
17 | "MIME-Version: 1.0\n"
18 | "Content-Type: text/plain; charset=UTF-8\n"
19 | "Content-Transfer-Encoding: 8bit\n"
20 | "Plural-Forms: nplurals=2; plural=(n != 1);\n"
21 | #: .\sql\models.py:111
22 | msgid "workflow_finish"
23 | msgstr "Workflow finished"
24 |
25 | #: .\sql\models.py:112
26 | msgid "workflow_abort"
27 | msgstr "Aborted"
28 |
29 | #: .\sql\models.py:113
30 | msgid "workflow_manreviewing"
31 | msgstr "Reviewing"
32 |
33 | #: .\sql\models.py:114
34 | msgid "workflow_review_pass"
35 | msgstr "Review passed"
36 |
37 | #: .\sql\models.py:115
38 | msgid "workflow_timingtask"
39 | msgstr "Scheduled"
40 |
41 | #: .\sql\models.py:116
42 | msgid "workflow_executing"
43 | msgstr "Executing"
44 |
45 | #: .\sql\models.py:117
46 | msgid "workflow_autoreviewwrong"
47 | msgstr "Rejected"
48 |
49 | #: .\sql\models.py:118
50 | msgid "workflow_exception"
51 | msgstr "Exceptions during execution"
52 |
--------------------------------------------------------------------------------
/src/script/analysis_slow_query.sh:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 | DIR="$( cd "$( dirname "$0" )" && pwd )"
3 | cd ${DIR}
4 |
5 | #配置Archery数据库的连接地址
6 | archery_db_host="127.0.0.1"
7 | archery_db_port=3306
8 | archery_db_user="root"
9 | archery_db_password="123456"
10 | archery_db_database="archery"
11 |
12 | #被分析实例的慢日志位置,建议定期清理日志文件,否则会影响分析效率
13 | slowquery_file="/home/mysql/log_slow.log"
14 |
15 | #pt-query-digest可执行文件路径
16 | pt_query_digest="/usr/bin/pt-query-digest"
17 |
18 | #被分析实例的连接信息
19 | hostname="mysql_host:mysql_port" # 需要和Archery实例配置中的内容保持一致,用于筛选,配置错误会导致数据无法展示
20 |
21 | #获取上次分析时间,初始化时请删除last_analysis_time_$hostname文件,可分析全部日志数据
22 | if [[ -s last_analysis_time_${hostname} ]]; then
23 | last_analysis_time=`cat last_analysis_time_${hostname}`
24 | else
25 | last_analysis_time='1000-01-01 00:00:00'
26 | fi
27 |
28 | #收集日志
29 | #RDS需要增加--no-version-check选项
30 | ${pt_query_digest} \
31 | --user=${archery_db_user} --password=${archery_db_password} --port=${archery_db_port} \
32 | --review h=${archery_db_host},D=${archery_db_database},t=mysql_slow_query_review \
33 | --history h=${archery_db_host},D=${archery_db_database},t=mysql_slow_query_review_history \
34 | --no-report --limit=100% --charset=utf8 \
35 | --since "$last_analysis_time" \
36 | --filter="\$event->{Bytes} = length(\$event->{arg}) and \$event->{hostname}=\"$hostname\" and \$event->{client}=\$event->{ip} " \
37 | ${slowquery_file} > /tmp/analysis_slow_query.log
38 |
39 | if [[ $? -ne 0 ]]; then
40 | echo "failed"
41 | else
42 | echo `date +"%Y-%m-%d %H:%M:%S"`>last_analysis_time_${hostname}
43 | fi
44 |
--------------------------------------------------------------------------------
/common/utils/permission.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | import simplejson as json
3 | from django.shortcuts import render
4 | from django.http import HttpResponse
5 |
6 |
7 | # 管理员操作权限验证
8 | def superuser_required(func):
9 | def wrapper(request, *args, **kw):
10 | # 获取用户信息,权限验证
11 | user = request.user
12 |
13 | if user.is_superuser is False:
14 | if request.is_ajax():
15 | result = {'status': 1, 'msg': '您无权操作,请联系管理员', 'data': []}
16 | return HttpResponse(json.dumps(result), content_type='application/json')
17 | else:
18 | context = {'errMsg': "您无权操作,请联系管理员"}
19 | return render(request, "error.html", context)
20 |
21 | return func(request, *args, **kw)
22 |
23 | return wrapper
24 |
25 |
26 | # 角色操作权限验证
27 | def role_required(roles=()):
28 | def _deco(func):
29 | def wrapper(request, *args, **kw):
30 | # 获取用户信息,权限验证
31 | user = request.user
32 | if user.role not in roles and user.is_superuser is False:
33 | if request.is_ajax():
34 | result = {'status': 1, 'msg': '您无权操作,请联系管理员', 'data': []}
35 | return HttpResponse(json.dumps(result), content_type='application/json')
36 | else:
37 | context = {'errMsg': "您无权操作,请联系管理员"}
38 | return render(request, "error.html", context)
39 |
40 | return func(request, *args, **kw)
41 |
42 | return wrapper
43 |
44 | return _deco
45 |
--------------------------------------------------------------------------------
/common/static/bootstrap-table/js/bootstrap-table-zh-CN.min.js:
--------------------------------------------------------------------------------
1 | /**
2 | * bootstrap-table - An extended Bootstrap table with radio, checkbox, sort, pagination, and other added features. (supports twitter bootstrap v2 and v3).
3 | *
4 | * @version v1.14.1
5 | * @homepage https://bootstrap-table.com
6 | * @author wenzhixin (http://wenzhixin.net.cn/)
7 | * @license MIT
8 | */
9 |
10 | (function(a,b){if('function'==typeof define&&define.amd)define([],b);else if('undefined'!=typeof exports)b();else{b(),a.bootstrapTableZhCN={exports:{}}.exports}})(this,function(){'use strict';(function(a){a.fn.bootstrapTable.locales['zh-CN']={formatLoadingMessage:function(){return'\u6B63\u5728\u52AA\u529B\u5730\u52A0\u8F7D\u6570\u636E\u4E2D\uFF0C\u8BF7\u7A0D\u5019\u2026\u2026'},formatRecordsPerPage:function(a){return'\u6BCF\u9875\u663E\u793A '+a+' \u6761\u8BB0\u5F55'},formatShowingRows:function(a,b,c){return'\u663E\u793A\u7B2C '+a+' \u5230\u7B2C '+b+' \u6761\u8BB0\u5F55\uFF0C\u603B\u5171 '+c+' \u6761\u8BB0\u5F55'},formatSearch:function(){return'\u641C\u7D22'},formatNoMatches:function(){return'\u6CA1\u6709\u627E\u5230\u5339\u914D\u7684\u8BB0\u5F55'},formatPaginationSwitch:function(){return'\u9690\u85CF/\u663E\u793A\u5206\u9875'},formatRefresh:function(){return'\u5237\u65B0'},formatToggle:function(){return'\u5207\u6362'},formatColumns:function(){return'\u5217'},formatExport:function(){return'\u5BFC\u51FA\u6570\u636E'},formatClearFilters:function(){return'\u6E05\u7A7A\u8FC7\u6EE4'}},a.extend(a.fn.bootstrapTable.defaults,a.fn.bootstrapTable.locales['zh-CN'])})(jQuery)});
--------------------------------------------------------------------------------
/src/charts/templates/NOTES.txt:
--------------------------------------------------------------------------------
1 | 1. Get the application URL by running these commands:
2 | {{- if .Values.ingress.enabled }}
3 | {{- range $host := .Values.ingress.hosts }}
4 | {{- range $.Values.ingress.paths }}
5 | http{{ if $.Values.ingress.tls }}s{{ end }}://{{ $host }}{{ . }}
6 | {{- end }}
7 | {{- end }}
8 | {{- else if contains "NodePort" .Values.service.type }}
9 | export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "archery.fullname" . }})
10 | export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}")
11 | echo http://$NODE_IP:$NODE_PORT
12 | {{- else if contains "LoadBalancer" .Values.service.type }}
13 | NOTE: It may take a few minutes for the LoadBalancer IP to be available.
14 | You can watch the status of by running 'kubectl get svc -w {{ include "archery.fullname" . }}'
15 | export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ include "archery.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}')
16 | echo http://$SERVICE_IP:{{ .Values.service.port }}
17 | {{- else if contains "ClusterIP" .Values.service.type }}
18 | export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "app.kubernetes.io/name={{ include "archery.name" . }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}")
19 | echo "Visit http://127.0.0.1:9123 to use your application"
20 | kubectl port-forward $POD_NAME 9123:9123
21 | {{- end }}
22 |
--------------------------------------------------------------------------------
/common/utils/aes_decryptor.py:
--------------------------------------------------------------------------------
1 | from Crypto.Cipher import AES
2 | from binascii import b2a_hex, a2b_hex
3 |
4 |
5 | class Prpcrypt():
6 | def __init__(self):
7 | self.key = 'eCcGFZQj6PNoSSma31LR39rTzTbLkU8E'.encode('utf-8')
8 | self.mode = AES.MODE_CBC
9 |
10 | # 加密函数,如果text不足16位就用空格补足为16位,
11 | # 如果大于16当时不是16的倍数,那就补足为16的倍数。
12 | def encrypt(self, text):
13 | cryptor = AES.new(self.key, self.mode, b'0000000000000000')
14 | # 这里密钥key 长度必须为16(AES-128),
15 | # 24(AES-192),或者32 (AES-256)Bytes 长度
16 | # 目前AES-128 足够目前使用
17 | length = 16
18 | count = len(text)
19 | if count < length:
20 | add = (length - count)
21 | # \0 backspace
22 | text = text + ('\0' * add)
23 | elif count > length:
24 | add = (length - (count % length))
25 | text = text + ('\0' * add)
26 | self.ciphertext = cryptor.encrypt(text.encode('utf-8'))
27 | # 因为AES加密时候得到的字符串不一定是ascii字符集的,输出到终端或者保存时候可能存在问题
28 | # 所以这里统一把加密后的字符串转化为16进制字符串
29 | return b2a_hex(self.ciphertext).decode(encoding='utf-8')
30 |
31 | # 解密后,去掉补足的空格用strip() 去掉
32 | def decrypt(self, text):
33 | cryptor = AES.new(self.key, self.mode, b'0000000000000000')
34 | plain_text = cryptor.decrypt(a2b_hex(text))
35 | return plain_text.decode().rstrip('\0')
36 |
37 |
38 | if __name__ == '__main__':
39 | pc = Prpcrypt() # 初始化密钥
40 | e = pc.encrypt('123456') # 加密
41 | d = pc.decrypt(e) # 解密
42 | print("加密:", str(e))
43 | print("解密:", str(d))
44 |
--------------------------------------------------------------------------------
/src/charts/charts/inception/templates/NOTES.txt:
--------------------------------------------------------------------------------
1 | 1. Get the application URL by running these commands:
2 | {{- if .Values.ingress.enabled }}
3 | {{- range $host := .Values.ingress.hosts }}
4 | {{- range $.Values.ingress.paths }}
5 | http{{ if $.Values.ingress.tls }}s{{ end }}://{{ $host }}{{ . }}
6 | {{- end }}
7 | {{- end }}
8 | {{- else if contains "NodePort" .Values.service.type }}
9 | export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "inception.fullname" . }})
10 | export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}")
11 | echo http://$NODE_IP:$NODE_PORT
12 | {{- else if contains "LoadBalancer" .Values.service.type }}
13 | NOTE: It may take a few minutes for the LoadBalancer IP to be available.
14 | You can watch the status of by running 'kubectl get svc -w {{ include "inception.fullname" . }}'
15 | export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ include "inception.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}')
16 | echo http://$SERVICE_IP:{{ .Values.service.port }}
17 | {{- else if contains "ClusterIP" .Values.service.type }}
18 | export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "app.kubernetes.io/name={{ include "inception.name" . }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}")
19 | echo "Visit http://127.0.0.1:8080 to use your application"
20 | kubectl port-forward $POD_NAME 8080:80
21 | {{- end }}
22 |
--------------------------------------------------------------------------------
/src/charts/charts/goinception/templates/NOTES.txt:
--------------------------------------------------------------------------------
1 | 1. Get the application URL by running these commands:
2 | {{- if .Values.ingress.enabled }}
3 | {{- range $host := .Values.ingress.hosts }}
4 | {{- range $.Values.ingress.paths }}
5 | http{{ if $.Values.ingress.tls }}s{{ end }}://{{ $host }}{{ . }}
6 | {{- end }}
7 | {{- end }}
8 | {{- else if contains "NodePort" .Values.service.type }}
9 | export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "goinception.fullname" . }})
10 | export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}")
11 | echo http://$NODE_IP:$NODE_PORT
12 | {{- else if contains "LoadBalancer" .Values.service.type }}
13 | NOTE: It may take a few minutes for the LoadBalancer IP to be available.
14 | You can watch the status of by running 'kubectl get svc -w {{ include "goinception.fullname" . }}'
15 | export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ include "goinception.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}')
16 | echo http://$SERVICE_IP:{{ .Values.service.port }}
17 | {{- else if contains "ClusterIP" .Values.service.type }}
18 | export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "app.kubernetes.io/name={{ include "goinception.name" . }},app.kubernetes.io/instance={{ .Release.Name }}" -o jsonpath="{.items[0].metadata.name}")
19 | echo "Visit http://127.0.0.1:8080 to use your application"
20 | kubectl port-forward $POD_NAME 8080:80
21 | {{- end }}
22 |
--------------------------------------------------------------------------------
/sql/utils/tasks.py:
--------------------------------------------------------------------------------
1 | # -*- coding:utf-8 -*-
2 | from django_q.tasks import schedule
3 | from django_q.models import Schedule
4 |
5 | import logging
6 |
7 | logger = logging.getLogger('default')
8 |
9 |
10 | def add_sql_schedule(name, run_date, workflow_id):
11 | """添加/修改sql定时任务"""
12 | del_schedule(name)
13 | schedule('sql.utils.execute_sql.execute', workflow_id,
14 | hook='sql.utils.execute_sql.execute_callback',
15 | name=name, schedule_type='O', next_run=run_date, repeats=1, timeout=-1)
16 | logger.debug(f"添加SQL定时执行任务:{name} 执行时间:{run_date}")
17 |
18 |
19 | def add_kill_conn_schedule(name, run_date, instance_id, thread_id):
20 | """添加/修改终止数据库连接的定时任务"""
21 | del_schedule(name)
22 | schedule('sql.query.kill_query_conn', instance_id, thread_id,
23 | name=name, schedule_type='O', next_run=run_date, repeats=1, timeout=-1)
24 |
25 |
26 | def add_sync_ding_user_schedule():
27 | """添加钉钉同步用户定时任务"""
28 | del_schedule(name='同步钉钉用户ID')
29 | schedule('common.utils.ding_api.sync_ding_user_id',
30 | name='同步钉钉用户ID', schedule_type='D', repeats=-1, timeout=-1)
31 |
32 |
33 | def del_schedule(name):
34 | """删除task"""
35 | try:
36 | sql_schedule = Schedule.objects.get(name=name)
37 | Schedule.delete(sql_schedule)
38 | # logger.debug(f'删除task:{name}')
39 | except Schedule.DoesNotExist:
40 | pass
41 |
42 |
43 | def task_info(name):
44 | """获取定时任务详情"""
45 | try:
46 | sql_schedule = Schedule.objects.get(name=name)
47 | return sql_schedule
48 | except Schedule.DoesNotExist:
49 | pass
50 |
--------------------------------------------------------------------------------
/src/docker-compose/docker-compose.yml:
--------------------------------------------------------------------------------
1 | version: '3'
2 |
3 | services:
4 | redis:
5 | image: redis:5
6 | container_name: redis
7 | restart: always
8 | command: redis-server --requirepass 123456
9 | expose:
10 | - "6379"
11 |
12 | mysql:
13 | image: mysql:5.7
14 | container_name: mysql
15 | restart: always
16 | ports:
17 | - "3306:3306"
18 | volumes:
19 | - "./mysql/my.cnf:/etc/mysql/my.cnf"
20 | - "./mysql/datadir:/var/lib/mysql"
21 | environment:
22 | MYSQL_DATABASE: archery
23 | MYSQL_ROOT_PASSWORD: 123456
24 |
25 | inception:
26 | image: hhyo/inception
27 | container_name: inception
28 | restart: always
29 | expose:
30 | - "6669"
31 | volumes:
32 | - "./inception/inc.cnf:/etc/inc.cnf"
33 |
34 | goinception:
35 | image: hanchuanchuan/goinception
36 | container_name: goinception
37 | restart: always
38 | expose:
39 | - "4000"
40 | volumes:
41 | - "./inception/config.toml:/etc/config.toml"
42 |
43 | archery:
44 | image: hhyo/archery:1.7.7
45 | container_name: archery
46 | restart: always
47 | ports:
48 | - "9123:9123"
49 | volumes:
50 | - "./archery/settings.py:/opt/archery/archery/settings.py"
51 | - "./archery/soar.yaml:/etc/soar.yaml"
52 | - "./archery/docs.md:/opt/archery/docs/docs.md"
53 | - "./archery/downloads:/opt/archery/downloads"
54 | - "./archery/sql/migrations:/opt/archery/sql/migrations"
55 | - "./archery/logs:/opt/archery/logs"
56 | entrypoint: "dockerize -wait tcp://mysql:3306 -wait tcp://redis:6379 -timeout 60s /opt/archery/src/docker/startup.sh"
57 | environment:
58 | NGINX_PORT: 9123
59 |
--------------------------------------------------------------------------------
/sql/plugins/schemasync.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: schemasync.py
6 | @time: 2019/03/05
7 | """
8 | __author__ = 'hhyo'
9 |
10 | from common.config import SysConfig
11 | from sql.plugins.plugin import Plugin
12 |
13 |
14 | class SchemaSync(Plugin):
15 |
16 | def __init__(self):
17 | self.path = 'schemasync'
18 | self.required_args = []
19 | self.disable_args = []
20 | super(Plugin, self).__init__()
21 |
22 | def generate_args2cmd(self, args, shell):
23 | """
24 | 转换请求参数为命令行
25 | :param args:
26 | :param shell:
27 | :return:
28 | """
29 | k_options = ['sync-auto-inc', 'sync-comments']
30 | kv_options = ['tag', 'output-directory', 'log-directory']
31 | v_options = ['source', 'target']
32 | if shell:
33 | cmd_args = self.path if self.path else ''
34 | for name, value in args.items():
35 | if name in k_options and value:
36 | cmd_args += f' --{name}'
37 | elif name in kv_options:
38 | cmd_args += f' --{name}={value}'
39 | elif name in v_options:
40 | cmd_args += f' {value}'
41 | else:
42 | cmd_args = [self.path]
43 | for name, value in args.items():
44 | if name in k_options and value:
45 | cmd_args.append(f'--{name}')
46 | elif name in kv_options:
47 | cmd_args.append(f'--{name}')
48 | cmd_args.append(f'{value}')
49 | elif name in ['source', 'target']:
50 | cmd_args.append(f'{value}')
51 | return cmd_args
52 |
--------------------------------------------------------------------------------
/sql/plugins/pt_archiver.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: pt_archiver.py
6 | @time: 2020/01/10
7 | """
8 | from common.config import SysConfig
9 | from sql.plugins.plugin import Plugin
10 |
11 | __author__ = 'hhyo'
12 |
13 |
14 | class PtArchiver(Plugin):
15 | """
16 | pt-archiver归档数据
17 | """
18 |
19 | def __init__(self):
20 | self.path = 'pt-archiver'
21 | self.required_args = []
22 | self.disable_args = ['analyze']
23 | super(Plugin, self).__init__()
24 |
25 | def generate_args2cmd(self, args, shell):
26 | """
27 | 转换请求参数为命令行
28 | :param args:
29 | :param shell:
30 | :return:
31 | """
32 | k_options = ['no-version-check', 'statistics', 'bulk-insert', 'bulk-delete', 'purge', 'no-delete']
33 | kv_options = ['source', 'dest', 'file', 'where', 'progress', 'charset', 'limit', 'txn-size', 'sleep']
34 | if shell:
35 | cmd_args = self.path if self.path else ''
36 | for name, value in args.items():
37 | if name in k_options and value:
38 | cmd_args += f' --{name}'
39 | elif name in kv_options:
40 | if name == 'where':
41 | cmd_args += f' --{name} "{value}"'
42 | else:
43 | cmd_args += f' --{name} {value}'
44 | else:
45 | cmd_args = [self.path]
46 | for name, value in args.items():
47 | if name in k_options and value:
48 | cmd_args.append(f'--{name}')
49 | elif name in kv_options:
50 | cmd_args.append(f'--{name}')
51 | cmd_args.append(f'{value}')
52 | return cmd_args
53 |
--------------------------------------------------------------------------------
/common/static/sb-admin-2/js/sb-admin-2.js:
--------------------------------------------------------------------------------
1 | /*!
2 | * Start Bootstrap - SB Admin 2 v3.3.7+1 (http://startbootstrap.com/template-overviews/sb-admin-2)
3 | * Copyright 2013-2016 Start Bootstrap
4 | * Licensed under MIT (https://github.com/BlackrockDigital/startbootstrap/blob/gh-pages/LICENSE)
5 | */
6 | $(function() {
7 | $('#side-menu').metisMenu();
8 | });
9 |
10 | //Loads the correct sidebar on window load,
11 | //collapses the sidebar on window resize.
12 | // Sets the min-height of #page-wrapper to window size
13 | $(function() {
14 | $(window).bind("load resize", function() {
15 | var topOffset = 50;
16 | var width = (this.window.innerWidth > 0) ? this.window.innerWidth : this.screen.width;
17 | if (width < 768) {
18 | $('div.navbar-collapse').addClass('collapse');
19 | topOffset = 100; // 2-row-menu
20 | } else {
21 | $('div.navbar-collapse').removeClass('collapse');
22 | }
23 |
24 | var height = ((this.window.innerHeight > 0) ? this.window.innerHeight : this.screen.height) - 1;
25 | height = height - topOffset;
26 | if (height < 1) height = 1;
27 | if (height > topOffset) {
28 | $("#page-wrapper").css("min-height", (height) + "px");
29 | }
30 | });
31 |
32 | var url = window.location;
33 | // var element = $('ul.nav a').filter(function() {
34 | // return this.href == url;
35 | // }).addClass('active').parent().parent().addClass('in').parent();
36 | var element = $('ul.nav a').filter(function() {
37 | return this.href == url;
38 | }).addClass('active').parent();
39 |
40 | while (true) {
41 | if (element.is('li')) {
42 | element = element.parent().addClass('in').parent();
43 | } else {
44 | break;
45 | }
46 | }
47 | });
48 |
--------------------------------------------------------------------------------
/sql/utils/resource_group.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 |
3 | from sql.models import Users, Instance, ResourceGroup
4 |
5 |
6 | def user_groups(user):
7 | """
8 | 获取用户关联资源组列表
9 | :param user:
10 | :return:
11 | """
12 | if user.is_superuser:
13 | group_list = [group for group in ResourceGroup.objects.filter(is_deleted=0)]
14 | else:
15 | group_list = [group for group in Users.objects.get(id=user.id).resource_group.filter(is_deleted=0)]
16 | return group_list
17 |
18 |
19 | def user_instances(user, type=None, db_type=None, tag_codes=None):
20 | """
21 | 获取用户实例列表(通过资源组间接关联)
22 | :param user:
23 | :param type: 实例类型 all:全部,master主库,salve从库
24 | :param db_type: 数据库类型, ['mysql','mssql']
25 | :param tag_codes: 标签code列表, ['can_write', 'can_read']
26 | :return:
27 | """
28 | # 拥有所有实例权限的用户
29 | if user.has_perm('sql.query_all_instances'):
30 | instances = Instance.objects.all()
31 | else:
32 | # 先获取用户关联的资源组
33 | resource_groups = ResourceGroup.objects.filter(users=user, is_deleted=0)
34 | # 再获取实例
35 | instances = Instance.objects.filter(resource_group__in=resource_groups)
36 | # 过滤type
37 | if type:
38 | instances = instances.filter(type=type)
39 |
40 | # 过滤db_type
41 | if db_type:
42 | instances = instances.filter(db_type__in=db_type)
43 |
44 | # 过滤tag
45 | if tag_codes:
46 | for tag_code in tag_codes:
47 | instances = instances.filter(instance_tag__tag_code=tag_code, instance_tag__active=True)
48 | return instances.distinct()
49 |
50 |
51 | def auth_group_users(auth_group_names, group_id):
52 | """
53 | 获取资源组内关联指定权限组的用户
54 | :param auth_group_names: 权限组名称list
55 | :param group_id: 资源组ID
56 | :return:
57 | """
58 | # 获取资源组关联的用户
59 | users = ResourceGroup.objects.get(group_id=group_id).users_set.all()
60 | # 过滤在该权限组中的用户
61 | users = users.filter(groups__name__in=auth_group_names)
62 | return users
63 |
--------------------------------------------------------------------------------
/src/charts/charts/inception/templates/deployment.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: apps/v1beta2
2 | kind: Deployment
3 | metadata:
4 | name: {{ include "inception.fullname" . }}
5 | labels:
6 | app.kubernetes.io/name: {{ include "inception.name" . }}
7 | helm.sh/chart: {{ include "inception.chart" . }}
8 | app.kubernetes.io/instance: {{ .Release.Name }}
9 | app.kubernetes.io/managed-by: {{ .Release.Service }}
10 | spec:
11 | replicas: {{ .Values.replicaCount }}
12 | selector:
13 | matchLabels:
14 | app.kubernetes.io/name: {{ include "inception.name" . }}
15 | app.kubernetes.io/instance: {{ .Release.Name }}
16 | template:
17 | metadata:
18 | labels:
19 | app.kubernetes.io/name: {{ include "inception.name" . }}
20 | app.kubernetes.io/instance: {{ .Release.Name }}
21 | spec:
22 | containers:
23 | - name: {{ .Chart.Name }}
24 | image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}"
25 | imagePullPolicy: {{ .Values.image.pullPolicy }}
26 | ports:
27 | - name: inception
28 | containerPort: 6669
29 | protocol: TCP
30 | livenessProbe:
31 | tcpSocket:
32 | port: inception
33 | resources:
34 | {{- toYaml .Values.resources | nindent 12 }}
35 | {{- with .Values.volumeMounts }}
36 | volumeMounts:
37 | - name: inception-config-volume
38 | subPath: inc.cnf
39 | mountPath: /etc/inc.cnf
40 | {{- end }}
41 | {{- with .Values.volumes }}
42 | volumes:
43 | {{- toYaml . | nindent 8 }}
44 | {{- end }}
45 | {{- with .Values.nodeSelector }}
46 | nodeSelector:
47 | {{- toYaml . | nindent 8 }}
48 | {{- end }}
49 | {{- with .Values.affinity }}
50 | affinity:
51 | {{- toYaml . | nindent 8 }}
52 | {{- end }}
53 | {{- with .Values.tolerations }}
54 | tolerations:
55 | {{- toYaml . | nindent 8 }}
56 | {{- end }}
57 |
--------------------------------------------------------------------------------
/src/charts/charts/goinception/templates/deployment.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: apps/v1beta2
2 | kind: Deployment
3 | metadata:
4 | name: {{ include "goinception.fullname" . }}
5 | labels:
6 | app.kubernetes.io/name: {{ include "goinception.name" . }}
7 | helm.sh/chart: {{ include "goinception.chart" . }}
8 | app.kubernetes.io/instance: {{ .Release.Name }}
9 | app.kubernetes.io/managed-by: {{ .Release.Service }}
10 | spec:
11 | replicas: {{ .Values.replicaCount }}
12 | selector:
13 | matchLabels:
14 | app.kubernetes.io/name: {{ include "goinception.name" . }}
15 | app.kubernetes.io/instance: {{ .Release.Name }}
16 | template:
17 | metadata:
18 | labels:
19 | app.kubernetes.io/name: {{ include "goinception.name" . }}
20 | app.kubernetes.io/instance: {{ .Release.Name }}
21 | spec:
22 | containers:
23 | - name: {{ .Chart.Name }}
24 | image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}"
25 | imagePullPolicy: {{ .Values.image.pullPolicy }}
26 | ports:
27 | - name: goinception
28 | containerPort: 4000
29 | protocol: TCP
30 | livenessProbe:
31 | tcpSocket:
32 | port: goinception
33 | resources:
34 | {{- toYaml .Values.resources | nindent 12 }}
35 | {{- with .Values.volumeMounts }}
36 | volumeMounts:
37 | - name: goinception-config-volume
38 | subPath: config.toml
39 | mountPath: /etc/config.toml
40 | {{- end }}
41 | {{- with .Values.volumes }}
42 | volumes:
43 | {{- toYaml . | nindent 8 }}
44 | {{- end }}
45 | {{- with .Values.nodeSelector }}
46 | nodeSelector:
47 | {{- toYaml . | nindent 8 }}
48 | {{- end }}
49 | {{- with .Values.affinity }}
50 | affinity:
51 | {{- toYaml . | nindent 8 }}
52 | {{- end }}
53 | {{- with .Values.tolerations }}
54 | tolerations:
55 | {{- toYaml . | nindent 8 }}
56 | {{- end }}
57 |
--------------------------------------------------------------------------------
/sql/completer/__init__.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: completion_engines.py
6 | @time: 2019/03/09
7 | """
8 |
9 | __author__ = 'hhyo'
10 |
11 |
12 | class Completer:
13 | @property
14 | def name(self):
15 | """返回engine名称"""
16 | return 'Completer engine'
17 |
18 | @property
19 | def info(self):
20 | """返回引擎简介"""
21 | return 'Base Completer engine'
22 |
23 | def _get_sql_execute(self):
24 | """
25 | 初始化数据库执行连接
26 | :return:
27 | """
28 |
29 | def refresh_completions(self, reset=False):
30 | """
31 | 刷新completer对象元数据
32 | :param reset:
33 | :return:
34 | """
35 |
36 | def _on_completions_refreshed(self, new_completer):
37 | """
38 | 刷新completer对象回调函数,替换对象
39 | :param new_completer:
40 | :return:
41 | """
42 |
43 | def get_completions(self, text, cursor_position):
44 | """
45 | 获取补全提示
46 | :param text:
47 | :param cursor_position:
48 | :return:
49 | """
50 |
51 | @staticmethod
52 | def convert2ace_js(completions):
53 | """
54 | 转换completions为ace.js所需要的补全列表格式[{"caption":,"meta":,"name":,"value":,"score":]
55 | caption :字幕,也就是展示在列表中的内容
56 | meta :展示类型
57 | name :名称
58 | value :值
59 | score :分数,越大的排在越上面
60 | :return:
61 | """
62 | ace_completions = []
63 | for completion in completions:
64 | comp = dict()
65 | comp["caption"] = completion.display
66 | comp["meta"] = completion.display_meta
67 | comp["name"] = ''
68 | comp["value"] = ''
69 | comp["score"] = completion.start_position
70 | ace_completions.append(comp)
71 | return ace_completions
72 |
73 |
74 | def get_comp_engine(instance=None, db_name=None):
75 | """获取SQL补全engine"""
76 | if instance.db_type == 'mysql':
77 | from .mysql import MysqlComEngine
78 | return MysqlComEngine(instance=instance, db_name=db_name)
79 |
--------------------------------------------------------------------------------
/common/static/ace/mode-sql.js:
--------------------------------------------------------------------------------
1 | define("ace/mode/sql_highlight_rules",["require","exports","module","ace/lib/oop","ace/mode/text_highlight_rules"],function(e,t,n){"use strict";var r=e("../lib/oop"),i=e("./text_highlight_rules").TextHighlightRules,s=function(){var e="select|insert|update|delete|from|where|and|or|group|by|order|limit|offset|having|as|case|when|then|else|end|type|left|right|join|on|outer|desc|asc|union|create|table|primary|key|if|foreign|not|references|default|null|inner|cross|natural|database|drop|grant",t="true|false",n="avg|count|first|last|max|min|sum|ucase|lcase|mid|len|round|rank|now|format|coalesce|ifnull|isnull|nvl",r="int|numeric|decimal|date|varchar|char|bigint|float|double|bit|binary|text|set|timestamp|money|real|number|integer",i=this.createKeywordMapper({"support.function":n,keyword:e,"constant.language":t,"storage.type":r},"identifier",!0);this.$rules={start:[{token:"comment",regex:"--.*$"},{token:"comment",start:"/\\*",end:"\\*/"},{token:"string",regex:'".*?"'},{token:"string",regex:"'.*?'"},{token:"string",regex:"`.*?`"},{token:"constant.numeric",regex:"[+-]?\\d+(?:(?:\\.\\d*)?(?:[eE][+-]?\\d+)?)?\\b"},{token:i,regex:"[a-zA-Z_$][a-zA-Z0-9_$]*\\b"},{token:"keyword.operator",regex:"\\+|\\-|\\/|\\/\\/|%|<@>|@>|<@|&|\\^|~|<|>|<=|=>|==|!=|<>|="},{token:"paren.lparen",regex:"[\\(]"},{token:"paren.rparen",regex:"[\\)]"},{token:"text",regex:"\\s+"}]},this.normalizeRules()};r.inherits(s,i),t.SqlHighlightRules=s}),define("ace/mode/sql",["require","exports","module","ace/lib/oop","ace/mode/text","ace/mode/sql_highlight_rules"],function(e,t,n){"use strict";var r=e("../lib/oop"),i=e("./text").Mode,s=e("./sql_highlight_rules").SqlHighlightRules,o=function(){this.HighlightRules=s,this.$behaviour=this.$defaultBehaviour};r.inherits(o,i),function(){this.lineCommentStart="--",this.$id="ace/mode/sql"}.call(o.prototype),t.Mode=o}); (function() {
2 | window.require(["ace/mode/sql"], function(m) {
3 | if (typeof module == "object" && typeof exports == "object" && module) {
4 | module.exports = m;
5 | }
6 | });
7 | })();
8 |
--------------------------------------------------------------------------------
/common/static/metisMenu/css/metisMenu.min.css:
--------------------------------------------------------------------------------
1 | /*!
2 | * metismenu - v2.7.8
3 | * A jQuery menu plugin
4 | * https://github.com/onokumus/metismenu#readme
5 | *
6 | * Made by Osman Nuri Okumus (https://github.com/onokumus)
7 | * Under MIT License
8 | */.metismenu .arrow{float:right;line-height:1.42857}[dir=rtl] .metismenu .arrow{float:left}.metismenu .glyphicon.arrow:before{content:"\e079"}.metismenu .active>a>.glyphicon.arrow:before{content:"\e114"}.metismenu .fa.arrow:before{content:"\f104"}.metismenu .active>a>.fa.arrow:before{content:"\f107"}.metismenu .ion.arrow:before{content:"\f3d2"}.metismenu .active>a>.ion.arrow:before{content:"\f3d0"}.metismenu .plus-times{float:right}[dir=rtl] .metismenu .plus-times{float:left}.metismenu .fa.plus-times:before{content:"\f067"}.metismenu .active>a>.fa.plus-times{-webkit-transform:rotate(45deg);transform:rotate(45deg)}.metismenu .plus-minus{float:right}[dir=rtl] .metismenu .plus-minus{float:left}.metismenu .fa.plus-minus:before{content:"\f067"}.metismenu .active>a>.fa.plus-minus:before{content:"\f068"}.metismenu .collapse{display:none}.metismenu .collapse.in{display:block}.metismenu .collapsing{height:0;overflow:hidden;position:relative;transition-duration:.35s;transition-property:height,visibility;transition-timing-function:ease}.metismenu .has-arrow{position:relative}.metismenu .has-arrow:after{-webkit-transform:rotate(-45deg) translateY(-50%);-webkit-transform-origin:top;border-color:initial;border-style:solid;border-width:1px 0 0 1px;content:"";height:.5em;position:absolute;right:1em;top:50%;transform:rotate(-45deg) translateY(-50%);transform-origin:top;transition:all .3s ease-out;width:.5em}[dir=rtl] .metismenu .has-arrow:after{-webkit-transform:rotate(135deg) translateY(-50%);left:1em;right:auto;transform:rotate(135deg) translateY(-50%)}.metismenu .active>.has-arrow:after,.metismenu .has-arrow[aria-expanded=true]:after{-webkit-transform:rotate(-135deg) translateY(-50%);transform:rotate(-135deg) translateY(-50%)}[dir=rtl] .metismenu .active>.has-arrow:after,[dir=rtl] .metismenu .has-arrow[aria-expanded=true]:after{-webkit-transform:rotate(225deg) translateY(-50%);transform:rotate(225deg) translateY(-50%)}
9 | /*# sourceMappingURL=metisMenu.min.css.map */
10 |
--------------------------------------------------------------------------------
/sql/plugins/plugin.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: plugin.py
6 | @time: 2019/03/04
7 | """
8 | __author__ = 'hhyo'
9 |
10 | import logging
11 | import subprocess
12 | import traceback
13 |
14 | logger = logging.getLogger('default')
15 |
16 |
17 | class Plugin:
18 | def __init__(self, path):
19 | self.path = path
20 | self.required_args = [] # 必须参数
21 | self.disable_args = [] # 禁用参数
22 |
23 | def check_args(self, args):
24 | """
25 | 检查请求参数列表
26 | :return: {'status': 0, 'msg': 'ok', 'data': {}}
27 | """
28 | args_check_result = {'status': 0, 'msg': 'ok', 'data': {}}
29 | # 检查路径
30 | if self.path is None:
31 | return {'status': 1, 'msg': '可执行文件路径不能为空!', 'data': {}}
32 | # 检查禁用参数
33 | for arg in args.keys():
34 | if arg in self.disable_args:
35 | return {'status': 1, 'msg': '{arg}参数已被禁用'.format(arg=arg), 'data': {}}
36 | # 检查必须参数
37 | for req_arg in self.required_args:
38 | if req_arg not in args.keys():
39 | return {'status': 1, 'msg': '必须指定{arg}参数'.format(arg=req_arg), 'data': {}}
40 | elif args[req_arg] is None or args[req_arg] == '':
41 | return {'status': 1, 'msg': '{arg}参数值不能为空'.format(arg=req_arg), 'data': {}}
42 | return args_check_result
43 |
44 | def generate_args2cmd(self, args, shell):
45 | """
46 | 将请求参数转换为命令行参数
47 | :return:
48 | """
49 |
50 | @staticmethod
51 | def execute_cmd(cmd_args, shell):
52 | """
53 | 执行命令并且返回process
54 | :return:
55 | """
56 | try:
57 | p = subprocess.Popen(cmd_args,
58 | shell=shell,
59 | stdout=subprocess.PIPE,
60 | stderr=subprocess.PIPE,
61 | universal_newlines=True)
62 | return p
63 | except Exception as e:
64 | logger.error("命令执行失败\n{}".format(traceback.format_exc()))
65 | raise RuntimeError('命令执行失败,失败原因:%s' % str(e))
66 |
--------------------------------------------------------------------------------
/common/static/bootstrap-table/export-libs/FileSaver/FileSaver.min.js:
--------------------------------------------------------------------------------
1 | /*
2 | @source http://purl.eligrey.com/github/FileSaver.js/blob/master/src/FileSaver.js
3 | */
4 | var module$FileSaver={},saveAs$$module$FileSaver=saveAs$$module$FileSaver||function(b){if(!("undefined"===typeof b||"undefined"!==typeof navigator&&/MSIE [1-9]\./.test(navigator.userAgent))){var f=b.document.createElementNS("http://www.w3.org/1999/xhtml","a"),q="download"in f,r=/constructor/i.test(b.HTMLElement)||b.safari,h=/CriOS\/[\d]+/.test(navigator.userAgent),k=b.setImmediate||b.setTimeout,t=function(a){k(function(){throw a;},0)},l=function(a){setTimeout(function(){"string"===typeof a?(b.URL||
5 | b.webkitURL||b).revokeObjectURL(a):a.remove()},4E4)},m=function(a){return/^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(a.type)?new Blob([String.fromCharCode(65279),a],{type:a.type}):a},p=function(a,c,u){u||(a=m(a));var d=this,n="application/octet-stream"===a.type,g=function(){var a=["writestart","progress","write","writeend"];a=[].concat(a);for(var b=a.length;b--;){var c=d["on"+a[b]];if("function"===typeof c)try{c.call(d,d)}catch(v){t(v)}}};d.readyState=d.INIT;
6 | if(q){var e=(b.URL||b.webkitURL||b).createObjectURL(a);k(function(){f.href=e;f.download=c;var a=new MouseEvent("click");f.dispatchEvent(a);g();l(e);d.readyState=d.DONE},0)}else(function(){if((h||n&&r)&&b.FileReader){var c=new FileReader;c.onloadend=function(){var a=h?c.result:c.result.replace(/^data:[^;]*;/,"data:attachment/file;");b.open(a,"_blank")||(b.location.href=a);d.readyState=d.DONE;g()};c.readAsDataURL(a);d.readyState=d.INIT}else e||(e=(b.URL||b.webkitURL||b).createObjectURL(a)),n?b.location.href=
7 | e:b.open(e,"_blank")||(b.location.href=e),d.readyState=d.DONE,g(),l(e)})()},c=p.prototype;if("undefined"!==typeof navigator&&navigator.msSaveOrOpenBlob)return function(a,b,c){b=b||a.name||"download";c||(a=m(a));return navigator.msSaveOrOpenBlob(a,b)};c.abort=function(){};c.readyState=c.INIT=0;c.WRITING=1;c.DONE=2;c.error=c.onwritestart=c.onprogress=c.onwrite=c.onabort=c.onerror=c.onwriteend=null;return function(a,b,c){return new p(a,b||a.name||"download",c)}}}("undefined"!==typeof self&&self||"undefined"!==
8 | typeof window&&window||this);module$FileSaver.saveAs=saveAs$$module$FileSaver;
9 |
--------------------------------------------------------------------------------
/common/utils/const.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 |
3 | class Const(object):
4 | # 定时任务id的前缀
5 | workflowJobprefix = {
6 | 'query': 'query',
7 | 'sqlreview': 'sqlreview',
8 | 'archive': 'archive'
9 | }
10 |
11 |
12 | class WorkflowDict:
13 | # 工作流申请类型,1.query,2.SQL上线申请
14 | workflow_type = {
15 | 'query': 1,
16 | 'query_display': '查询权限申请',
17 | 'sqlreview': 2,
18 | 'sqlreview_display': 'SQL上线申请',
19 | 'archive': 3,
20 | 'archive_display': '数据归档申请',
21 | }
22 |
23 | # 工作流状态,0.待审核 1.审核通过 2.审核不通过 3.审核取消
24 | workflow_status = {
25 | 'audit_wait': 0,
26 | 'audit_wait_display': '待审核',
27 | 'audit_success': 1,
28 | 'audit_success_display': '审核通过',
29 | 'audit_reject': 2,
30 | 'audit_reject_display': '审核不通过',
31 | 'audit_abort': 3,
32 | 'audit_abort_display': '审核取消',
33 | }
34 |
35 |
36 | class SQLTuning:
37 | SYS_PARM_FILTER = [
38 | 'BINLOG_CACHE_SIZE',
39 | 'BULK_INSERT_BUFFER_SIZE',
40 | 'HAVE_PARTITION_ENGINE',
41 | 'HAVE_QUERY_CACHE',
42 | 'INTERACTIVE_TIMEOUT',
43 | 'JOIN_BUFFER_SIZE',
44 | 'KEY_BUFFER_SIZE',
45 | 'KEY_CACHE_AGE_THRESHOLD',
46 | 'KEY_CACHE_BLOCK_SIZE',
47 | 'KEY_CACHE_DIVISION_LIMIT',
48 | 'LARGE_PAGES',
49 | 'LOCKED_IN_MEMORY',
50 | 'LONG_QUERY_TIME',
51 | 'MAX_ALLOWED_PACKET',
52 | 'MAX_BINLOG_CACHE_SIZE',
53 | 'MAX_BINLOG_SIZE',
54 | 'MAX_CONNECT_ERRORS',
55 | 'MAX_CONNECTIONS',
56 | 'MAX_JOIN_SIZE',
57 | 'MAX_LENGTH_FOR_SORT_DATA',
58 | 'MAX_SEEKS_FOR_KEY',
59 | 'MAX_SORT_LENGTH',
60 | 'MAX_TMP_TABLES',
61 | 'MAX_USER_CONNECTIONS',
62 | 'OPTIMIZER_PRUNE_LEVEL',
63 | 'OPTIMIZER_SEARCH_DEPTH',
64 | 'QUERY_CACHE_SIZE',
65 | 'QUERY_CACHE_TYPE',
66 | 'QUERY_PREALLOC_SIZE',
67 | 'RANGE_ALLOC_BLOCK_SIZE',
68 | 'READ_BUFFER_SIZE',
69 | 'READ_RND_BUFFER_SIZE',
70 | 'SORT_BUFFER_SIZE',
71 | 'SQL_MODE',
72 | 'TABLE_CACHE',
73 | 'THREAD_CACHE_SIZE',
74 | 'TMP_TABLE_SIZE',
75 | 'WAIT_TIMEOUT'
76 | ]
77 |
--------------------------------------------------------------------------------
/src/init_sql/v1.6.1_v1.6.2.sql:
--------------------------------------------------------------------------------
1 | -- 资源组和用户关联表
2 | CREATE TABLE `resource_group_user` (
3 | `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键',
4 | `resource_group_id` int(11) NOT NULL COMMENT '资源组',
5 | `user_id` int(11) NOT NULL COMMENT '用户',
6 | `create_time` datetime(6) NOT NULL COMMENT '创建时间',
7 | PRIMARY KEY (`id`),
8 | KEY `idx_user_id` (`user_id`),
9 | UNIQUE uniq_resource_group_id_instance_id(`resource_group_id`,`user_id`),
10 | CONSTRAINT `fk_resource_group_user_resource_group` FOREIGN KEY (`resource_group_id`) REFERENCES `resource_group` (`group_id`),
11 | CONSTRAINT `fk_resource_group_user` FOREIGN KEY (`user_id`) REFERENCES `sql_users` (`id`)
12 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
13 |
14 | -- 资源组和实例关联表
15 | CREATE TABLE `resource_group_instance` (
16 | `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键',
17 | `resource_group_id` int(11) NOT NULL COMMENT '资源组',
18 | `instance_id` int(11) NOT NULL COMMENT '实例',
19 | `create_time` datetime(6) NOT NULL COMMENT '创建时间',
20 | PRIMARY KEY (`id`),
21 | KEY `idx_instance_id` (`instance_id`),
22 | UNIQUE uniq_resource_group_id_instance_id(`resource_group_id`,`instance_id`),
23 | CONSTRAINT `fk_resource_group_instance_resource_group` FOREIGN KEY (`resource_group_id`) REFERENCES `resource_group` (`group_id`),
24 | CONSTRAINT `fk_resource_group_instance` FOREIGN KEY (`instance_id`) REFERENCES `sql_instance` (`id`)
25 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
26 |
27 | -- 数据清洗
28 | set foreign_key_checks = 0;
29 | -- 用户关系数据
30 | insert into resource_group_user (resource_group_id, user_id, create_time)
31 | select group_id,object_id,create_time from resource_group_relations where object_type=0;
32 | -- 实例关系数据
33 | insert into resource_group_instance (resource_group_id, instance_id, create_time)
34 | select group_id,object_id,create_time from resource_group_relations where object_type=1;
35 | set foreign_key_checks = 1;
36 |
37 | -- 删除旧表
38 | drop table resource_group_relations;
39 |
40 | -- SQL上线工单增加可执行时间选择
41 | ALTER TABLE sql_workflow
42 | ADD run_date_start datetime(6) DEFAULT NULL COMMENT '可执行起始时间',
43 | ADD run_date_end datetime(6) DEFAULT NULL COMMENT '可执行结束时间';
44 |
45 | -- 实例配置增加默认字符集信息
46 | ALTER TABLE sql_instance
47 | ADD `charset` varchar(20) DEFAULT NULL COMMENT '字符集' after `password`;
48 |
--------------------------------------------------------------------------------
/sql/completer/tests.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: tests.py
6 | @time: 2019/03/11
7 | """
8 | from django.conf import settings
9 | from django.test import TestCase
10 | from prompt_toolkit.completion import Completion
11 | from sql.completer import get_comp_engine
12 |
13 | from sql.models import Instance
14 |
15 | __author__ = 'hhyo'
16 |
17 |
18 | class TestCompleter(TestCase):
19 | @classmethod
20 | def setUpClass(cls):
21 | """
22 | 初始化补全引擎
23 | :return:
24 | """
25 | # 使用 travis.ci 时实例和测试service保持一致
26 | cls.master = Instance(instance_name='test_instance', type='master', db_type='mysql',
27 | host=settings.DATABASES['default']['HOST'],
28 | port=settings.DATABASES['default']['PORT'],
29 | user=settings.DATABASES['default']['USER'],
30 | password=settings.DATABASES['default']['PASSWORD'])
31 | cls.master.save()
32 | cls.comp_engine = get_comp_engine(instance=cls.master, db_name=settings.DATABASES['default']['TEST']['NAME'])
33 | # 等待completion_refresher刷新完成
34 | while cls.comp_engine.completion_refresher.is_refreshing():
35 | import time
36 | time.sleep(1)
37 |
38 | @classmethod
39 | def tearDownClass(cls):
40 | """
41 | :return:
42 | """
43 | cls.master.delete()
44 | cls.comp_engine.refresh_completions(reset=True)
45 |
46 | def test_table_names_after_from(self):
47 | text = 'SELECT * FROM '
48 | position = len('SELECT * FROM ')
49 | self.comp_engine.get_completions(text=text, cursor_position=position)
50 |
51 | def test_suggested_column_names(self):
52 | text = 'SELECT FROM sql_users'
53 | position = len('SELECT ')
54 | self.comp_engine.get_completions(text=text, cursor_position=position)
55 |
56 | def test_function_name_completion(self):
57 | text = 'SELECT MA'
58 | position = len('SELECT MA')
59 | result = self.comp_engine.get_completions(text=text, cursor_position=position)
60 | self.comp_engine.convert2ace_js(result)
61 | self.assertListEqual(result, [Completion(text='MAX', start_position=-2),
62 | Completion(text='MASTER', start_position=-2)])
63 |
--------------------------------------------------------------------------------
/sql/plugins/binglog2sql.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: binglog2sql.py
6 | @time: 2019/03/23
7 | """
8 | from common.config import SysConfig
9 | from sql.plugins.plugin import Plugin
10 |
11 | __author__ = 'hhyo'
12 |
13 |
14 | class Binlog2Sql(Plugin):
15 |
16 | def __init__(self):
17 | self.path = SysConfig().get('binlog2sql')
18 | self.required_args = []
19 | self.disable_args = []
20 | super(Plugin, self).__init__()
21 |
22 | def generate_args2cmd(self, args, shell):
23 | """
24 | 转换请求参数为命令行
25 | :param args:
26 | :param shell:
27 | :return:
28 | """
29 | conn_options = ['conn_options']
30 | parse_mode_options = ['stop-never', 'no-primary-key', 'flashback']
31 | range_options = ['back-interval', 'start-file', 'start-position', 'stop-file', 'stop-position',
32 | 'start-datetime', 'stop-datetime']
33 | filter_options = ['databases', 'tables', 'only-dml', 'sql-type']
34 | if shell:
35 | cmd_args = f'python {self.path}' if self.path else ''
36 | for name, value in args.items():
37 | if name in conn_options:
38 | cmd_args += f' {value}'
39 | elif name in parse_mode_options and value:
40 | cmd_args += f' --{name}'
41 | elif name in range_options and value:
42 | cmd_args += f" --{name}='{value}'"
43 | elif name in filter_options and value:
44 | if name == 'only-dml':
45 | cmd_args += f' --{name}'
46 | else:
47 | cmd_args += f' --{name} {value}'
48 | else:
49 | cmd_args = [self.path]
50 | for name, value in args.items():
51 | if name in conn_options:
52 | cmd_args.append(f'{value}')
53 | elif name in parse_mode_options:
54 | cmd_args.append(f'--{name}')
55 | elif name in range_options:
56 | cmd_args.append(f'--{name}')
57 | cmd_args.append(f'{value}')
58 | elif name in filter_options:
59 | cmd_args.append(f'--{name}')
60 | cmd_args.append(f'{value}')
61 | return cmd_args
62 |
--------------------------------------------------------------------------------
/src/charts/charts/inception/values.yaml:
--------------------------------------------------------------------------------
1 | # Default values for inception.
2 | # This is a YAML-formatted file.
3 | # Declare variables to be passed into your templates.
4 |
5 | replicaCount: 1
6 |
7 | image:
8 | repository: hhyo/inception
9 | tag: latest
10 | pullPolicy: IfNotPresent
11 |
12 | nameOverride: ""
13 | fullnameOverride: ""
14 |
15 | service:
16 | type: ClusterIP
17 | port: 6669
18 | targetPort: 6669
19 |
20 | ingress:
21 | enabled: false
22 | annotations: {}
23 | # kubernetes.io/ingress.class: nginx
24 | # kubernetes.io/tls-acme: "true"
25 | paths: []
26 | hosts:
27 | - chart-example.local
28 | tls: []
29 | # - secretName: chart-example-tls
30 | # hosts:
31 | # - chart-example.local
32 |
33 | configMap:
34 | enabled: true
35 | data:
36 | inc.cnf: |-
37 | [inception]
38 | general_log=1
39 | general_log_file=inception.log
40 | port=6669
41 | socket=/tmp/inc.socket
42 | character-set-client-handshake=0
43 | character-set-server=utf8mb4
44 | inception_language_code=zh-CN
45 | inception_remote_system_password=archery
46 | inception_remote_system_user=root
47 | inception_remote_backup_port=3306
48 | inception_remote_backup_host=archery-mysql
49 | inception_support_charset=utf8,utf8mb4
50 | inception_enable_nullable=0
51 | inception_check_primary_key=1
52 | inception_check_column_comment=1
53 | inception_check_table_comment=1
54 | inception_osc_on=OFF
55 | inception_osc_bin_dir=/usr/bin
56 | inception_osc_min_table_size=10
57 | inception_osc_chunk_time=0.1
58 | inception_enable_blob_type=1
59 | inception_check_column_default_value=1
60 |
61 | resources: {}
62 | # We usually recommend not to specify default resources and to leave this as a conscious
63 | # choice for the user. This also increases chances charts run on environments with little
64 | # resources, such as Minikube. If you do want to specify resources, uncomment the following
65 | # lines, adjust them as necessary, and remove the curly braces after 'resources:'.
66 | # limits:
67 | # cpu: 100m
68 | # memory: 128Mi
69 | # requests:
70 | # cpu: 100m
71 | # memory: 128Mi
72 |
73 | nodeSelector: {}
74 |
75 | tolerations: []
76 |
77 | affinity: {}
78 |
79 | volumes:
80 | - name: inception-config-volume
81 | configMap:
82 | name: inception-config
83 |
--------------------------------------------------------------------------------
/common/static/ace/snippets/sqlserver.js:
--------------------------------------------------------------------------------
1 | define("ace/snippets/sqlserver",["require","exports","module"],function(e,t,n){"use strict";t.snippetText="# ISNULL\nsnippet isnull\n ISNULL(${1:check_expression}, ${2:replacement_value})\n# FORMAT\nsnippet format\n FORMAT(${1:value}, ${2:format})\n# CAST\nsnippet cast\n CAST(${1:expression} AS ${2:data_type})\n# CONVERT\nsnippet convert\n CONVERT(${1:data_type}, ${2:expression})\n# DATEPART\nsnippet datepart\n DATEPART(${1:datepart}, ${2:date})\n# DATEDIFF\nsnippet datediff\n DATEDIFF(${1:datepart}, ${2:startdate}, ${3:enddate})\n# DATEADD\nsnippet dateadd\n DATEADD(${1:datepart}, ${2:number}, ${3:date})\n# DATEFROMPARTS \nsnippet datefromparts\n DATEFROMPARTS(${1:year}, ${2:month}, ${3:day})\n# OBJECT_DEFINITION\nsnippet objectdef\n SELECT OBJECT_DEFINITION(OBJECT_ID('${1:sys.server_permissions /*object name*/}'))\n# STUFF XML\nsnippet stuffxml\n STUFF((SELECT ', ' + ${1:ColumnName}\n FROM ${2:TableName}\n WHERE ${3:WhereClause}\n FOR XML PATH('')), 1, 1, '') AS ${4:Alias}\n ${5:/*https://msdn.microsoft.com/en-us/library/ms188043.aspx*/}\n# Create Procedure\nsnippet createproc\n -- =============================================\n -- Author: ${1:Author}\n -- Create date: ${2:Date}\n -- Description: ${3:Description}\n -- =============================================\n CREATE PROCEDURE ${4:Procedure_Name}\n ${5:/*Add the parameters for the stored procedure here*/}\n AS\n BEGIN\n -- SET NOCOUNT ON added to prevent extra result sets from interfering with SELECT statements.\n SET NOCOUNT ON;\n \n ${6:/*Add the T-SQL statements to compute the return value here*/}\n \n END\n GO\n# Create Scalar Function\nsnippet createfn\n -- =============================================\n -- Author: ${1:Author}\n -- Create date: ${2:Date}\n -- Description: ${3:Description}\n -- =============================================\n CREATE FUNCTION ${4:Scalar_Function_Name}\n -- Add the parameters for the function here\n RETURNS ${5:Function_Data_Type}\n AS\n BEGIN\n DECLARE @Result ${5:Function_Data_Type}\n \n ${6:/*Add the T-SQL statements to compute the return value here*/}\n \n END\n GO",t.scope="sqlserver"}); (function() {
2 | window.require(["ace/snippets/sqlserver"], function(m) {
3 | if (typeof module == "object" && typeof exports == "object" && module) {
4 | module.exports = m;
5 | }
6 | });
7 | })();
8 |
--------------------------------------------------------------------------------
/common/static/ace/theme-github.js:
--------------------------------------------------------------------------------
1 | define("ace/theme/github",["require","exports","module","ace/lib/dom"],function(e,t,n){t.isDark=!1,t.cssClass="ace-github",t.cssText='.ace-github .ace_gutter {background: #e8e8e8;color: #AAA;}.ace-github {background: #fff;color: #000;}.ace-github .ace_keyword {font-weight: bold;}.ace-github .ace_string {color: #D14;}.ace-github .ace_variable.ace_class {color: teal;}.ace-github .ace_constant.ace_numeric {color: #099;}.ace-github .ace_constant.ace_buildin {color: #0086B3;}.ace-github .ace_support.ace_function {color: #0086B3;}.ace-github .ace_comment {color: #998;font-style: italic;}.ace-github .ace_variable.ace_language {color: #0086B3;}.ace-github .ace_paren {font-weight: bold;}.ace-github .ace_boolean {font-weight: bold;}.ace-github .ace_string.ace_regexp {color: #009926;font-weight: normal;}.ace-github .ace_variable.ace_instance {color: teal;}.ace-github .ace_constant.ace_language {font-weight: bold;}.ace-github .ace_cursor {color: black;}.ace-github.ace_focus .ace_marker-layer .ace_active-line {background: rgb(255, 255, 204);}.ace-github .ace_marker-layer .ace_active-line {background: rgb(245, 245, 245);}.ace-github .ace_marker-layer .ace_selection {background: rgb(181, 213, 255);}.ace-github.ace_multiselect .ace_selection.ace_start {box-shadow: 0 0 3px 0px white;}.ace-github.ace_nobold .ace_line > span {font-weight: normal !important;}.ace-github .ace_marker-layer .ace_step {background: rgb(252, 255, 0);}.ace-github .ace_marker-layer .ace_stack {background: rgb(164, 229, 101);}.ace-github .ace_marker-layer .ace_bracket {margin: -1px 0 0 -1px;border: 1px solid rgb(192, 192, 192);}.ace-github .ace_gutter-active-line {background-color : rgba(0, 0, 0, 0.07);}.ace-github .ace_marker-layer .ace_selected-word {background: rgb(250, 250, 255);border: 1px solid rgb(200, 200, 250);}.ace-github .ace_invisible {color: #BFBFBF}.ace-github .ace_print-margin {width: 1px;background: #e8e8e8;}.ace-github .ace_indent-guide {background: url("") right repeat-y;}';var r=e("../lib/dom");r.importCssString(t.cssText,t.cssClass)}); (function() {
2 | window.require(["ace/theme/github"], function(m) {
3 | if (typeof module == "object" && typeof exports == "object" && module) {
4 | module.exports = m;
5 | }
6 | });
7 | })();
8 |
--------------------------------------------------------------------------------
/src/init_sql/v1.6.7_v1.7.0.sql:
--------------------------------------------------------------------------------
1 | -- 删除Themis权限
2 | set @perm_id=(select id from auth_permission where codename='menu_themis');
3 | delete from auth_group_permissions where permission_id=@perm_id;
4 | delete from sql_users_user_permissions where permission_id=@perm_id;
5 | delete from auth_permission where codename='menu_themis';
6 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
7 | -- 增加实例账号管理权限,变更菜单权限信息
8 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 管理实例账号', @content_type_id, 'instance_account_manage');
9 | UPDATE auth_permission set name='菜单 实例账号管理',codename='menu_instance_account' where codename='menu_instance_user';
10 | -- 增加实例数据库权限
11 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 数据库管理', @content_type_id, 'menu_database');
12 | -- 增加资源组粒度的查询权限
13 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('可查询所在资源组内的所有实例', @content_type_id, 'query_resource_group_instance');
14 | -- 增加工具插件的权限
15 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 工具插件', @content_type_id, 'menu_menu_tools');
16 |
17 | -- 添加钉钉user id
18 | alter table sql_users
19 | add ding_user_id varchar(64) default null comment '钉钉user_id';
20 |
21 |
22 | -- 增加实例账号表
23 | CREATE TABLE `instance_account` (
24 | `id` int(11) NOT NULL AUTO_INCREMENT,
25 | `user` varchar(128) NOT NULL COMMENT '账号',
26 | `host` varchar(64) NOT NULL COMMENT '主机',
27 | `password` varchar(128) NOT NULL COMMENT '密码',
28 | `remark` varchar(255) NOT NULL COMMENT '备注',
29 | `sys_time` datetime(6) NOT NULL COMMENT '系统时间',
30 | `instance_id` int(11) NOT NULL COMMENT '实例',
31 | PRIMARY KEY (`id`),
32 | UNIQUE KEY `uniq_instance_id_user_host` (`instance_id`,`user`,`host`),
33 | CONSTRAINT `fk_account_sql_instance_id` FOREIGN KEY (`instance_id`) REFERENCES `sql_instance` (`id`)
34 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
35 |
36 |
37 | -- 增加实例数据库表
38 | CREATE TABLE `instance_database` (
39 | `id` int(11) NOT NULL AUTO_INCREMENT,
40 | `db_name` varchar(128) NOT NULL COMMENT '账号',
41 | `owner` varchar(30) NOT NULL COMMENT '负责人',
42 | `owner_display` varchar(50) NOT NULL DEFAULT '负责人中文名',
43 | `remark` varchar(255) NOT NULL COMMENT '备注',
44 | `sys_time` datetime(6) NOT NULL COMMENT '系统时间',
45 | `instance_id` int(11) NOT NULL COMMENT '实例',
46 | PRIMARY KEY (`id`),
47 | CONSTRAINT `fk_database_sql_instance_id` FOREIGN KEY (`instance_id`) REFERENCES `sql_instance` (`id`)
48 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
49 |
50 |
51 |
--------------------------------------------------------------------------------
/sql/completer/mysql.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: mysql.py
6 | @time: 2019/03/09
7 | """
8 | import threading
9 |
10 | import mycli.sqlcompleter as completer
11 | from prompt_toolkit.document import Document
12 | from mycli.completion_refresher import CompletionRefresher
13 | from mycli.sqlexecute import SQLExecute
14 | from . import Completer
15 |
16 | __author__ = 'hhyo'
17 |
18 |
19 | class MysqlComEngine(Completer):
20 | def __init__(self, instance=None, db_name=None):
21 | self.instance = instance
22 | self.db_name = db_name
23 | self.completion_refresher = CompletionRefresher()
24 | self.completer = completer.SQLCompleter(smart_completion=True)
25 | self._completer_lock = threading.Lock()
26 | self.sql_execute = self._get_sql_execute()
27 | self.refresh_completions()
28 |
29 | def _get_sql_execute(self):
30 | """
31 | 初始化数据库执行连接
32 | :return:
33 | """
34 | return SQLExecute(self.db_name, self.instance.user, self.instance.password, self.instance.host,
35 | int(self.instance.port),
36 | socket='', charset=self.instance.charset or 'utf8mb4',
37 | local_infile='', ssl='', ssh_user='', ssh_host='', ssh_port='',
38 | ssh_password='', ssh_key_filename=''
39 | )
40 |
41 | def refresh_completions(self, reset=False):
42 | """
43 | 刷新completer对象元数据
44 | :param reset:
45 | :return:
46 | """
47 | if reset:
48 | with self._completer_lock:
49 | self.completer.reset_completions()
50 | self.completion_refresher.refresh(self.sql_execute, self._on_completions_refreshed)
51 | return [(None, None, None, 'Auto-completion refresh started in the background.')]
52 |
53 | def _on_completions_refreshed(self, new_completer):
54 | """
55 | 刷新completer对象回调函数,替换对象
56 | :param new_completer:
57 | :return:
58 | """
59 | with self._completer_lock:
60 | self.completer = new_completer
61 |
62 | def get_completions(self, text, cursor_position):
63 | """
64 | 获取补全提示
65 | :param text:
66 | :param cursor_position:
67 | :return:
68 | """
69 | with self._completer_lock:
70 | return self.completer.get_completions(
71 | Document(text=text, cursor_position=cursor_position), None)
72 |
--------------------------------------------------------------------------------
/sql/static/css/basic.css:
--------------------------------------------------------------------------------
1 | /*
2 | * basic.css
3 | * ~~~~~~~~~
4 | *
5 | * Sphinx stylesheet -- basic theme.
6 | *
7 | * :copyright: Copyright 2007-2017 by the Sphinx team, see AUTHORS.
8 | * :license: BSD, see LICENSE for details.
9 | *
10 | */
11 |
12 | /* -- sidebar --------------------------------------------------------------- */
13 |
14 | div.sphinxsidebarwrapper {
15 | padding: 10px 5px 0 10px;
16 | }
17 |
18 | div.sphinxsidebar {
19 | float: left;
20 | width: 230px;
21 | margin-left: -100%;
22 | font-size: 90%;
23 | word-wrap: break-word;
24 | overflow-wrap : break-word;
25 | }
26 |
27 | div.sphinxsidebar ul {
28 | list-style: none;
29 | }
30 |
31 | div.sphinxsidebar ul ul,
32 | div.sphinxsidebar ul.want-points {
33 | margin-left: 20px;
34 | list-style: square;
35 | }
36 |
37 | div.sphinxsidebar ul ul {
38 | margin-top: 0;
39 | margin-bottom: 0;
40 | }
41 |
42 | div.sphinxsidebar form {
43 | margin-top: 10px;
44 | }
45 |
46 | div.sphinxsidebar input {
47 | border: 1px solid #98dbcc;
48 | font-family: sans-serif;
49 | font-size: 1em;
50 | }
51 |
52 | div.sphinxsidebar #searchbox input[type="text"] {
53 | width: 170px;
54 | }
55 |
56 | /* -- general index --------------------------------------------------------- */
57 |
58 | table.indextable {
59 | width: 100%;
60 | }
61 |
62 | table.indextable td {
63 | text-align: left;
64 | vertical-align: top;
65 | }
66 |
67 | table.indextable ul {
68 | margin-top: 0;
69 | margin-bottom: 0;
70 | list-style-type: none;
71 | }
72 |
73 | table.indextable > tbody > tr > td > ul {
74 | padding-left: 0em;
75 | }
76 |
77 | table.indextable tr.pcap {
78 | height: 10px;
79 | }
80 |
81 | table.indextable tr.cap {
82 | margin-top: 10px;
83 | background-color: #f2f2f2;
84 | }
85 |
86 | img.toggler {
87 | margin-right: 3px;
88 | margin-top: 3px;
89 | cursor: pointer;
90 | }
91 |
92 | div.modindex-jumpbox {
93 | border-top: 1px solid #ddd;
94 | border-bottom: 1px solid #ddd;
95 | margin: 1em 0 1em 0;
96 | padding: 0.4em;
97 | }
98 |
99 | div.genindex-jumpbox {
100 | border-top: 1px solid #ddd;
101 | border-bottom: 1px solid #ddd;
102 | margin: 1em 0 1em 0;
103 | padding: 0.4em;
104 | }
105 |
106 | /* -- domain module index --------------------------------------------------- */
107 |
108 | table.modindextable td {
109 | padding: 2px;
110 | border-collapse: collapse;
111 | }
112 |
--------------------------------------------------------------------------------
/src/charts/templates/deployment.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: apps/v1beta2
2 | kind: Deployment
3 | metadata:
4 | name: {{ include "archery.fullname" . }}
5 | labels:
6 | app.kubernetes.io/name: {{ include "archery.name" . }}
7 | helm.sh/chart: {{ include "archery.chart" . }}
8 | app.kubernetes.io/instance: {{ .Release.Name }}
9 | app.kubernetes.io/managed-by: {{ .Release.Service }}
10 | spec:
11 | replicas: {{ .Values.replicaCount }}
12 | selector:
13 | matchLabels:
14 | app.kubernetes.io/name: {{ include "archery.name" . }}
15 | app.kubernetes.io/instance: {{ .Release.Name }}
16 | template:
17 | metadata:
18 | labels:
19 | app.kubernetes.io/name: {{ include "archery.name" . }}
20 | app.kubernetes.io/instance: {{ .Release.Name }}
21 | spec:
22 | initContainers:
23 | - image: busybox
24 | name: init-container
25 | imagePullPolicy: IfNotPresent
26 | args:
27 | - "mkdir"
28 | - "-p"
29 | - "/opt/archery/downloads/{binlog2sql,log,schemasync}"
30 | volumeMounts:
31 | - mountPath: /opt/archery/downloads
32 | name: archery-download
33 | - mountPath: /opt/archery/src/script
34 | name: archery-script
35 | - image: busybox
36 | name: init-mysql
37 | imagePullPolicy: IfNotPresent
38 | command: ['sh', '-c', 'until nslookup archery-mysql; do echo waiting for mysql; sleep 2; done;']
39 | containers:
40 | - name: {{ .Chart.Name }}
41 | image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}"
42 | imagePullPolicy: {{ .Values.image.pullPolicy }}
43 | env:
44 | - name: NGINX_PORT
45 | value: "9123"
46 | ports:
47 | - name: archery
48 | containerPort: 9123
49 | protocol: TCP
50 | livenessProbe:
51 | httpGet:
52 | path: /
53 | port: archery
54 | lifecycle:
55 | postStart:
56 | exec:
57 | command: ['/bin/bash','/opt/archery/src/docker/init-archery.sh']
58 | resources:
59 | {{- toYaml .Values.resources | nindent 12 }}
60 | {{- with .Values.volumeMounts }}
61 | volumeMounts:
62 | {{- toYaml . | nindent 12 }}
63 | {{- end }}
64 | {{- with .Values.volumes }}
65 | volumes:
66 | {{- toYaml . | nindent 8 }}
67 | {{- end }}
68 | {{- with .Values.nodeSelector }}
69 | nodeSelector:
70 | {{- toYaml . | nindent 8 }}
71 | {{- end }}
72 | {{- with .Values.affinity }}
73 | affinity:
74 | {{- toYaml . | nindent 8 }}
75 | {{- end }}
76 | {{- with .Values.tolerations }}
77 | tolerations:
78 | {{- toYaml . | nindent 8 }}
79 | {{- end }}
80 |
--------------------------------------------------------------------------------
/common/static/ace/theme-textmate.js:
--------------------------------------------------------------------------------
1 | define("ace/theme/textmate",["require","exports","module","ace/lib/dom"],function(e,t,n){"use strict";t.isDark=!1,t.cssClass="ace-tm",t.cssText='.ace-tm .ace_gutter {background: #f0f0f0;color: #333;}.ace-tm .ace_print-margin {width: 1px;background: #e8e8e8;}.ace-tm .ace_fold {background-color: #6B72E6;}.ace-tm {background-color: #FFFFFF;color: black;}.ace-tm .ace_cursor {color: black;}.ace-tm .ace_invisible {color: rgb(191, 191, 191);}.ace-tm .ace_storage,.ace-tm .ace_keyword {color: blue;}.ace-tm .ace_constant {color: rgb(197, 6, 11);}.ace-tm .ace_constant.ace_buildin {color: rgb(88, 72, 246);}.ace-tm .ace_constant.ace_language {color: rgb(88, 92, 246);}.ace-tm .ace_constant.ace_library {color: rgb(6, 150, 14);}.ace-tm .ace_invalid {background-color: rgba(255, 0, 0, 0.1);color: red;}.ace-tm .ace_support.ace_function {color: rgb(60, 76, 114);}.ace-tm .ace_support.ace_constant {color: rgb(6, 150, 14);}.ace-tm .ace_support.ace_type,.ace-tm .ace_support.ace_class {color: rgb(109, 121, 222);}.ace-tm .ace_keyword.ace_operator {color: rgb(104, 118, 135);}.ace-tm .ace_string {color: rgb(3, 106, 7);}.ace-tm .ace_comment {color: rgb(76, 136, 107);}.ace-tm .ace_comment.ace_doc {color: rgb(0, 102, 255);}.ace-tm .ace_comment.ace_doc.ace_tag {color: rgb(128, 159, 191);}.ace-tm .ace_constant.ace_numeric {color: rgb(0, 0, 205);}.ace-tm .ace_variable {color: rgb(49, 132, 149);}.ace-tm .ace_xml-pe {color: rgb(104, 104, 91);}.ace-tm .ace_entity.ace_name.ace_function {color: #0000A2;}.ace-tm .ace_heading {color: rgb(12, 7, 255);}.ace-tm .ace_list {color:rgb(185, 6, 144);}.ace-tm .ace_meta.ace_tag {color:rgb(0, 22, 142);}.ace-tm .ace_string.ace_regex {color: rgb(255, 0, 0)}.ace-tm .ace_marker-layer .ace_selection {background: rgb(181, 213, 255);}.ace-tm.ace_multiselect .ace_selection.ace_start {box-shadow: 0 0 3px 0px white;}.ace-tm .ace_marker-layer .ace_step {background: rgb(252, 255, 0);}.ace-tm .ace_marker-layer .ace_stack {background: rgb(164, 229, 101);}.ace-tm .ace_marker-layer .ace_bracket {margin: -1px 0 0 -1px;border: 1px solid rgb(192, 192, 192);}.ace-tm .ace_marker-layer .ace_active-line {background: rgba(0, 0, 0, 0.07);}.ace-tm .ace_gutter-active-line {background-color : #dcdcdc;}.ace-tm .ace_marker-layer .ace_selected-word {background: rgb(250, 250, 255);border: 1px solid rgb(200, 200, 250);}.ace-tm .ace_indent-guide {background: url("") right repeat-y;}',t.$id="ace/theme/textmate";var r=e("../lib/dom");r.importCssString(t.cssText,t.cssClass)}); (function() {
2 | window.require(["ace/theme/textmate"], function(m) {
3 | if (typeof module == "object" && typeof exports == "object" && module) {
4 | module.exports = m;
5 | }
6 | });
7 | })();
8 |
--------------------------------------------------------------------------------
/src/init_sql/v1.5.3_v1.6.0.sql:
--------------------------------------------------------------------------------
1 | -- 增加oracle实例相关字段
2 | ALTER TABLE sql_instance
3 | ADD `sid` varchar(50) DEFAULT NULL COMMENT 'Oracle sid' AFTER password,
4 | ADD `service_name` varchar(50) DEFAULT NULL COMMENT 'Oracle Service name' AFTER password;
5 |
6 | -- 变更字段名
7 | ALTER TABLE param_history CHANGE update_time create_time TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '参数修改时间';
8 |
9 | -- 变更字符集为utf8mb4,本身是utf8mb4的无需执行该语句
10 | ALTER TABLE sql_workflow_content
11 | modify `sql_content` longtext CHARACTER SET utf8mb4 NOT NULL COMMENT '提交的SQL文本',
12 | modify `review_content` longtext CHARACTER SET utf8mb4 NOT NULL COMMENT '自动审核内容的JSON格式',
13 | modify `execute_result` longtext CHARACTER SET utf8mb4 NOT NULL COMMENT '执行结果的JSON格式';
14 |
15 | ALTER TABLE query_log
16 | modify `sqllog` longtext CHARACTER SET utf8mb4 NOT NULL COMMENT '执行的sql查询';
17 |
18 | -- 增加实例标签配置
19 | CREATE TABLE `sql_instance_tag` (
20 | `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '标签id',
21 | `tag_code` varchar(20) NOT NULL COMMENT '标签代码',
22 | `tag_name` varchar(20) NOT NULL COMMENT '标签名称',
23 | `active` tinyint(1) NOT NULL COMMENT '激活状态',
24 | `create_time` datetime(6) NOT NULL COMMENT '创建时间',
25 | PRIMARY KEY (`id`),
26 | UNIQUE KEY `uniq_tag_code` (`tag_code`),
27 | UNIQUE KEY `uniq_tag_name` (`tag_name`)
28 | ) COMMENT '实例标签配置' ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
29 |
30 | CREATE TABLE `sql_instance_tag_relations` (
31 | `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '关联id',
32 | `instance_id` int(11) NOT NULL COMMENT '关联实例ID',
33 | `instance_tag_id` int(11) NOT NULL COMMENT '关联标签ID',
34 | `active` tinyint(1) NOT NULL COMMENT '激活状态',
35 | `create_time` datetime(6) NOT NULL COMMENT '创建时间',
36 | PRIMARY KEY (`id`),
37 | KEY `idx_instance_tag_id` (`instance_tag_id`),
38 | UNIQUE KEY `uniq_instance_id_instance_tag_id` (`instance_id`,`instance_tag_id`),
39 | CONSTRAINT `fk_itr_instance` FOREIGN KEY (`instance_id`) REFERENCES `sql_instance` (`id`),
40 | CONSTRAINT `fk_itr_instance_tag` FOREIGN KEY (`instance_tag_id`) REFERENCES `sql_instance_tag` (`id`)
41 | ) COMMENT '实例标签关系' ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
42 |
43 | -- 初始化标签数据
44 | INSERT INTO sql_instance_tag (id, tag_code, tag_name, active, create_time) VALUES (1, 'can_write', '支持上线', 1, '2019-05-03 00:00:00.000000');
45 | INSERT INTO sql_instance_tag (id, tag_code, tag_name, active, create_time) VALUES (2, 'can_read', '支持查询', 1, '2019-05-03 00:00:00.000000');
46 |
47 | -- 给原有主从数据增加标签
48 | insert into sql_instance_tag_relations (instance_id, instance_tag_id, active, create_time)
49 | select id,1,1,now() from sql_instance where type='master';
50 | insert into sql_instance_tag_relations (instance_id, instance_tag_id, active, create_time)
51 | select id,2,1,now() from sql_instance where type='slave';
52 |
53 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
54 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('可查询所有实例', @content_type_id, 'query_all_instances');
55 |
--------------------------------------------------------------------------------
/sql/fixtures/auth_group.sql:
--------------------------------------------------------------------------------
1 | -- sql_instance_tag
2 | INSERT INTO sql_instance_tag (id, tag_code, tag_name, active, create_time)
3 | VALUES (1, 'can_write', '支持上线', 1, now()),(2, 'can_read', '支持查询', 1, now());
4 |
5 | -- auth_group
6 | INSERT INTO auth_group (id, name)
7 | VALUES (1, 'Default'),(2, 'RD'),(3, 'DBA'),(4, 'PM'),(5, 'QA');
8 |
9 | SET FOREIGN_KEY_CHECKS=0;
10 | -- Default
11 | insert into auth_group_permissions (group_id, permission_id)
12 | select 1,id
13 | from auth_permission
14 | where codename in ('menu_sqlworkflow', 'menu_query', 'menu_sqlquery', 'menu_queryapplylist', 'menu_document');
15 |
16 | -- RD
17 | insert into auth_group_permissions (group_id, permission_id)
18 | select 2,id
19 | from auth_permission
20 | where codename in ('menu_dashboard','menu_sqlcheck','menu_sqlworkflow','menu_sqlanalyze','menu_query','menu_sqlquery','menu_queryapplylist','menu_sqloptimize','menu_sqladvisor','menu_slowquery','menu_data_dictionary','menu_tools','menu_archive','menu_document','sql_submit','sql_execute','sql_analyze','optimize_sqladvisor','optimize_soar','query_applypriv','query_submit','archive_apply');
21 |
22 | -- DBA
23 | insert into auth_group_permissions (group_id, permission_id)
24 | select 3,id
25 | from auth_permission
26 | where codename in ('menu_dashboard','menu_sqlcheck','menu_sqlworkflow','menu_sqlanalyze','menu_query','menu_sqlquery','menu_queryapplylist','menu_sqloptimize','menu_sqladvisor','menu_slowquery','menu_instance','menu_instance_list','menu_dbdiagnostic','menu_database','menu_instance_account','menu_param','menu_data_dictionary','menu_tools','menu_archive','menu_binlog2sql','menu_schemasync','menu_system','menu_document','sql_submit','sql_review','sql_execute_for_resource_group','sql_execute','sql_analyze','optimize_sqladvisor','optimize_sqltuning','optimize_soar','query_applypriv','query_mgtpriv','query_review','query_submit','query_all_instances','query_resource_group_instance','process_view','process_kill','tablespace_view','trx_view','trxandlocks_view','instance_account_manage','param_view','param_edit','data_dictionary_export','archive_apply','archive_review','archive_mgt');
27 |
28 | -- PM
29 | insert into auth_group_permissions (group_id, permission_id)
30 | select 4,id
31 | from auth_permission
32 | where codename in ('menu_dashboard','menu_sqlcheck','menu_sqlworkflow','menu_sqlanalyze','menu_query','menu_sqlquery','menu_queryapplylist','menu_sqloptimize','menu_sqladvisor','menu_slowquery','menu_data_dictionary','menu_tools','menu_archive','menu_document','sql_submit','sql_review','sql_execute_for_resource_group','sql_execute','sql_analyze','optimize_sqladvisor','optimize_soar','query_applypriv','query_review','query_submit','archive_apply','archive_review');
33 |
34 | -- QA
35 | insert into auth_group_permissions (group_id, permission_id)
36 | select 5,id
37 | from auth_permission
38 | where codename in ('menu_dashboard','menu_sqlcheck','menu_sqlworkflow','menu_query','menu_sqlquery','menu_queryapplylist','menu_data_dictionary','menu_document','sql_submit','sql_execute','query_applypriv','query_submit');
39 |
40 | SET FOREIGN_KEY_CHECKS=1;
41 |
--------------------------------------------------------------------------------
/common/workflow.py:
--------------------------------------------------------------------------------
1 | import simplejson as json
2 | from django.contrib.auth.models import Group
3 | from django.http import HttpResponse
4 |
5 | from common.utils.const import WorkflowDict
6 | from common.utils.extend_json_encoder import ExtendJSONEncoder
7 | from sql.models import WorkflowAudit, WorkflowLog
8 | from sql.utils.resource_group import user_groups
9 |
10 |
11 | # 获取审核列表
12 | def lists(request):
13 | # 获取用户信息
14 | user = request.user
15 |
16 | limit = int(request.POST.get('limit'))
17 | offset = int(request.POST.get('offset'))
18 | workflow_type = int(request.POST.get('workflow_type'))
19 | limit = offset + limit
20 | search = request.POST.get('search', '')
21 |
22 | # 先获取用户所在资源组列表
23 | group_list = user_groups(user)
24 | group_ids = [group.group_id for group in group_list]
25 | # 再获取用户所在权限组列表
26 | if user.is_superuser:
27 | auth_group_ids = [group.id for group in Group.objects.all()]
28 | else:
29 | auth_group_ids = [group.id for group in Group.objects.filter(user=user)]
30 |
31 | # 只返回所在资源组当前待自己审核的数据
32 | workflow_audit = WorkflowAudit.objects.filter(
33 | workflow_title__icontains=search,
34 | current_status=WorkflowDict.workflow_status['audit_wait'],
35 | group_id__in=group_ids,
36 | current_audit__in=auth_group_ids
37 | )
38 | # 过滤工单类型
39 | if workflow_type != 0:
40 | workflow_audit = workflow_audit.filter(workflow_type=workflow_type)
41 |
42 | audit_list_count = workflow_audit.count()
43 | audit_list = workflow_audit.order_by('-audit_id')[offset:limit].values(
44 | 'audit_id', 'workflow_type',
45 | 'workflow_title', 'create_user_display',
46 | 'create_time', 'current_status',
47 | 'audit_auth_groups',
48 | 'current_audit',
49 | 'group_name')
50 |
51 | # QuerySet 序列化
52 | rows = [row for row in audit_list]
53 |
54 | result = {"total": audit_list_count, "rows": rows}
55 | # 返回查询结果
56 | return HttpResponse(json.dumps(result, cls=ExtendJSONEncoder, bigint_as_string=True),
57 | content_type='application/json')
58 |
59 |
60 | # 获取工单日志
61 | def log(request):
62 | workflow_id = request.POST.get('workflow_id')
63 | workflow_type = request.POST.get('workflow_type')
64 | try:
65 | audit_id = WorkflowAudit.objects.get(workflow_id=workflow_id, workflow_type=workflow_type).audit_id
66 | workflow_logs = WorkflowLog.objects.filter(audit_id=audit_id).order_by('-id').values(
67 | 'operation_type_desc',
68 | 'operation_info',
69 | 'operator_display',
70 | 'operation_time')
71 | count = WorkflowLog.objects.filter(audit_id=audit_id).count()
72 | except Exception:
73 | workflow_logs = []
74 | count = 0
75 |
76 | # QuerySet 序列化
77 | rows = [row for row in workflow_logs]
78 | result = {"total": count, "rows": rows}
79 | # 返回查询结果
80 | return HttpResponse(json.dumps(result, cls=ExtendJSONEncoder, bigint_as_string=True),
81 | content_type='application/json')
82 |
--------------------------------------------------------------------------------
/admin.sh:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | #########################################################################
4 | # Update Time : 2020-02-25
5 | # Author: alenx
6 | #########################################################################
7 |
8 | function init() {
9 | echo "Initing archery"
10 | echo "----------------"
11 | echo "安装/更新可能缺少的依赖: mysql-community-devel gcc gcc-c++ python-devel"
12 | sudo yum install -y epel-release
13 | sudo yum install -y mysql-devel gcc gcc-c++ python-devel MySQL-python
14 | sudo yum install -y python36 python3-devel python36-pip openldap-devel unixODBC-devel gettext
15 |
16 | python3 -m pip install virtualenv -i https://mirrors.aliyun.com/pypi/simple/
17 | if [ ! -d "venv" ]; then
18 | virtualenv --system-site-packages -p python3 venv
19 | fi
20 | source ./venv/bin/activate
21 | ./venv/bin/python3 -m pip install -r requirements.txt -i https://mirrors.aliyun.com/pypi/simple/
22 | echo "************************************************"
23 | echo -e "\033[32m init archery success \033[0m"
24 | echo -e "\033[32m welcome to archery 2.0 \033[0m"
25 | }
26 |
27 | function start() {
28 | echo "Starting archery"
29 | echo "----------------"
30 | source ./venv/bin/activate
31 | python3 manage.py collectstatic -v0 --noinput
32 | supervisord -c supervisord.conf
33 | echo -e "Start archery: [\033[32m ok \033[0m]"
34 | }
35 |
36 | function stop() {
37 | echo "Stoping archery"
38 | echo "----------------"
39 | source ./venv/bin/activate
40 | supervisorctl -c supervisord.conf stop all
41 | kill -9 $(ps -ef | grep "Archery" | grep -v grep | awk '{print $2}')
42 | echo -e "Stop archery: [\033[32m ok \033[0m]"
43 | }
44 |
45 | function restart() {
46 | stop
47 | echo ""
48 | start
49 | }
50 |
51 | function adduser() {
52 | echo "Add Admin Users "
53 | source ./venv/bin/activate
54 | python3 manage.py createsuperuser
55 | echo -e "Add Users: [\033[32m ok \033[0m]"
56 | }
57 |
58 | function migration() {
59 | echo "Migration archery"
60 | echo "----------------"
61 | source ./venv/bin/activate
62 | python3 manage.py makemigrations sql
63 | python3 manage.py migrate
64 | python3 manage.py dbshell" + gettext(value) + ""
25 | }
26 | else if (value === "workflow_abort") {
27 | return "" + gettext(value) + ""
28 | }
29 | else if (value === "workflow_manreviewing") {
30 | return "" + gettext(value) + ""
31 | }
32 | else if (value === "workflow_review_pass") {
33 | return "" + gettext(value) + ""
34 | }
35 | else if (value === "workflow_timingtask") {
36 | return "" + gettext(value) + ""
37 | }
38 | else if (value === "workflow_executing") {
39 | return "" + gettext(value) + ""
40 | }
41 | else if (value === "workflow_autoreviewwrong") {
42 | return "" + gettext(value) + ""
43 | }
44 | else if (value === "workflow_exception") {
45 | return "" + gettext(value) + ""
46 | }
47 | else if (value === "workflow_finish_manual") {
48 | return "" + gettext(value) + ""
49 | }
50 | else {
51 | return "" + "未知状态" + ""
52 | }
53 | }
54 |
55 | function workflow_type_formatter(value) {
56 | if (value === workflow_type.query) {
57 | return workflow_type.query_display
58 | }
59 | else if (value === workflow_type.sqlreview) {
60 | return workflow_type.sqlreview_display
61 | }
62 | else {
63 | return '未知状态'
64 | }
65 | }
66 |
67 | function workflow_status_formatter(value) {
68 | if (value === workflow_status.audit_wait) {
69 | return "" + workflow_status.audit_wait_display + " "
70 | }
71 | else if (value === workflow_status.audit_success) {
72 | return " " + workflow_status.audit_success_display + " "
73 | }
74 | else if (value === workflow_status.audit_reject) {
75 | return "" + workflow_status.audit_reject_display + " "
76 | }
77 | else if (value === workflow_status.audit_abort) {
78 | return "" + workflow_status.audit_abort_display + " "
79 | }
80 | else {
81 | return "" + '未知状态' + ""
82 | }
83 | }
84 |
--------------------------------------------------------------------------------
/common/static/ace/theme-mongodb.js:
--------------------------------------------------------------------------------
1 | /* eslint-disable */
2 | ace.define("ace/theme/mongodb",["require","exports","module","ace/lib/dom"], function(acequire, exports, module) {
3 |
4 | exports.isDark = false;
5 | exports.cssClass = "ace-mongodb";
6 | exports.cssText = "\
7 | .ace-mongodb .ace_gutter {\
8 | background: #f5f6f7;\
9 | color: #999999;\
10 | }\
11 | .ace-mongodb {\
12 | background: #f5f6f7;\
13 | color: #000;\
14 | }\
15 | .ace-mongodb .ace_keyword {\
16 | color: #999999;\
17 | font-weight: normal;\
18 | }\
19 | .ace-mongodb .ace_gutter-cell {\
20 | padding-left: 5px;\
21 | padding-right: 10px;\
22 | }\
23 | .ace-mongodb .ace_string {\
24 | color: #5b81a9;\
25 | }\
26 | .ace-mongodb .ace_boolean {\
27 | color: #5b81a9;\
28 | font-weight: normal;\
29 | }\
30 | .ace-mongodb .ace_constant.ace_numeric {\
31 | color: #5b81a9;\
32 | }\
33 | .ace-mongodb .ace_string.ace_regexp {\
34 | color: #5b81a9;\
35 | }\
36 | .ace-mongodb .ace_variable.ace_class {\
37 | color: teal;\
38 | }\
39 | .ace-mongodb .ace_constant.ace_buildin {\
40 | color: #0086B3;\
41 | }\
42 | .ace-mongodb .ace_support.ace_function {\
43 | color: #0086B3;\
44 | }\
45 | .ace-mongodb .ace_comment {\
46 | color: #998;\
47 | font-style: italic;\
48 | }\
49 | .ace-mongodb .ace_variable.ace_language {\
50 | color: #0086B3;\
51 | }\
52 | .ace-mongodb .ace_paren {\
53 | font-weight: normal;\
54 | }\
55 | .ace-mongodb .ace_variable.ace_instance {\
56 | color: teal;\
57 | }\
58 | .ace-mongodb .ace_constant.ace_language {\
59 | font-weight: bold;\
60 | }\
61 | .ace-mongodb .ace_cursor {\
62 | color: #999999;\
63 | }\
64 | .ace-mongodb.ace_focus .ace_marker-layer .ace_active-line {\
65 | background: #f5f6f7;\
66 | }\
67 | .ace-mongodb .ace_marker-layer .ace_active-line {\
68 | background: rgb(245, 245, 245);\
69 | }\
70 | .ace-mongodb .ace_marker-layer .ace_selection {\
71 | background: rgb(181, 213, 255);\
72 | }\
73 | .ace-mongodb.ace_multiselect .ace_selection.ace_start {\
74 | box-shadow: 0 0 3px 0px white;\
75 | }\
76 | .ace-mongodb.ace_nobold .ace_line > span {\
77 | font-weight: normal !important;\
78 | }\
79 | .ace-mongodb .ace_marker-layer .ace_step {\
80 | background: rgb(252, 255, 0);\
81 | }\
82 | .ace-mongodb .ace_marker-layer .ace_stack {\
83 | background: rgb(164, 229, 101);\
84 | }\
85 | .ace-mongodb .ace_marker-layer .ace_bracket {\
86 | margin: -1px 0 0 -1px;\
87 | border: 1px solid rgb(192, 192, 192);\
88 | }\
89 | .ace-mongodb .ace_gutter-active-line {\
90 | background: #f5f6f7;\
91 | }\
92 | .ace-mongodb .ace_marker-layer .ace_selected-word {\
93 | background: rgb(250, 250, 255);\
94 | border: 1px solid rgb(200, 200, 250);\
95 | }\
96 | .ace-mongodb .ace_invisible {\
97 | color: #BFBFBF\
98 | }\
99 | .ace-mongodb .ace_print-margin {\
100 | width: 1px;\
101 | background: #e8e8e8;\
102 | }\
103 | .ace-mongodb .ace_hidden-cursors {\
104 | opacity: 0;\
105 | }\
106 | .ace-mongodb .ace_indent-guide {\
107 | background: url(\"\") right repeat-y;\
108 | }";
109 | var dom = acequire("../lib/dom");
110 | dom.importCssString(exports.cssText, exports.cssClass);
111 | });
112 |
--------------------------------------------------------------------------------
/src/docker/Dockerfile-base:
--------------------------------------------------------------------------------
1 | FROM docker.io/centos:7
2 |
3 | ENV PYTHON_VERSION 3.6.5
4 | ENV DOCKERIZE_VERSION v0.6.1
5 | ENV SOAR_VERSION 0.11.0
6 |
7 | ENV TZ=Asia/Shanghai
8 | ENV LANG en_US.UTF-8
9 |
10 | WORKDIR /opt
11 |
12 | #locale
13 | RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ > /etc/timezone \
14 | && yum -y install kde-l10n-Chinese \
15 | && localedef -c -f UTF-8 -i zh_CN zh_CN.utf8
16 |
17 | ENV LC_ALL zh_CN.utf8
18 |
19 | #python
20 | RUN yum -y install libffi-devel wget gcc make zlib-devel openssl openssl-devel ncurses-devel openldap-devel gettext \
21 | && cd /opt \
22 | && wget "https://www.python.org/ftp/python/$PYTHON_VERSION/Python-$PYTHON_VERSION.tar.xz" \
23 | && tar -xvJf Python-$PYTHON_VERSION.tar.xz \
24 | && cd /opt/Python-$PYTHON_VERSION \
25 | && ./configure prefix=/usr/local/python3 \
26 | && make && make install \
27 | && ln -fs /usr/local/python3/bin/python3 /usr/bin/python3 \
28 | && ln -fs /usr/local/python3/bin/pip3 /usr/bin/pip3 \
29 | && pip3 install virtualenv \
30 | && cd /opt \
31 | && ln -fs /usr/local/python3/bin/virtualenv /usr/bin/virtualenv \
32 | && virtualenv venv4archery --python=python3 \
33 | && rm -rf Python-$PYTHON_VERSION \
34 | && rm -rf Python-$PYTHON_VERSION.tar.xz \
35 | #dockerize
36 | && wget https://github.com/jwilder/dockerize/releases/download/$DOCKERIZE_VERSION/dockerize-alpine-linux-amd64-$DOCKERIZE_VERSION.tar.gz \
37 | && tar -C /usr/local/bin -xzvf dockerize-alpine-linux-amd64-$DOCKERIZE_VERSION.tar.gz \
38 | && rm dockerize-alpine-linux-amd64-$DOCKERIZE_VERSION.tar.gz \
39 | #sqladvisor
40 | && yum -y install epel-release \
41 | && yum -y install cmake bison gcc-c++ git mysql-devel libaio-devel glib2 glib2-devel \
42 | && yum -y install https://repo.percona.com/yum/percona-release-latest.noarch.rpm \
43 | && yum -y install Percona-Server-devel-56 Percona-Server-shared-56 Percona-Server-client-56 \
44 | && yum -y install percona-toolkit \
45 | && cd /opt \
46 | && git clone https://github.com/hhyo/SQLAdvisor.git --depth 3 \
47 | && cd /opt/SQLAdvisor/ \
48 | && cmake -DBUILD_CONFIG=mysql_release -DCMAKE_BUILD_TYPE=debug -DCMAKE_INSTALL_PREFIX=/usr/local/sqlparser ./ \
49 | && make && make install \
50 | && cd sqladvisor/ \
51 | && cmake -DCMAKE_BUILD_TYPE=debug ./ \
52 | && make \
53 | && mv /opt/SQLAdvisor/sqladvisor/sqladvisor /opt \
54 | && rm -rf /opt/SQLAdvisor/
55 | #soar
56 | RUN cd /opt \
57 | && wget https://github.com/XiaoMi/soar/releases/download/$SOAR_VERSION/soar.linux-amd64 -O soar \
58 | && chmod a+x soar \
59 | #binlog2sql
60 | && cd /opt \
61 | && git clone https://github.com/danfengcao/binlog2sql.git \
62 | && mv binlog2sql/binlog2sql/ tmp_binlog2sql \
63 | && rm -rf binlog2sql \
64 | #msodbc
65 | && cd /opt \
66 | && curl https://packages.microsoft.com/config/rhel/7/prod.repo > /etc/yum.repos.d/mssql-release.repo \
67 | && ACCEPT_EULA=Y yum -y install msodbcsql17 \
68 | && yum -y install unixODBC-devel \
69 | #oracle instantclient
70 | && yum -y install http://yum.oracle.com/repo/OracleLinux/OL7/oracle/instantclient/x86_64/getPackage/oracle-instantclient19.3-basiclite-19.3.0.0.0-1.x86_64.rpm
71 |
72 |
--------------------------------------------------------------------------------
/sql/engines/mongo.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | import re
3 | import pymongo
4 | import logging
5 | import traceback
6 | import json
7 |
8 | from . import EngineBase
9 | from .models import ResultSet
10 | from bson import json_util
11 | from pymongo.errors import OperationFailure
12 |
13 | __author__ = 'jackie'
14 |
15 | logger = logging.getLogger('default')
16 |
17 |
18 | class MongoEngine(EngineBase):
19 | def get_connection(self, db_name=None):
20 | self.db_name = self.db_name or 'admin'
21 | conn = pymongo.MongoClient(self.host, self.port, authSource=self.db_name, connect=True, connectTimeoutMS=10000)
22 | if self.user and self.password:
23 | conn[self.db_name].authenticate(self.user, self.password, self.db_name)
24 | return conn
25 |
26 | @property
27 | def name(self): # pragma: no cover
28 | return 'Mongo'
29 |
30 | @property
31 | def info(self): # pragma: no cover
32 | return 'Mongo engine'
33 |
34 | def get_all_databases(self):
35 | result = ResultSet()
36 | conn = self.get_connection()
37 | try:
38 | result.rows = conn.list_database_names()
39 | except OperationFailure:
40 | result.rows = [self.db_name]
41 | return result
42 |
43 | def get_all_tables(self, db_name, **kwargs):
44 | result = ResultSet()
45 | conn = self.get_connection()
46 | db = conn[db_name]
47 | result.rows = db.list_collection_names()
48 | return result
49 |
50 | def query_check(self, db_name=None, sql=''):
51 | """提交查询前的检查"""
52 | result = {'msg': '', 'bad_query': True, 'filtered_sql': sql, 'has_star': False}
53 | safe_cmd = ['find']
54 | sql = sql.split('.')[1]
55 | for cmd in safe_cmd:
56 | if re.match(fr'^{cmd}\(.*', sql.strip(), re.I):
57 | result['bad_query'] = False
58 | break
59 | if result['bad_query']:
60 | result['msg'] = """禁止执行该命令!正确格式为:{collection_name}.find() or {collection_name}.find(expression)""", \
61 | """如 : 'test.find({"id":{"$gt":1.0}})'"""
62 | return result
63 |
64 | def query(self, db_name=None, sql='', limit_num=0, close_conn=True, **kwargs):
65 | result_set = ResultSet(full_sql=sql)
66 | try:
67 | conn = self.get_connection()
68 | db = conn[db_name]
69 | collect = db[sql.split('.')[0]]
70 | match = re.compile(r'[(](.*)[)]', re.S)
71 | sql = re.findall(match, sql)[0]
72 | if sql != '':
73 | sql = json.loads(sql)
74 | result = collect.find(sql).limit(limit_num)
75 | else:
76 | result = collect.find(sql).limit(limit_num)
77 | rows = json.loads(json_util.dumps(result))
78 | result_set.column_list = ['Result']
79 | if isinstance(rows, list):
80 | result_set.rows = tuple([json.dumps(x, ensure_ascii=False)] for x in rows)
81 | result_set.affected_rows = len(rows)
82 | except Exception as e:
83 | logger.warning(f"Mongo命令执行报错,语句:{sql}, 错误信息:{traceback.format_exc()}")
84 | result_set.error = str(e)
85 | return result_set
86 |
--------------------------------------------------------------------------------
/sql/sql_analyze.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: sql_analyze.py
6 | @time: 2019/03/14
7 | """
8 | import simplejson as json
9 | from django.contrib.auth.decorators import permission_required
10 |
11 | from common.config import SysConfig
12 | from sql.plugins.soar import Soar
13 | from sql.utils.resource_group import user_instances
14 | from sql.utils.sql_utils import generate_sql
15 | from django.http import HttpResponse, JsonResponse
16 | from common.utils.extend_json_encoder import ExtendJSONEncoder
17 | from .models import Instance
18 |
19 | __author__ = 'hhyo'
20 |
21 |
22 | @permission_required('sql.sql_analyze', raise_exception=True)
23 | def generate(request):
24 | """
25 | 解析上传文件为SQL列表
26 | :param request:
27 | :return:
28 | """
29 | text = request.POST.get('text')
30 | if text is None:
31 | result = {"total": 0, "rows": []}
32 | else:
33 | rows = generate_sql(text)
34 | result = {"total": len(rows), "rows": rows}
35 | return HttpResponse(json.dumps(result, cls=ExtendJSONEncoder, bigint_as_string=True),
36 | content_type='application/json')
37 |
38 |
39 | @permission_required('sql.sql_analyze', raise_exception=True)
40 | def analyze(request):
41 | """
42 | 利用soar分析SQL
43 | :param request:
44 | :return:
45 | """
46 | text = request.POST.get('text')
47 | instance_name = request.POST.get('instance_name')
48 | db_name = request.POST.get('db_name')
49 | if not text:
50 | result = {"total": 0, "rows": []}
51 | else:
52 | soar = Soar()
53 | if instance_name != '' and db_name != '':
54 | try:
55 | instance_info = user_instances(request.user, db_type=['mysql']).get(instance_name=instance_name)
56 | except Instance.DoesNotExist:
57 | return JsonResponse({'status': 1, 'msg': '你所在组未关联该实例!', 'data': []})
58 | soar_test_dsn = SysConfig().get('soar_test_dsn')
59 | # 获取实例连接信息
60 | online_dsn = "{user}:{pwd}@{host}:{port}/{db}".format(user=instance_info.user,
61 | pwd=instance_info.password,
62 | host=instance_info.host,
63 | port=instance_info.port,
64 | db=db_name)
65 | else:
66 | online_dsn = ''
67 | soar_test_dsn = ''
68 | args = {"report-type": "markdown",
69 | "query": '',
70 | "online-dsn": online_dsn,
71 | "test-dsn": soar_test_dsn,
72 | "allow-online-as-test": "false"}
73 | rows = generate_sql(text)
74 | for row in rows:
75 | args['query'] = row['sql'].replace('"', '\\"').replace('`', '').replace('\n', ' ')
76 | cmd_args = soar.generate_args2cmd(args=args, shell=True)
77 | stdout, stderr = soar.execute_cmd(cmd_args, shell=True).communicate()
78 | row['report'] = stdout if stdout else stderr
79 | result = {"total": len(rows), "rows": rows}
80 | return HttpResponse(json.dumps(result, cls=ExtendJSONEncoder, bigint_as_string=True),
81 | content_type='application/json')
82 |
--------------------------------------------------------------------------------
/src/init_sql/v1.7.4_v1.7.5.sql:
--------------------------------------------------------------------------------
1 | -- 增加归档相关权限
2 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
3 | insert IGNORE INTO auth_permission (name, content_type_id, codename) VALUES
4 | ('菜单 数据归档', @content_type_id, 'menu_archive'),
5 | ('提交归档申请', @content_type_id, 'archive_apply'),
6 | ('审核归档申请', @content_type_id, 'archive_audit'),
7 | ('管理归档申请', @content_type_id, 'archive_mgt');
8 |
9 | -- 归档配置表
10 | CREATE TABLE `archive_config` (
11 | `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键',
12 | `title` varchar(50) NOT NULL COMMENT '归档配置说明',
13 | `resource_group_id` int(11) NOT NULL COMMENT '资源组',
14 | `audit_auth_groups` varchar(255) NOT NULL COMMENT '审批权限组列表',
15 | `src_instance_id` int(11) NOT NULL COMMENT '源实例',
16 | `src_db_name` varchar(64) NOT NULL COMMENT '源数据库',
17 | `src_table_name` varchar(64) NOT NULL COMMENT '源表',
18 | `dest_instance_id` int(11) DEFAULT NULL COMMENT '目标实例',
19 | `dest_db_name` varchar(64) DEFAULT NULL COMMENT '目标数据库',
20 | `dest_table_name` varchar(64) DEFAULT NULL COMMENT '目标表',
21 | `condition` varchar(1000) NOT NULL COMMENT '归档条件,where条件',
22 | `mode` varchar(10) NOT NULL COMMENT '归档模式',
23 | `no_delete` tinyint(1) NOT NULL COMMENT '是否保留源数据',
24 | `sleep` int(11) NOT NULL COMMENT '归档limit行记录后的休眠秒数',
25 | `status` int(11) NOT NULL COMMENT '审核状态',
26 | `state` tinyint(1) NOT NULL COMMENT '是否启用归档',
27 | `user_name` varchar(30) NOT NULL COMMENT '申请人',
28 | `user_display` varchar(50) NOT NULL COMMENT '申请人中文名',
29 | `create_time` datetime(6) NOT NULL COMMENT '创建时间',
30 | `last_archive_time` datetime(6) DEFAULT NULL COMMENT '最近归档时间',
31 | `sys_time` datetime(6) NOT NULL COMMENT '系统时间修改',
32 | PRIMARY KEY (`id`),
33 | KEY `idx_dest_instance_id` (`dest_instance_id`),
34 | KEY `idx_resource_group_id` (`resource_group_id`),
35 | KEY `idx_src_instance_id` (`src_instance_id`),
36 | CONSTRAINT `fk_archive_dest_instance_id` FOREIGN KEY (`dest_instance_id`) REFERENCES `sql_instance` (`id`),
37 | CONSTRAINT `fk_archive_resource_id` FOREIGN KEY (`resource_group_id`) REFERENCES `resource_group` (`group_id`),
38 | CONSTRAINT `fk_archive_src_instance_id` FOREIGN KEY (`src_instance_id`) REFERENCES `sql_instance` (`id`)
39 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT '归档配置表';
40 |
41 | -- 归档日志表
42 | CREATE TABLE `archive_log` (
43 | `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键',
44 | `archive_id` int(11) NOT NULL COMMENT '归档配置ID',
45 | `cmd` varchar(2000) NOT NULL COMMENT '归档命令',
46 | `condition` varchar(1000) NOT NULL COMMENT '归档条件',
47 | `mode` varchar(10) NOT NULL COMMENT '归档模式',
48 | `no_delete` tinyint(1) NOT NULL COMMENT '是否保留源数据',
49 | `sleep` int(11) NOT NULL COMMENT '归档limit行记录后的休眠秒数',
50 | `select_cnt` int(11) NOT NULL COMMENT '查询数量',
51 | `insert_cnt` int(11) NOT NULL COMMENT '插入数量',
52 | `delete_cnt` int(11) NOT NULL COMMENT '删除数量',
53 | `statistics` longtext NOT NULL COMMENT '归档统计日志',
54 | `success` tinyint(1) NOT NULL COMMENT '是否归档成功',
55 | `error_info` longtext NOT NULL COMMENT '错误信息',
56 | `start_time` datetime(6) NOT NULL COMMENT '开始时间',
57 | `end_time` datetime(6) NOT NULL COMMENT '结束时间',
58 | `sys_time` datetime(6) NOT NULL COMMENT '系统时间修改',
59 | PRIMARY KEY (`id`),
60 | KEY `idx_archive_id` (`archive_id`),
61 | CONSTRAINT `fk_archive_config_id` FOREIGN KEY (`archive_id`) REFERENCES `archive_config` (`id`)
62 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT '归档日志表';
63 |
--------------------------------------------------------------------------------
/src/init_sql/v1.4.5_v1.5.0.sql:
--------------------------------------------------------------------------------
1 | -- 增加权限
2 | set @content_type_id=(select id from django_content_type where app_label='sql' and model='permission');
3 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('菜单 参数配置', @content_type_id, 'menu_param');
4 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('查看实例参数列表', @content_type_id, 'param_view');
5 | INSERT INTO auth_permission (name, content_type_id, codename) VALUES ('修改实例参数', @content_type_id, 'param_edit');
6 |
7 | -- 修改布尔值
8 | -- sql_workflow.is_backup
9 | UPDATE sql_workflow SET is_backup=1 WHERE is_backup='是';
10 | UPDATE sql_workflow SET is_backup=0 WHERE is_backup='否';
11 | ALTER TABLE sql_workflow
12 | MODIFY is_backup TINYINT NOT NULL DEFAULT 1 COMMENT '是否备份';
13 |
14 | -- data_masking_columns.active
15 | ALTER TABLE data_masking_columns
16 | MODIFY active TINYINT NOT NULL DEFAULT 0 COMMENT '激活状态';
17 |
18 | -- query_log.masking
19 | UPDATE query_log SET priv_check=0 WHERE priv_check=2;
20 | UPDATE query_log SET hit_rule=0 WHERE hit_rule=2;
21 | UPDATE query_log SET masking=0 WHERE masking=2;
22 | ALTER TABLE query_log
23 | MODIFY priv_check TINYINT NOT NULL DEFAULT 0 COMMENT '查询权限是否正常校验',
24 | MODIFY hit_rule TINYINT NOT NULL DEFAULT 0 COMMENT '查询是否命中脱敏规则',
25 | MODIFY masking TINYINT NOT NULL DEFAULT 0 COMMENT '查询结果是否正常脱敏';
26 |
27 | -- aliyun_access_key.is_enable
28 | UPDATE aliyun_access_key SET is_enable=0 WHERE is_enable=2;
29 | ALTER TABLE aliyun_access_key
30 | MODIFY is_enable TINYINT NOT NULL DEFAULT 1 COMMENT '是否启用';
31 |
32 | -- aliyun_rds_config.is_enable
33 | UPDATE aliyun_rds_config SET is_enable=0 WHERE is_enable=2;
34 | ALTER TABLE aliyun_rds_config
35 | MODIFY is_enable TINYINT NOT NULL DEFAULT 1 COMMENT '是否启用';
36 |
37 | -- 用户名和密码增加默认值
38 | ALTER TABLE sql_instance
39 | MODIFY `user` VARCHAR(200) NOT NULL DEFAULT '' COMMENT '用户名',
40 | MODIFY `password` VARCHAR(200) NOT NULL DEFAULT '' COMMENT '密码';
41 |
42 | -- 用户权限表增加索引
43 | ALTER TABLE query_privileges
44 | ADD INDEX idx_user_name_instance_id_db_name_valid_date(user_name,instance_id,db_name,valid_date);
45 |
46 | -- 实例参数配置功能
47 | CREATE TABLE param_template(
48 | id INT NOT NULL AUTO_INCREMENT PRIMARY KEY ,
49 | db_type VARCHAR(10) NOT NULL COMMENT '数据库类型,mysql、mssql、redis、pgsql',
50 | variable_name VARCHAR(64) NOT NULL COMMENT '参数名',
51 | default_value VARCHAR(1024) NOT NULL COMMENT '默认参数值',
52 | editable TINYINT NOT NULL COMMENT '是否支持修改',
53 | valid_values VARCHAR(1024) NOT NULL COMMENT '有效参数值,',
54 | description VARCHAR(1024) NOT NULL COMMENT '参数描述',
55 | create_time TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
56 | sys_time TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '创建时间',
57 | UNIQUE uniq_db_type_variable_name(db_type, variable_name)
58 | ) COMMENT '实例参数配置表';
59 |
60 |
61 |
62 | CREATE TABLE param_history(
63 | id INT NOT NULL AUTO_INCREMENT PRIMARY KEY ,
64 | instance_id INT NOT NULL COMMENT '实例ID',
65 | variable_name VARCHAR(64) NOT NULL COMMENT '参数名',
66 | old_var VARCHAR(1024) NOT NULL COMMENT '修改前参数值',
67 | new_var VARCHAR(1024) NOT NULL COMMENT '修改后参数值',
68 | set_sql VARCHAR(1024) NOT NULL COMMENT '在线变更配置执行的SQL语句',
69 | user_name VARCHAR(30) NOT NULL COMMENT '修改人',
70 | user_display VARCHAR(50) NOT NULL COMMENT '修改人中文名',
71 | update_time TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间',
72 | INDEX idx_variable_name(variable_name),
73 | CONSTRAINT fk_param_instance FOREIGN KEY fk_param_instance (instance_id) REFERENCES sql_instance (id) ON DELETE RESTRICT ON UPDATE RESTRICT
74 | ) COMMENT '实例参数修改历史';
75 |
--------------------------------------------------------------------------------
/sql/utils/execute_sql.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 |
3 | from django.db import close_old_connections, connection
4 | from django_redis import get_redis_connection
5 | from common.utils.const import WorkflowDict
6 | from sql.engines.models import ReviewResult, ReviewSet
7 | from sql.models import SqlWorkflow
8 | from sql.notify import notify_for_execute
9 | from sql.utils.workflow_audit import Audit
10 | from sql.engines import get_engine
11 |
12 |
13 | def execute(workflow_id):
14 | """为延时或异步任务准备的execute, 传入工单ID即可"""
15 | workflow_detail = SqlWorkflow.objects.get(id=workflow_id)
16 | # 给定时执行的工单增加执行日志
17 | if workflow_detail.status == 'workflow_timingtask':
18 | # 将工单状态修改为执行中
19 | SqlWorkflow(id=workflow_id, status='workflow_executing').save(update_fields=['status'])
20 | audit_id = Audit.detail_by_workflow_id(workflow_id=workflow_id,
21 | workflow_type=WorkflowDict.workflow_type['sqlreview']).audit_id
22 | Audit.add_log(audit_id=audit_id,
23 | operation_type=5,
24 | operation_type_desc='执行工单',
25 | operation_info='系统定时执行',
26 | operator='',
27 | operator_display='系统'
28 | )
29 | execute_engine = get_engine(instance=workflow_detail.instance)
30 | return execute_engine.execute_workflow(workflow=workflow_detail)
31 |
32 |
33 | def execute_callback(task):
34 | """异步任务的回调, 将结果填入数据库等等
35 | 使用django-q的hook, 传入参数为整个task
36 | task.result 是真正的结果
37 | """
38 | # https://stackoverflow.com/questions/7835272/django-operationalerror-2006-mysql-server-has-gone-away
39 | if connection.connection and not connection.is_usable():
40 | close_old_connections()
41 | workflow_id = task.args[0]
42 | workflow = SqlWorkflow.objects.get(id=workflow_id)
43 | workflow.finish_time = task.stopped
44 |
45 | if not task.success:
46 | # 不成功会返回错误堆栈信息,构造一个错误信息
47 | workflow.status = 'workflow_exception'
48 | execute_result = ReviewSet(full_sql=workflow.sqlworkflowcontent.sql_content)
49 | execute_result.rows = [ReviewResult(
50 | stage='Execute failed',
51 | errlevel=2,
52 | stagestatus='异常终止',
53 | errormessage=task.result,
54 | sql=workflow.sqlworkflowcontent.sql_content)]
55 | elif task.result.warning or task.result.error:
56 | execute_result = task.result
57 | workflow.status = 'workflow_exception'
58 | else:
59 | execute_result = task.result
60 | workflow.status = 'workflow_finish'
61 | # 保存执行结果
62 | workflow.sqlworkflowcontent.execute_result = execute_result.json()
63 | workflow.sqlworkflowcontent.save()
64 | workflow.save()
65 |
66 | # 增加工单日志
67 | audit_id = Audit.detail_by_workflow_id(workflow_id=workflow_id,
68 | workflow_type=WorkflowDict.workflow_type['sqlreview']).audit_id
69 | Audit.add_log(audit_id=audit_id,
70 | operation_type=6,
71 | operation_type_desc='执行结束',
72 | operation_info='执行结果:{}'.format(workflow.get_status_display()),
73 | operator='',
74 | operator_display='系统'
75 | )
76 |
77 | # DDL工单结束后清空实例资源缓存
78 | if workflow.syntax_type == 1:
79 | r = get_redis_connection("default")
80 | for key in r.scan_iter(match='*insRes*', count=2000):
81 | r.delete(key)
82 |
83 | # 发送消息
84 | notify_for_execute(workflow)
85 |
--------------------------------------------------------------------------------
/CODE_OF_CONDUCT.md:
--------------------------------------------------------------------------------
1 | # Contributor Covenant Code of Conduct
2 |
3 | ## Our Pledge
4 |
5 | In the interest of fostering an open and welcoming environment, we as
6 | contributors and maintainers pledge to making participation in our project and
7 | our community a harassment-free experience for everyone, regardless of age, body
8 | size, disability, ethnicity, sex characteristics, gender identity and expression,
9 | level of experience, education, socio-economic status, nationality, personal
10 | appearance, race, religion, or sexual identity and orientation.
11 |
12 | ## Our Standards
13 |
14 | Examples of behavior that contributes to creating a positive environment
15 | include:
16 |
17 | * Using welcoming and inclusive language
18 | * Being respectful of differing viewpoints and experiences
19 | * Gracefully accepting constructive criticism
20 | * Focusing on what is best for the community
21 | * Showing empathy towards other community members
22 |
23 | Examples of unacceptable behavior by participants include:
24 |
25 | * The use of sexualized language or imagery and unwelcome sexual attention or
26 | advances
27 | * Trolling, insulting/derogatory comments, and personal or political attacks
28 | * Public or private harassment
29 | * Publishing others' private information, such as a physical or electronic
30 | address, without explicit permission
31 | * Other conduct which could reasonably be considered inappropriate in a
32 | professional setting
33 |
34 | ## Our Responsibilities
35 |
36 | Project maintainers are responsible for clarifying the standards of acceptable
37 | behavior and are expected to take appropriate and fair corrective action in
38 | response to any instances of unacceptable behavior.
39 |
40 | Project maintainers have the right and responsibility to remove, edit, or
41 | reject comments, commits, code, wiki edits, issues, and other contributions
42 | that are not aligned to this Code of Conduct, or to ban temporarily or
43 | permanently any contributor for other behaviors that they deem inappropriate,
44 | threatening, offensive, or harmful.
45 |
46 | ## Scope
47 |
48 | This Code of Conduct applies both within project spaces and in public spaces
49 | when an individual is representing the project or its community. Examples of
50 | representing a project or community include using an official project e-mail
51 | address, posting via an official social media account, or acting as an appointed
52 | representative at an online or offline event. Representation of a project may be
53 | further defined and clarified by project maintainers.
54 |
55 | ## Enforcement
56 |
57 | Instances of abusive, harassing, or otherwise unacceptable behavior may be
58 | reported by contacting the project team at rtttte@qq.com. All
59 | complaints will be reviewed and investigated and will result in a response that
60 | is deemed necessary and appropriate to the circumstances. The project team is
61 | obligated to maintain confidentiality with regard to the reporter of an incident.
62 | Further details of specific enforcement policies may be posted separately.
63 |
64 | Project maintainers who do not follow or enforce the Code of Conduct in good
65 | faith may face temporary or permanent repercussions as determined by other
66 | members of the project's leadership.
67 |
68 | ## Attribution
69 |
70 | This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4,
71 | available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html
72 |
73 | [homepage]: https://www.contributor-covenant.org
74 |
75 | For answers to common questions about this code of conduct, see
76 | https://www.contributor-covenant.org/faq
77 |
--------------------------------------------------------------------------------
/common/utils/aliyun_sdk.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | import datetime
3 | import traceback
4 |
5 | from aliyunsdkcore.client import AcsClient
6 | from aliyunsdkrds.request.v20140815 import DescribeSlowLogsRequest, DescribeSlowLogRecordsRequest, \
7 | RequestServiceOfCloudDBARequest
8 | import simplejson as json
9 | from common.config import SysConfig
10 | import logging
11 |
12 | logger = logging.getLogger('default')
13 |
14 |
15 | class Aliyun(object):
16 | def __init__(self):
17 | try:
18 | sys_config = SysConfig()
19 | ak = sys_config.get('aliyun_ak')
20 | secret = sys_config.get('aliyun_secret')
21 | self.clt = AcsClient(ak=ak, secret=secret)
22 | except Exception as m:
23 | raise Exception(f'阿里云认证失败:{m}{traceback.format_exc()}')
24 |
25 | def request_api(self, request, *values):
26 | if values:
27 | for value in values:
28 | for k, v in value.items():
29 | request.add_query_param(k, v)
30 | request.set_accept_format('json')
31 | result = self.clt.do_action_with_exception(request)
32 | return json.dumps(json.loads(result.decode('utf-8')), indent=4, sort_keys=False, ensure_ascii=False)
33 |
34 | # 阿里云UTC时间转换为本地时区时间
35 | @staticmethod
36 | def utc2local(utc, utc_format):
37 | utc_time = datetime.datetime.strptime(utc, utc_format)
38 | local_tm = datetime.datetime.fromtimestamp(0)
39 | utc_tm = datetime.datetime.utcfromtimestamp(0)
40 | localtime = utc_time + (local_tm - utc_tm)
41 | return localtime
42 |
43 | def DescribeSlowLogs(self, DBInstanceId, StartTime, EndTime, **kwargs):
44 | """获取实例慢日志列表DBName,SortKey、PageSize、PageNumber"""
45 | request = DescribeSlowLogsRequest.DescribeSlowLogsRequest()
46 | values = {"action_name": "DescribeSlowLogs", "DBInstanceId": DBInstanceId,
47 | "StartTime": StartTime, "EndTime": EndTime, "SortKey": "TotalExecutionCounts"}
48 | values = dict(values, **kwargs)
49 | result = self.request_api(request, values)
50 | return result
51 |
52 | def DescribeSlowLogRecords(self, DBInstanceId, StartTime, EndTime, **kwargs):
53 | """查看慢日志明细SQLId,DBName、PageSize、PageNumber"""
54 | request = DescribeSlowLogRecordsRequest.DescribeSlowLogRecordsRequest()
55 | values = {"action_name": "DescribeSlowLogRecords", "DBInstanceId": DBInstanceId,
56 | "StartTime": StartTime, "EndTime": EndTime}
57 | values = dict(values, **kwargs)
58 | result = self.request_api(request, values)
59 | return result
60 |
61 | def RequestServiceOfCloudDBA(self, DBInstanceId, ServiceRequestType, ServiceRequestParam, **kwargs):
62 | """
63 | 获取统计信息:'GetTimedMonData',{"Language":"zh","KeyGroup":"mem_cpu_usage","KeyName":"","StartTime":"2018-01-15T04:03:26Z","EndTime":"2018-01-15T05:03:26Z"}
64 | mem_cpu_usage、iops_usage、detailed_disk_space
65 | 获取process信息:'ShowProcessList',{"Language":"zh","Command":"Query"} -- Not Sleep , All
66 | 终止进程:'ConfirmKillSessionRequest',{"Language":"zh","SQLRequestID":75865,"SQLStatement":"kill 34022786;"}
67 | 获取表空间信息:'GetSpaceStatForTables',{"Language": "zh", "OrderType": "Data"}
68 | 获取资源利用信息:'GetResourceUsage',{"Language":"zh"}
69 | """
70 | request = RequestServiceOfCloudDBARequest.RequestServiceOfCloudDBARequest()
71 | values = {"action_name": "RequestServiceOfCloudDBA", "DBInstanceId": DBInstanceId,
72 | "ServiceRequestType": ServiceRequestType, "ServiceRequestParam": ServiceRequestParam}
73 | values = dict(values, **kwargs)
74 | result = self.request_api(request, values)
75 | return result
76 |
--------------------------------------------------------------------------------
/common/config.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | import logging
3 | import traceback
4 |
5 | import simplejson as json
6 | from django.http import HttpResponse
7 |
8 | from common.utils.permission import superuser_required
9 | from sql.models import Config
10 | from django.db import transaction
11 | from django.core.cache import cache
12 |
13 | logger = logging.getLogger('default')
14 |
15 |
16 | class SysConfig(object):
17 | def __init__(self):
18 | self.sys_config = {}
19 | self.get_all_config()
20 |
21 | def get_all_config(self):
22 | # 优先获取缓存数据
23 | try:
24 | sys_config = cache.get('sys_config')
25 | except Exception as m:
26 | sys_config = None
27 | logger.error(f"读取缓存失败:{m}{traceback.format_exc()}")
28 |
29 | # 缓存获取失败从数据库获取并且尝试更新缓存
30 | if sys_config:
31 | self.sys_config = sys_config
32 | else:
33 | try:
34 | # 获取系统配置信息
35 | all_config = Config.objects.all().values('item', 'value')
36 | sys_config = {}
37 | for items in all_config:
38 | if items['value'] in ('true', 'True'):
39 | items['value'] = True
40 | elif items['value'] in ('false', 'False'):
41 | items['value'] = False
42 | sys_config[items['item']] = items['value']
43 | self.sys_config = sys_config
44 | except Exception as m:
45 | logger.error(f"获取系统配置信息失败:{m}{traceback.format_exc()}")
46 | self.sys_config = {}
47 | else:
48 | try:
49 | # 更新缓存
50 | cache.set('sys_config', self.sys_config, timeout=None)
51 | except Exception as m:
52 | logger.error(f"更新缓存失败:{m}{traceback.format_exc()}")
53 |
54 | def get(self, key, default_value=None):
55 | value = self.sys_config.get(key, default_value)
56 | # 是字符串的话, 如果是空, 或者全是空格, 返回默认值
57 | if isinstance(value, str) and value.strip() == '':
58 | return default_value
59 | return value
60 |
61 | def set(self, key, value):
62 | if value is True:
63 | value = 'true'
64 | elif value is False:
65 | value = 'false'
66 | config_item, created = Config.objects.get_or_create(item=key)
67 | config_item.value = value
68 | config_item.save()
69 | # 删除并更新缓存
70 | try:
71 | cache.delete('sys_config')
72 | except Exception as m:
73 | logger.error(f"删除缓存失败:{m}{traceback.format_exc()}")
74 | finally:
75 | self.get_all_config()
76 |
77 | def replace(self, configs):
78 | result = {'status': 0, 'msg': 'ok', 'data': []}
79 |
80 | # 清空并替换
81 | try:
82 | self.purge()
83 | with transaction.atomic():
84 | Config.objects.bulk_create(
85 | [Config(item=items['key'].strip(),
86 | value=str(items['value']).strip()) for items in json.loads(configs)])
87 | except Exception as e:
88 | logger.error(traceback.format_exc())
89 | result['status'] = 1
90 | result['msg'] = str(e)
91 | finally:
92 | self.get_all_config()
93 | return result
94 |
95 | def purge(self):
96 | """清除所有配置, 供测试以及replace方法使用"""
97 | try:
98 | self.sys_config = {}
99 | cache.delete('sys_config')
100 | except Exception as m:
101 | logger.error(f"删除缓存失败:{m}{traceback.format_exc()}")
102 | with transaction.atomic():
103 | Config.objects.all().delete()
104 |
105 |
106 | # 修改系统配置
107 | @superuser_required
108 | def change_config(request):
109 | configs = request.POST.get('configs')
110 | archer_config = SysConfig()
111 | result = archer_config.replace(configs)
112 | # 返回结果
113 | return HttpResponse(json.dumps(result), content_type='application/json')
114 |
--------------------------------------------------------------------------------
/common/static/bootstrap-table/extensions/editable/bootstrap-table-editable.min.js:
--------------------------------------------------------------------------------
1 | /**
2 | * bootstrap-table - An extended Bootstrap table with radio, checkbox, sort, pagination, and other added features. (supports twitter bootstrap v2 and v3).
3 | *
4 | * @version v1.14.2
5 | * @homepage https://bootstrap-table.com
6 | * @author wenzhixin (http://wenzhixin.net.cn/)
7 | * @license MIT
8 | */
9 |
10 | (function(a,b){if('function'==typeof define&&define.amd)define([],b);else if('undefined'!=typeof exports)b();else{b(),a.bootstrapTableEditable={exports:{}}.exports}})(this,function(){'use strict';function a(a,b){if(!(a instanceof b))throw new TypeError('Cannot call a class as a function')}function b(a,b){if(!a)throw new ReferenceError('this hasn\'t been initialised - super() hasn\'t been called');return b&&('object'==typeof b||'function'==typeof b)?b:a}function c(a,b){if('function'!=typeof b&&null!==b)throw new TypeError('Super expression must either be null or a function, not '+typeof b);a.prototype=Object.create(b&&b.prototype,{constructor:{value:a,enumerable:!1,writable:!0,configurable:!0}}),b&&(Object.setPrototypeOf?Object.setPrototypeOf(a,b):a.__proto__=b)}var d=function(){function a(a,b){for(var c,d=0;d':l}}})}},{key:'initBody',value:function(a){var b=this;e(i.prototype.__proto__||Object.getPrototypeOf(i.prototype),'initBody',this).call(this,a),this.options.editable&&(f.each(this.columns,function(a,c){if(c.editable){var d=b.getData(),e=b.$body.find('a[data-name="'+c.field+'"]');e.each(function(a,b){var e=f(b),h=e.closest('tr'),i=h.data('index'),j=d[i],k=g.calculateObjectValue(c,c.editable,[i,j,e],{});e.editable(k)}),e.off('save').on('save',function(a,d){var e=a.currentTarget,g=d.submitValue,h=f(e),i=b.getData(),j=h.parents('tr[data-index]').data('index'),k=i[j],l=k[c.field];h.data('value',g),k[c.field]=g,b.trigger('editable-save',c.field,k,l,h),b.resetFooter()}),e.off('shown').on('shown',function(a,d){var e=a.currentTarget,g=f(e),h=b.getData(),i=g.parents('tr[data-index]').data('index'),j=h[i];b.trigger('editable-shown',c.field,j,g,d)}),e.off('hidden').on('hidden',function(a,d){var e=a.currentTarget,g=f(e),h=b.getData(),i=g.parents('tr[data-index]').data('index'),j=h[i];b.trigger('editable-hidden',c.field,j,g,d)})}}),this.trigger('editable-init'))}}]),i}(f.BootstrapTable)})(jQuery)});
--------------------------------------------------------------------------------
/common/static/bootstrap-fileinput/js/locales/zh.js:
--------------------------------------------------------------------------------
1 | /*!
2 | * FileInput Chinese Translations
3 | *
4 | * This file must be loaded after 'fileinput.js'. Patterns in braces '{}', or
5 | * any HTML markup tags in the messages must not be converted or translated.
6 | *
7 | * @see http://github.com/kartik-v/bootstrap-fileinput
8 | * @author kangqf
9 | *
10 | * NOTE: this file must be saved in UTF-8 encoding.
11 | */
12 | (function ($) {
13 | "use strict";
14 |
15 | $.fn.fileinputLocales['zh'] = {
16 | fileSingle: '文件',
17 | filePlural: '个文件',
18 | browseLabel: '选择 …',
19 | removeLabel: '移除',
20 | removeTitle: '清除选中文件',
21 | cancelLabel: '取消',
22 | cancelTitle: '取消进行中的上传',
23 | uploadLabel: '上传',
24 | uploadTitle: '上传选中文件',
25 | msgNo: '没有',
26 | msgNoFilesSelected: '',
27 | msgCancelled: '取消',
28 | msgZoomModalHeading: '详细预览',
29 | msgFileRequired: '必须选择一个文件上传.',
30 | msgSizeTooSmall: '文件 "{name}" ({size} KB) 必须大于限定大小 {minSize} KB.',
31 | msgSizeTooLarge: '文件 "{name}" ({size} KB) 超过了允许大小 {maxSize} KB.',
32 | msgFilesTooLess: '你必须选择最少 {n} {files} 来上传. ',
33 | msgFilesTooMany: '选择的上传文件个数 ({n}) 超出最大文件的限制个数 {m}.',
34 | msgFileNotFound: '文件 "{name}" 未找到!',
35 | msgFileSecured: '安全限制,为了防止读取文件 "{name}".',
36 | msgFileNotReadable: '文件 "{name}" 不可读.',
37 | msgFilePreviewAborted: '取消 "{name}" 的预览.',
38 | msgFilePreviewError: '读取 "{name}" 时出现了一个错误.',
39 | msgInvalidFileName: '文件名 "{name}" 包含非法字符.',
40 | msgInvalidFileType: '不正确的类型 "{name}". 只支持 "{types}" 类型的文件.',
41 | msgInvalidFileExtension: '不正确的文件扩展名 "{name}". 只支持 "{extensions}" 的文件扩展名.',
42 | msgFileTypes: {
43 | 'image': 'image',
44 | 'html': 'HTML',
45 | 'text': 'text',
46 | 'video': 'video',
47 | 'audio': 'audio',
48 | 'flash': 'flash',
49 | 'pdf': 'PDF',
50 | 'object': 'object'
51 | },
52 | msgUploadAborted: '该文件上传被中止',
53 | msgUploadThreshold: '处理中...',
54 | msgUploadBegin: '正在初始化...',
55 | msgUploadEnd: '完成',
56 | msgUploadEmpty: '无效的文件上传.',
57 | msgUploadError: 'Error',
58 | msgValidationError: '验证错误',
59 | msgLoading: '加载第 {index} 文件 共 {files} …',
60 | msgProgress: '加载第 {index} 文件 共 {files} - {name} - {percent}% 完成.',
61 | msgSelected: '{n} {files} 选中',
62 | msgFoldersNotAllowed: '只支持拖拽文件! 跳过 {n} 拖拽的文件夹.',
63 | msgImageWidthSmall: '图像文件的"{name}"的宽度必须是至少{size}像素.',
64 | msgImageHeightSmall: '图像文件的"{name}"的高度必须至少为{size}像素.',
65 | msgImageWidthLarge: '图像文件"{name}"的宽度不能超过{size}像素.',
66 | msgImageHeightLarge: '图像文件"{name}"的高度不能超过{size}像素.',
67 | msgImageResizeError: '无法获取的图像尺寸调整。',
68 | msgImageResizeException: '调整图像大小时发生错误。{errors}',
69 | msgAjaxError: '{operation} 发生错误. 请重试!',
70 | msgAjaxProgressError: '{operation} 失败',
71 | ajaxOperations: {
72 | deleteThumb: '删除文件',
73 | uploadThumb: '上传文件',
74 | uploadBatch: '批量上传',
75 | uploadExtra: '表单数据上传'
76 | },
77 | dropZoneTitle: '拖拽文件到这里 …
支持多文件同时上传',
78 | dropZoneClickTitle: '
(或点击{files}按钮选择文件)',
79 | fileActionSettings: {
80 | removeTitle: '删除文件',
81 | uploadTitle: '上传文件',
82 | uploadRetryTitle: 'Retry upload',
83 | zoomTitle: '查看详情',
84 | dragTitle: '移动 / 重置',
85 | indicatorNewTitle: '没有上传',
86 | indicatorSuccessTitle: '上传',
87 | indicatorErrorTitle: '上传错误',
88 | indicatorLoadingTitle: '上传 ...'
89 | },
90 | previewZoomButtonTitles: {
91 | prev: '预览上一个文件',
92 | next: '预览下一个文件',
93 | toggleheader: '缩放',
94 | fullscreen: '全屏',
95 | borderless: '无边界模式',
96 | close: '关闭当前预览'
97 | }
98 | };
99 | })(window.jQuery);
100 |
--------------------------------------------------------------------------------
/sql/utils/sql_utils.py:
--------------------------------------------------------------------------------
1 | # -*- coding: UTF-8 -*-
2 | """
3 | @author: hhyo
4 | @license: Apache Licence
5 | @file: sql_utils.py
6 | @time: 2019/03/13
7 | """
8 | import re
9 | import xml
10 | import mybatis_mapper2sql
11 | import sqlparse
12 |
13 | from sql.utils.extract_tables import extract_tables as extract_tables_by_sql_parse
14 |
15 | __author__ = 'hhyo'
16 |
17 |
18 | def get_syntax_type(sql, parser=True, db_type='mysql'):
19 | """
20 | 返回SQL语句类型,仅判断DDL和DML
21 | :param sql:
22 | :param parser: 是否使用sqlparse解析
23 | :param db_type: 不使用sqlparse解析时需要提供该参数
24 | :return:
25 | """
26 | sql = remove_comments(sql=sql, db_type=db_type)
27 | if parser:
28 | try:
29 | statement = sqlparse.parse(sql)[0]
30 | syntax_type = statement.token_first(skip_cm=True).ttype.__str__()
31 | if syntax_type == 'Token.Keyword.DDL':
32 | syntax_type = 'DDL'
33 | elif syntax_type == 'Token.Keyword.DML':
34 | syntax_type = 'DML'
35 | except Exception:
36 | syntax_type = None
37 | else:
38 | if db_type == 'mysql':
39 | ddl_re = r"^alter|^create|^drop|^rename|^truncate"
40 | dml_re = r"^call|^delete|^do|^handler|^insert|^load\s+data|^load\s+xml|^replace|^select|^update"
41 | else:
42 | # TODO 其他数据库的解析正则
43 | return None
44 | if re.match(ddl_re, sql, re.I):
45 | syntax_type = 'DDL'
46 | elif re.match(dml_re, sql, re.I):
47 | syntax_type = 'DML'
48 | else:
49 | syntax_type = None
50 | return syntax_type
51 |
52 |
53 | def remove_comments(sql, db_type='mysql'):
54 | """
55 | 去除SQL语句中的注释信息
56 | 来源:https://stackoverflow.com/questions/35647841/parse-sql-file-with-comments-into-sqlite-with-python
57 | :param sql:
58 | :param db_type:
59 | :return:
60 | """
61 | sql_comments_re = {
62 | 'oracle':
63 | [r'(?:--)[^\n]*\n', r'(?:\W|^)(?:remark|rem)\s+[^\n]*\n'],
64 | 'mysql':
65 | [r'(?:#|--\s)[^\n]*\n']
66 | }
67 | specific_comment_re = sql_comments_re[db_type]
68 | additional_patterns = "|"
69 | if isinstance(specific_comment_re, str):
70 | additional_patterns += specific_comment_re
71 | elif isinstance(specific_comment_re, list):
72 | additional_patterns += "|".join(specific_comment_re)
73 | pattern = r"(\".*?\"|\'.*?\')|(/\*.*?\*/{})".format(additional_patterns)
74 | regex = re.compile(pattern, re.MULTILINE | re.DOTALL)
75 |
76 | def _replacer(match):
77 | if match.group(2):
78 | return ""
79 | else:
80 | return match.group(1)
81 |
82 | return regex.sub(_replacer, sql).strip()
83 |
84 |
85 | def extract_tables(sql):
86 | """
87 | 获取sql语句中的库、表名
88 | :param sql:
89 | :return:
90 | """
91 | tables = list()
92 | for i in extract_tables_by_sql_parse(sql):
93 | tables.append({
94 | "schema": i.schema,
95 | "name": i.name,
96 | })
97 | return tables
98 |
99 |
100 | def generate_sql(text):
101 | """
102 | 从SQL文本、MyBatis3 Mapper XML file文件中解析出sql 列表
103 | :param text:
104 | :return: [{"sql_id": key, "sql": soar.compress(value)}]
105 | """
106 | # 尝试XML解析
107 | try:
108 | mapper, xml_raw_text = mybatis_mapper2sql.create_mapper(xml_raw_text=text)
109 | statements = mybatis_mapper2sql.get_statement(mapper, result_type='list')
110 | rows = []
111 | # 压缩SQL语句,方便展示
112 | for statement in statements:
113 | for key, value in statement.items():
114 | row = {"sql_id": key, "sql": value}
115 | rows.append(row)
116 | except xml.etree.ElementTree.ParseError:
117 | # 删除注释语句
118 | text = sqlparse.format(text, strip_comments=True)
119 | statements = sqlparse.split(text)
120 | rows = []
121 | num = 0
122 | for statement in statements:
123 | num = num + 1
124 | row = {"sql_id": num, "sql": statement}
125 | rows.append(row)
126 | return rows
127 |
--------------------------------------------------------------------------------