├── .gitignore ├── LICENSE ├── README.md ├── archer ├── __init__.py ├── settings.py ├── urls.py └── wsgi.py ├── debug.bat ├── debug.sh ├── manage.py ├── patch └── django_1.8.17_admin_secure_archer.patch ├── requirements.txt ├── sql ├── __init__.py ├── admin.py ├── aes_decryptor.py ├── aliyun_api.py ├── aliyun_function.py ├── check_login_middleware.py ├── const.py ├── dao.py ├── data_masking.py ├── exception_logging_middleware.py ├── extend_json_encoder.py ├── inception.py ├── jobs.py ├── models.py ├── permission.py ├── processor.py ├── query.py ├── sendmail.py ├── sqlreview.py ├── static │ ├── ace │ │ ├── ace.js │ │ ├── ace_init.js │ │ ├── ext-language_tools.js │ │ ├── mode-sql.js │ │ ├── snippets │ │ │ └── sql.js │ │ ├── theme-github.js │ │ └── theme-textmate.js │ ├── admin │ │ ├── css │ │ │ ├── base.css │ │ │ ├── changelists.css │ │ │ ├── dashboard.css │ │ │ ├── forms.css │ │ │ ├── ie.css │ │ │ ├── login.css │ │ │ ├── rtl.css │ │ │ └── widgets.css │ │ ├── img │ │ │ ├── changelist-bg.gif │ │ │ ├── changelist-bg_rtl.gif │ │ │ ├── default-bg-reverse.gif │ │ │ ├── default-bg.gif │ │ │ ├── deleted-overlay.gif │ │ │ ├── gis │ │ │ │ ├── move_vertex_off.png │ │ │ │ └── move_vertex_on.png │ │ │ ├── icon-no.gif │ │ │ ├── icon-unknown.gif │ │ │ ├── icon-yes.gif │ │ │ ├── icon_addlink.gif │ │ │ ├── icon_alert.gif │ │ │ ├── icon_calendar.gif │ │ │ ├── icon_changelink.gif │ │ │ ├── icon_clock.gif │ │ │ ├── icon_deletelink.gif │ │ │ ├── icon_error.gif │ │ │ ├── icon_searchbox.png │ │ │ ├── icon_success.gif │ │ │ ├── inline-delete-8bit.png │ │ │ ├── inline-delete.png │ │ │ ├── inline-restore-8bit.png │ │ │ ├── inline-restore.png │ │ │ ├── inline-splitter-bg.gif │ │ │ ├── nav-bg-grabber.gif │ │ │ ├── nav-bg-reverse.gif │ │ │ ├── nav-bg-selected.gif │ │ │ ├── nav-bg.gif │ │ │ ├── selector-icons.gif │ │ │ ├── selector-search.gif │ │ │ ├── sorting-icons.gif │ │ │ ├── tooltag-add.png │ │ │ └── tooltag-arrowright.png │ │ └── js │ │ │ ├── LICENSE-JQUERY.txt │ │ │ ├── SelectBox.js │ │ │ ├── SelectFilter2.js │ │ │ ├── actions.js │ │ │ ├── actions.min.js │ │ │ ├── admin │ │ │ ├── DateTimeShortcuts.js │ │ │ └── RelatedObjectLookups.js │ │ │ ├── calendar.js │ │ │ ├── collapse.js │ │ │ ├── collapse.min.js │ │ │ ├── core.js │ │ │ ├── inlines.js │ │ │ ├── inlines.min.js │ │ │ ├── jquery.init.js │ │ │ ├── jquery.js │ │ │ ├── jquery.min.js │ │ │ ├── prepopulate.js │ │ │ ├── prepopulate.min.js │ │ │ ├── related-widget-wrapper.js │ │ │ ├── timeparse.js │ │ │ └── urlify.js │ ├── base.html │ ├── bootstrap-3.3.7-dist │ │ ├── css │ │ │ ├── bootstrap-theme.css │ │ │ ├── bootstrap-theme.css.map │ │ │ ├── bootstrap-theme.min.css │ │ │ ├── bootstrap-theme.min.css.map │ │ │ ├── bootstrap.css │ │ │ ├── bootstrap.css.map │ │ │ ├── bootstrap.min.css │ │ │ ├── bootstrap.min.css.map │ │ │ └── dashboard.css │ │ ├── fonts │ │ │ ├── glyphicons-halflings-regular.eot │ │ │ ├── glyphicons-halflings-regular.svg │ │ │ ├── glyphicons-halflings-regular.ttf │ │ │ ├── glyphicons-halflings-regular.woff │ │ │ └── glyphicons-halflings-regular.woff2 │ │ └── js │ │ │ ├── bootstrap.js │ │ │ ├── bootstrap.min.js │ │ │ └── npm.js │ ├── bootstrap-table │ │ ├── css │ │ │ └── bootstrap-table.min.css │ │ └── js │ │ │ ├── bootstrap-table-export.min.js │ │ │ ├── bootstrap-table-zh-CN.min.js │ │ │ ├── bootstrap-table.min.js │ │ │ └── tableExport.min.js │ ├── charts.html │ ├── daterangepicker │ │ ├── css │ │ │ └── daterangepicker.css │ │ └── js │ │ │ ├── daterangepicker.js │ │ │ └── moment.min.js │ ├── datetimepicker │ │ ├── css │ │ │ ├── bootstrap-datetimepicker.css │ │ │ ├── daterangepicker-bs2.css │ │ │ └── daterangepicker-bs3.css │ │ └── js │ │ │ ├── bootstrap-datetimepicker.js │ │ │ └── bootstrap-datetimepicker.zh-CN.js │ ├── dbaprinciples.html │ ├── detail.html │ ├── diagnosis.html │ ├── dist │ │ ├── css │ │ │ ├── bootstrap-select.css │ │ │ ├── bootstrap-select.css.map │ │ │ └── bootstrap-select.min.css │ │ └── js │ │ │ ├── bootstrap-select.js │ │ │ ├── bootstrap-select.js.map │ │ │ ├── bootstrap-select.min.js │ │ │ ├── i18n │ │ │ ├── defaults-bg_BG.js │ │ │ ├── defaults-bg_BG.min.js │ │ │ ├── defaults-cs_CZ.js │ │ │ ├── defaults-cs_CZ.min.js │ │ │ ├── defaults-da_DK.js │ │ │ ├── defaults-da_DK.min.js │ │ │ ├── defaults-de_DE.js │ │ │ ├── defaults-de_DE.min.js │ │ │ ├── defaults-en_US.js │ │ │ ├── defaults-en_US.min.js │ │ │ ├── defaults-es_CL.js │ │ │ ├── defaults-es_CL.min.js │ │ │ ├── defaults-eu.js │ │ │ ├── defaults-eu.min.js │ │ │ ├── defaults-fa_IR.js │ │ │ ├── defaults-fa_IR.min.js │ │ │ ├── defaults-fr_FR.js │ │ │ ├── defaults-fr_FR.min.js │ │ │ ├── defaults-hu_HU.js │ │ │ ├── defaults-hu_HU.min.js │ │ │ ├── defaults-it_IT.js │ │ │ ├── defaults-it_IT.min.js │ │ │ ├── defaults-ko_KR.js │ │ │ ├── defaults-ko_KR.min.js │ │ │ ├── defaults-nl_NL.js │ │ │ ├── defaults-nl_NL.min.js │ │ │ ├── defaults-pl_PL.js │ │ │ ├── defaults-pl_PL.min.js │ │ │ ├── defaults-pt_BR.js │ │ │ ├── defaults-pt_BR.min.js │ │ │ ├── defaults-pt_PT.js │ │ │ ├── defaults-pt_PT.min.js │ │ │ ├── defaults-ro_RO.js │ │ │ ├── defaults-ro_RO.min.js │ │ │ ├── defaults-ru_RU.js │ │ │ ├── defaults-ru_RU.min.js │ │ │ ├── defaults-sk_SK.js │ │ │ ├── defaults-sk_SK.min.js │ │ │ ├── defaults-sl_SI.js │ │ │ ├── defaults-sl_SI.min.js │ │ │ ├── defaults-sv_SE.js │ │ │ ├── defaults-sv_SE.min.js │ │ │ ├── defaults-tr_TR.js │ │ │ ├── defaults-tr_TR.min.js │ │ │ ├── defaults-ua_UA.js │ │ │ ├── defaults-ua_UA.min.js │ │ │ ├── defaults-zh_CN.js │ │ │ ├── defaults-zh_CN.min.js │ │ │ ├── defaults-zh_TW.js │ │ │ └── defaults-zh_TW.min.js │ │ │ └── sql-formatter.min.js │ ├── error.html │ ├── fileinput │ │ ├── css │ │ │ └── fileinput.min.css │ │ ├── img │ │ │ └── loading.gif │ │ └── js │ │ │ ├── fileinput.min.js │ │ │ └── locales │ │ │ └── zh.js │ ├── highcharts │ │ └── Chart.min.js │ ├── jquery │ │ └── jquery.min.js │ ├── login.html │ ├── queryapplydetail.html │ ├── queryapplylist.html │ ├── queryuserprivileges.html │ ├── rollback.html │ ├── slowquery.html │ ├── sqladvisor.html │ ├── sqlquery.html │ ├── sqlworkflow.html │ ├── submitSql.html │ ├── user │ │ ├── css │ │ │ └── user1.css │ │ └── js │ │ │ ├── autoreview.js │ │ │ ├── charts.js │ │ │ ├── formatter.js │ │ │ ├── login.js │ │ │ └── rollback.js │ └── workflow.html ├── templatetags │ ├── __init__.py │ └── format_tags.py ├── tests.py ├── urls.py ├── views.py ├── views_ajax.py └── workflow.py ├── src ├── docker │ ├── Dockerfile │ ├── nginx.conf │ ├── pymysql │ │ ├── connections.py │ │ └── cursors.py │ ├── requirements.txt │ └── startup.sh ├── docs │ ├── mysql_db_design_guide.docx │ └── mysql_db_design_guide.md ├── init_sql │ ├── master->v2.0.sql │ └── v1.1.1->v2.0.sql ├── screenshots │ ├── admin.png │ ├── allworkflow.png │ ├── applyforprivileges.png │ ├── autoreview.png │ ├── bugs │ │ ├── bug1.png │ │ ├── bug2.png │ │ └── bug3.png │ ├── charts.png │ ├── datamasking.png │ ├── datamaskingcolumns.png │ ├── datamaskingrules.png │ ├── finish.png │ ├── login.png │ ├── manageprivileges.png │ ├── osc_progress.png │ ├── process.png │ ├── pymysql.png │ ├── query.png │ ├── querylog.png │ ├── rollback.png │ ├── slowquery.png │ ├── slowquerylog.png │ ├── sqladvisor.png │ ├── submitsql.png │ ├── waitingforme.png │ └── workflowconfig.png └── script │ ├── analysis_slow_query.sh │ ├── centos7_install.sh │ └── mysql_slow_query_review.sql ├── startup.sh └── stop.sh /.gitignore: -------------------------------------------------------------------------------- 1 | *.pyc 2 | *.swp 3 | *.lock 4 | *.log 5 | .idea/ 6 | .DS_Store 7 | archer/settings.py.github 8 | archer/settings.py.dev 9 | sql/migrations/ 10 | -------------------------------------------------------------------------------- /archer/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/archer/__init__.py -------------------------------------------------------------------------------- /archer/urls.py: -------------------------------------------------------------------------------- 1 | """archer URL Configuration 2 | 3 | The `urlpatterns` list routes URLs to views. For more information please see: 4 | https://docs.djangoproject.com/en/1.8/topics/http/urls/ 5 | Examples: 6 | Function views 7 | 1. Add an import: from my_app import views 8 | 2. Add a URL to urlpatterns: url(r'^$', views.home, name='home') 9 | Class-based views 10 | 1. Add an import: from other_app.views import Home 11 | 2. Add a URL to urlpatterns: url(r'^$', Home.as_view(), name='home') 12 | Including another URLconf 13 | 1. Add a URL to urlpatterns: url(r'^blog/', include('blog.urls')) 14 | """ 15 | from django.conf.urls import include, url 16 | from django.contrib import admin 17 | 18 | urlpatterns = [ 19 | url(r'^admin/', include(admin.site.urls)), 20 | url(r'^', include('sql.urls', namespace="sql")), 21 | ] 22 | -------------------------------------------------------------------------------- /archer/wsgi.py: -------------------------------------------------------------------------------- 1 | """ 2 | WSGI config for archer 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", "archer.settings") 15 | 16 | application = get_wsgi_application() 17 | -------------------------------------------------------------------------------- /debug.bat: -------------------------------------------------------------------------------- 1 | python3 manage.py runserver 0.0.0.0:9123 -------------------------------------------------------------------------------- /debug.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | 3 | python3 manage.py runserver 0.0.0.0:9123 --insecure 4 | -------------------------------------------------------------------------------- /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", "archer.settings") 7 | 8 | from django.core.management import execute_from_command_line 9 | 10 | execute_from_command_line(sys.argv) 11 | -------------------------------------------------------------------------------- /patch/django_1.8.17_admin_secure_archer.patch: -------------------------------------------------------------------------------- 1 | --- views.py 2018-01-23 11:53:00.179201491 +0800 2 | +++ python/site-packages/django/contrib/auth/views.py 2018-01-23 11:58:10.668286140 +0800 3 | @@ -24,7 +24,14 @@ 4 | from django.views.decorators.cache import never_cache 5 | from django.views.decorators.csrf import csrf_protect 6 | from django.views.decorators.debug import sensitive_post_parameters 7 | - 8 | +# 账户锁定 9 | +from django.conf import settings 10 | +from sql.sendmail import MailSender 11 | +import datetime 12 | +import logging 13 | +logger = logging.getLogger('default') 14 | +login_failure_counter = {} 15 | +# 账户锁定end 16 | 17 | @sensitive_post_parameters() 18 | @csrf_protect 19 | @@ -41,8 +48,22 @@ 20 | 21 | if request.method == "POST": 22 | form = authentication_form(request, data=request.POST) 23 | - if form.is_valid(): 24 | - 25 | + 26 | + # 增加账户锁定 27 | + failed_cnt = settings.LOCK_CNT_THRESHOLD 28 | + locking_time = settings.LOCK_TIME_THRESHOLD 29 | + username = request.POST['username'] 30 | + mailSender = MailSender() 31 | + now_time = datetime.datetime.now() 32 | + mail_title = 'login inception admin' 33 | + login_failed_message = '' 34 | + 35 | + if username in login_failure_counter and login_failure_counter[username]['cnt'] >= failed_cnt and (now_time - login_failure_counter[username]["last_failure_time"]).seconds <= locking_time: 36 | + login_failed_message = 'user:{},login /admin failed, account locking...'.format(username) 37 | + logger.warning(login_failed_message) 38 | + mailSender.sendEmail(mail_title, login_failed_message, getattr(settings, 'MAIL_REVIEW_SECURE_ADDR')) 39 | + elif form.is_valid(): 40 | + logger.info('user:{},login /admin success'.format(username)) 41 | # Ensure the user-originating redirection url is safe. 42 | if not is_safe_url(url=redirect_to, host=request.get_host()): 43 | redirect_to = resolve_url(settings.LOGIN_REDIRECT_URL) 44 | @@ -51,6 +72,15 @@ 45 | auth_login(request, form.get_user()) 46 | 47 | return HttpResponseRedirect(redirect_to) 48 | + else: 49 | + if username in login_failure_counter and (now_time - login_failure_counter[username]["last_failure_time"]).seconds <= locking_time: 50 | + login_failure_counter[username]["cnt"] += 1 51 | + else: 52 | + login_failure_counter[username] = {"cnt":1, "last_failure_time": datetime.datetime.now()} 53 | + login_failed_message = 'user:{},login /admin failed, fail count:{}'.format(username, login_failure_counter[username]["cnt"]) 54 | + logger.warning(login_failed_message) 55 | + mailSender.sendEmail(mail_title, login_failed_message, getattr(settings, 'MAIL_REVIEW_SECURE_ADDR')) 56 | + #账户锁定end 57 | else: 58 | form = authentication_form(request) 59 | 60 | -------------------------------------------------------------------------------- /requirements.txt: -------------------------------------------------------------------------------- 1 | certifi==2022.12.7 2 | chardet==3.0.4 3 | Django==2.2.28 4 | idna==2.6 5 | Naked==0.1.31 6 | pycrypto==2.6.1 7 | PyMySQL==0.7.11 8 | PyYAML==5.4 9 | requests==2.20.0 10 | shellescape==3.4.1 11 | simplejson==3.14.0 12 | urllib3==1.26.5 13 | django-admin-bootstrapped==2.5.7 14 | django-apscheduler==0.2.8 15 | -------------------------------------------------------------------------------- /sql/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/__init__.py -------------------------------------------------------------------------------- /sql/admin.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | from django.contrib import admin 3 | from django.contrib.auth.admin import UserAdmin 4 | from django.contrib.auth.forms import UserCreationForm, UserChangeForm 5 | 6 | # Register your models here. 7 | from archer import settings 8 | from .models import users, master_config, slave_config, workflow, WorkflowAudit, WorkflowAuditSetting, \ 9 | DataMaskingColumns, DataMaskingRules, AliyunAccessKey, AliyunRdsConfig 10 | 11 | 12 | # 主库配置管理 13 | @admin.register(master_config) 14 | class master_configAdmin(admin.ModelAdmin): 15 | list_display = ('id', 'cluster_name', 'master_host', 'master_port', 'master_user', 'create_time') 16 | search_fields = ['id', 'cluster_name', 'master_host', 'master_port', 'master_user', 'master_password', 17 | 'create_time', 'update_time'] 18 | 19 | 20 | # SQL工单管理 21 | @admin.register(workflow) 22 | class workflowAdmin(admin.ModelAdmin): 23 | list_display = ('id', 'workflow_name', 'cluster_name', 'engineer', 'create_time', 'status', 'is_backup') 24 | search_fields = ['id', 'workflow_name', 'engineer', 'review_man', 'sql_content'] 25 | exclude = ('is_manual',) 26 | 27 | @admin.register(users) 28 | class usersAdmin(UserAdmin): 29 | def __init__(self, *args, **kwargs): 30 | super(UserAdmin, self).__init__(*args, **kwargs) 31 | self.list_display = ('id', 'username', 'display', 'role', 'email', 'is_superuser', 'is_staff', 'is_active') 32 | self.search_fields = ('id', 'username', 'display', 'role', 'email') 33 | # 以上的属性都可以在django源码的UserAdmin类中找到,我们做以覆盖 34 | 35 | def changelist_view(self, request, extra_context=None): 36 | # 这个方法在源码的admin/options.py文件的ModelAdmin这个类中定义,我们要重新定义它,以达到不同权限的用户,返回的表单内容不同 37 | if request.user.is_superuser: 38 | # 此字段定义UserChangeForm表单中的具体显示内容,并可以分类显示 39 | self.fieldsets = ( 40 | (('认证信息'), {'fields': ('username', 'password')}), 41 | (('个人信息'), {'fields': ('display', 'role', 'email')}), 42 | (('权限信息'), {'fields': ('is_active', 'is_staff')}), 43 | # (('Important dates'), {'fields': ('last_login', 'date_joined')}), 44 | ) 45 | # 此字段定义UserCreationForm表单中的具体显示内容 46 | self.add_fieldsets = ((None, {'classes': ('wide',), 47 | 'fields': ('username', 'display', 'role', 'email', 'password1', 'password2'), 48 | }), 49 | ) 50 | return super(UserAdmin, self).changelist_view(request, extra_context) 51 | 52 | 53 | if settings.QUERY: 54 | # 查询从库配置 55 | @admin.register(slave_config) 56 | class WorkflowAuditAdmin(admin.ModelAdmin): 57 | list_display = ( 58 | 'id', 'cluster_name', 'slave_host', 'slave_port', 'slave_user', 'create_time', 'update_time') 59 | search_fields = ['id', 'cluster_name', 'slave_host', 'slave_port', 'slave_user', 'slave_password', ] 60 | 61 | 62 | # 工作流审核配置 63 | @admin.register(WorkflowAuditSetting) 64 | class WorkflowAuditSettingAdmin(admin.ModelAdmin): 65 | list_display = ('audit_setting_id', 'workflow_type', 'audit_users',) 66 | 67 | if settings.DATA_MASKING_ON_OFF: 68 | # 脱敏字段页面定义 69 | @admin.register(DataMaskingColumns) 70 | class DataMaskingColumnsAdmin(admin.ModelAdmin): 71 | list_display = ( 72 | 'column_id', 'rule_type', 'active', 'cluster_name', 'table_schema', 'table_name', 'column_name', 73 | 'create_time',) 74 | search_fields = ['cluster_name', 'table_schema', 'table_name', 'column_name'] 75 | 76 | 77 | # 脱敏规则页面定义 78 | @admin.register(DataMaskingRules) 79 | class DataMaskingRulesAdmin(admin.ModelAdmin): 80 | list_display = ( 81 | 'rule_type', 'rule_regex', 'hide_group', 'rule_desc', 'sys_time',) 82 | 83 | if settings.ALIYUN_RDS_MANAGE: 84 | # 阿里云的认证信息 85 | @admin.register(AliyunAccessKey) 86 | class AliyunAccessKeyAdmin(admin.ModelAdmin): 87 | list_display = ('ak', 'secret', 'is_enable', 'remark',) 88 | 89 | 90 | # 阿里云集群配置信息 91 | @admin.register(AliyunRdsConfig) 92 | class AliyunRdsConfigAdmin(admin.ModelAdmin): 93 | list_display = ('cluster_name', 'rds_dbinstanceid',) 94 | -------------------------------------------------------------------------------- /sql/aes_decryptor.py: -------------------------------------------------------------------------------- 1 | from Crypto.Cipher import AES 2 | from binascii import b2a_hex, a2b_hex 3 | 4 | class Prpcrypt(): 5 | def __init__(self): 6 | self.key = 'eCcGFZQj6PNoSSma31LR39rTzTbLkU8E'.encode('utf-8') 7 | self.mode = AES.MODE_CBC 8 | 9 | #加密函数,如果text不足16位就用空格补足为16位, 10 | #如果大于16当时不是16的倍数,那就补足为16的倍数。 11 | def encrypt(self,text): 12 | cryptor = AES.new(self.key,self.mode,b'0000000000000000') 13 | #这里密钥key 长度必须为16(AES-128), 14 | #24(AES-192),或者32 (AES-256)Bytes 长度 15 | #目前AES-128 足够目前使用 16 | length = 16 17 | count = len(text) 18 | if count < length: 19 | add = (length-count) 20 | #\0 backspace 21 | text = text + ('\0' * add) 22 | elif count > length: 23 | add = (length-(count % length)) 24 | text = text + ('\0' * add) 25 | self.ciphertext = cryptor.encrypt(text.encode('utf-8')) 26 | #因为AES加密时候得到的字符串不一定是ascii字符集的,输出到终端或者保存时候可能存在问题 27 | #所以这里统一把加密后的字符串转化为16进制字符串 28 | return b2a_hex(self.ciphertext).decode(encoding='utf-8') 29 | 30 | #解密后,去掉补足的空格用strip() 去掉 31 | def decrypt(self,text): 32 | cryptor = AES.new(self.key,self.mode,b'0000000000000000') 33 | plain_text = cryptor.decrypt(a2b_hex(text)) 34 | return plain_text.decode().rstrip('\0') 35 | 36 | if __name__ == '__main__': 37 | pc = Prpcrypt() #初始化密钥 38 | e = pc.encrypt('123456') #加密 39 | d = pc.decrypt(e) #解密 40 | print("加密:",str(e)) 41 | print("解密:",str(d)) 42 | -------------------------------------------------------------------------------- /sql/aliyun_api.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | import datetime 3 | from aliyunsdkcore import client 4 | from aliyunsdkrds.request.v20140815 import DescribeSlowLogsRequest, DescribeSlowLogRecordsRequest, \ 5 | RequestServiceOfCloudDBARequest 6 | import simplejson as json 7 | from .models import AliyunAccessKey 8 | from .aes_decryptor import Prpcrypt 9 | import logging 10 | 11 | logger = logging.getLogger('default') 12 | 13 | class Aliyun(object): 14 | def __init__(self): 15 | try: 16 | auth = AliyunAccessKey.objects.filter(is_enable=1) 17 | prpCryptor = Prpcrypt() 18 | ak = prpCryptor.decrypt(auth[0].ak) 19 | secret = prpCryptor.decrypt(auth[0].secret) 20 | except Exception: 21 | logger.error('没有找到有效的ak信息!') 22 | else: 23 | self.clt = client.AcsClient( 24 | ak=ak, 25 | secret=secret) 26 | def request_api(self, request, *values): 27 | if values: 28 | for value in values: 29 | for k, v in value.items(): 30 | request.add_query_param(k, v) 31 | request.set_accept_format('json') 32 | result = self.clt.do_action_with_exception(request) 33 | return json.dumps(json.loads(result.decode('utf-8')), indent=4, sort_keys=False, ensure_ascii=False) 34 | 35 | # 阿里云2017-12-10T16:00:00Z时间加上8小时时区显示 36 | def aliyun_time_format(self, str_time): 37 | if 'T' in str_time: 38 | Ymd = str_time.split('T')[0] 39 | HMS = str_time.split('T')[1].split('Z')[0] 40 | str_time = '%s %s' % (Ymd, HMS) 41 | time = datetime.datetime.strptime(str_time, "%Y-%m-%d %H:%M:%S") 42 | format_time = time + datetime.timedelta(hours=8) 43 | elif 'Z' in str_time: 44 | Ymd = str_time.split('Z')[0] 45 | format_time = '%s' % Ymd 46 | else: 47 | format_time = str_time 48 | return format_time 49 | 50 | def DescribeSlowLogs(self, DBInstanceId, StartTime, EndTime, **kwargs): 51 | '''获取集群慢日志列表 52 | DBName,SortKey、PageSize、PageNumber''' 53 | request = DescribeSlowLogsRequest.DescribeSlowLogsRequest() 54 | values = {"action_name": "DescribeSlowLogs", "DBInstanceId": DBInstanceId, 55 | "StartTime": StartTime, "EndTime": EndTime, "SortKey": "TotalExecutionCounts"} 56 | values = dict(values, **kwargs) 57 | result = self.request_api(request, values) 58 | return result 59 | 60 | def DescribeSlowLogRecords(self, DBInstanceId, StartTime, EndTime, **kwargs): 61 | '''查看慢日志明细 62 | SQLId,DBName、PageSize、PageNumber''' 63 | request = DescribeSlowLogRecordsRequest.DescribeSlowLogRecordsRequest() 64 | values = {"action_name": "DescribeSlowLogRecords", "DBInstanceId": DBInstanceId, 65 | "StartTime": StartTime, "EndTime": EndTime} 66 | values = dict(values, **kwargs) 67 | result = self.request_api(request, values) 68 | return result 69 | 70 | def RequestServiceOfCloudDBA(self, DBInstanceId, ServiceRequestType, ServiceRequestParam, **kwargs): 71 | ''' 72 | 获取统计信息:'GetTimedMonData',{"Language":"zh","KeyGroup":"mem_cpu_usage","KeyName":"","StartTime":"2018-01-15T04:03:26Z","EndTime":"2018-01-15T05:03:26Z"} 73 | mem_cpu_usage、iops_usage、detailed_disk_space 74 | 获取process信息:'ShowProcessList',{"Language":"zh","Command":"Query"} -- Not Sleep , All 75 | 终止进程:'ConfirmKillSessionRequest',{"Language":"zh","SQLRequestID":75865,"SQLStatement":"kill 34022786;"} 76 | 获取表空间信息:'GetSpaceStatForTables',{"Language": "zh", "OrderType": "Data"} 77 | 获取资源利用信息:'GetResourceUsage',{"Language":"zh"} 78 | ''' 79 | request = RequestServiceOfCloudDBARequest.RequestServiceOfCloudDBARequest() 80 | values = {"action_name": "RequestServiceOfCloudDBA", "DBInstanceId": DBInstanceId, 81 | "ServiceRequestType": ServiceRequestType, "ServiceRequestParam": ServiceRequestParam} 82 | values = dict(values, **kwargs) 83 | result = self.request_api(request, values) 84 | return result 85 | -------------------------------------------------------------------------------- /sql/check_login_middleware.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | import re 3 | from django.http import HttpResponseRedirect 4 | 5 | class CheckLoginMiddleware(object): 6 | def process_request(self, request): 7 | """ 8 | 该函数在每个函数之前检查是否登录,若未登录,则重定向到/login/ 9 | """ 10 | if request.session.get('login_username', False) in (False, '匿名用户'): 11 | #以下是不用跳转到login页面的url白名单 12 | if request.path not in ('/login/', '/authenticate/') and re.match(r"/admin/\w*", request.path) is None: 13 | return HttpResponseRedirect('/login/') 14 | -------------------------------------------------------------------------------- /sql/const.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | 3 | class Const(object): 4 | workflowStatus = { 5 | 'finish': '已正常结束', 6 | 'abort': '人工终止流程', 7 | 'autoreviewing': '自动审核中', 8 | 'manreviewing': '等待审核人审核', 9 | 'pass': '审核通过', 10 | 'timingtask': '定时执行', 11 | 'executing': '执行中', 12 | 'autoreviewwrong': '自动审核不通过', 13 | 'exception': '执行有异常', 14 | } 15 | # 定时任务id的前缀 16 | workflowJobprefix = { 17 | 'query': 'query', 18 | 'sqlreview': 'sqlreview', 19 | } 20 | 21 | 22 | class WorkflowDict: 23 | # 工作流申请类型,1.query 24 | workflow_type = { 25 | 'query': 1, 26 | 'query_display': '查询权限申请', 27 | } 28 | 29 | # 工作流状态,0.待审核 1.审核通过 2.审核不通过 3.审核取消 30 | workflow_status = { 31 | 'audit_wait': 0, 32 | 'audit_wait_display': '待审核', 33 | 'audit_success': 1, 34 | 'audit_success_display': '审核通过', 35 | 'audit_reject': 2, 36 | 'audit_reject_display': '审核不通过', 37 | 'audit_abort': 3, 38 | 'audit_abort_display': '审核取消', 39 | } 40 | -------------------------------------------------------------------------------- /sql/exception_logging_middleware.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | import logging 3 | 4 | logger = logging.getLogger('default') 5 | 6 | 7 | class ExceptionLoggingMiddleware(object): 8 | def process_exception(self, request, exception): 9 | import traceback 10 | logger.error(traceback.format_exc()) 11 | -------------------------------------------------------------------------------- /sql/extend_json_encoder.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | import simplejson as json 3 | 4 | from datetime import datetime, date, timedelta 5 | from decimal import Decimal 6 | from functools import singledispatch 7 | 8 | 9 | class MyClass: 10 | def __init__(self, value): 11 | self._value = value 12 | 13 | def get_value(self): 14 | return self._value 15 | 16 | 17 | # 创建非内置类型的实例 18 | mc = MyClass('i am class MyClass ') 19 | dm = Decimal('11.11') 20 | dt = datetime.now() 21 | dat = date.today() 22 | 23 | 24 | @singledispatch 25 | def convert(o): 26 | raise TypeError('can not convert type') 27 | 28 | 29 | @convert.register(datetime) 30 | def _(o): 31 | return o.strftime('%Y-%m-%d %H:%M:%S') 32 | 33 | 34 | @convert.register(date) 35 | def _(o): 36 | return o.strftime('%Y-%m-%d') 37 | 38 | 39 | @convert.register(timedelta) 40 | def _(o): 41 | return o.total_seconds() 42 | 43 | # @convert.register(Decimal) 44 | # def _(o): 45 | # return float(o) 46 | 47 | 48 | @convert.register(MyClass) 49 | def _(o): 50 | return o.get_value() 51 | 52 | 53 | class ExtendJSONEncoder(json.JSONEncoder): 54 | def default(self, obj): 55 | try: 56 | return convert(obj) 57 | except TypeError: 58 | return super(ExtendJSONEncoder, self).default(obj) 59 | 60 | 61 | data = { 62 | 'mc': mc, 63 | 'dm': dm, 64 | 'dt': dt, 65 | 'dat': dat, 66 | 'tl': timedelta(minutes=30), 67 | 'bigint': 988983860501598208 68 | } 69 | 70 | # print(json.dumps(data, cls=ExtendJSONEncoder, bigint_as_string=True)) 71 | # print(json.dumps(data, cls=ExtendJSONEncoder, bigint_as_string=True, default=str)) 72 | -------------------------------------------------------------------------------- /sql/jobs.py: -------------------------------------------------------------------------------- 1 | # -*- coding:utf-8 -*- 2 | import datetime 3 | import time 4 | 5 | from apscheduler.schedulers.background import BackgroundScheduler 6 | from apscheduler.schedulers import SchedulerAlreadyRunningError, SchedulerNotRunningError 7 | from django.core.urlresolvers import reverse 8 | from django.http import HttpResponseRedirect 9 | from django.shortcuts import render 10 | from django_apscheduler.jobstores import DjangoJobStore, register_events, register_job 11 | 12 | from sql.const import Const 13 | from sql.models import workflow 14 | from .sqlreview import execute_job, getDetailUrl 15 | 16 | import logging 17 | 18 | logging.basicConfig() 19 | logging.getLogger('apscheduler').setLevel(logging.DEBUG) 20 | 21 | logger = logging.getLogger('default') 22 | 23 | # 初始化scheduler 24 | scheduler = BackgroundScheduler() 25 | scheduler.add_jobstore(DjangoJobStore(), "default") 26 | register_events(scheduler) 27 | try: 28 | scheduler.start() 29 | logger.debug("Scheduler started!") 30 | except SchedulerAlreadyRunningError: 31 | logger.debug("Scheduler is already running!") 32 | 33 | 34 | # 添加/修改sql执行任务 35 | def add_sqlcronjob(job_id, run_date, workflowId, url): 36 | scheduler = BackgroundScheduler() 37 | scheduler.add_jobstore(DjangoJobStore(), "default") 38 | scheduler.add_job(execute_job, 'date', run_date=run_date, args=[workflowId, url], id=job_id, 39 | replace_existing=True) 40 | register_events(scheduler) 41 | try: 42 | scheduler.start() 43 | logger.debug("Scheduler started!") 44 | except SchedulerAlreadyRunningError: 45 | logger.debug("Scheduler is already running!") 46 | logger.debug('add_sqlcronjob:' + job_id + " run_date:" + run_date.strftime('%Y-%m-%d %H:%M:%S')) 47 | 48 | 49 | # 删除sql执行任务 50 | def del_sqlcronjob(job_id): 51 | logger.debug('del_sqlcronjob:' + job_id) 52 | return scheduler.remove_job(job_id) 53 | 54 | 55 | # 获取任务详情 56 | def job_info(job_id): 57 | return scheduler.get_job(job_id) 58 | -------------------------------------------------------------------------------- /sql/permission.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | import simplejson as json 3 | from django.shortcuts import render 4 | from django.http import HttpResponse 5 | from .models import users 6 | 7 | 8 | # 管理员操作权限验证 9 | def superuser_required(func): 10 | def wrapper(request, *args, **kw): 11 | # 获取用户信息,权限验证 12 | loginUser = request.session.get('login_username', False) 13 | loginUserOb = users.objects.get(username=loginUser) 14 | 15 | if loginUserOb.is_superuser is False: 16 | if request.is_ajax(): 17 | finalResult = {'status': 1, 'msg': '您无权操作,请联系管理员', 'data': []} 18 | return HttpResponse(json.dumps(finalResult), content_type='application/json') 19 | else: 20 | context = {'errMsg': "您无权操作,请联系管理员"} 21 | return render(request, "error.html", context) 22 | 23 | return func(request, *args, **kw) 24 | 25 | return wrapper 26 | 27 | 28 | # 角色操作权限验证 29 | def role_required(roles=()): 30 | def _deco(func): 31 | def wrapper(request, *args, **kw): 32 | # 获取用户信息,权限验证 33 | loginUser = request.session.get('login_username', False) 34 | loginUserOb = users.objects.get(username=loginUser) 35 | loginrole = loginUserOb.role 36 | 37 | if loginrole not in roles and loginUserOb.is_superuser is False: 38 | if request.is_ajax(): 39 | finalResult = {'status': 1, 'msg': '您无权操作,请联系管理员', 'data': []} 40 | return HttpResponse(json.dumps(finalResult), content_type='application/json') 41 | else: 42 | context = {'errMsg': "您无权操作,请联系管理员"} 43 | return render(request, "error.html", context) 44 | 45 | return func(request, *args, **kw) 46 | 47 | return wrapper 48 | 49 | return _deco 50 | -------------------------------------------------------------------------------- /sql/processor.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | from sql.workflow import Workflow 3 | from .models import users 4 | from django.conf import settings 5 | 6 | leftMenuBtnsCommon = ( 7 | {'key': 'allworkflow', 'name': 'SQL上线工单', 'url': '/allworkflow/', 'class': 'glyphicon glyphicon-home', 8 | 'display': True}, 9 | {'key': 'sqlquery', 'name': 'SQL在线查询', 'url': '/sqlquery/', 'class': 'glyphicon glyphicon-search', 10 | 'display': settings.QUERY}, 11 | {'key': 'slowquery', 'name': 'SQL慢查日志', 'url': '/slowquery/', 'class': 'glyphicon glyphicon-align-right', 12 | 'display': settings.SLOWQUERY}, 13 | {'key': 'sqladvisor', 'name': 'SQL优化工具', 'url': '/sqladvisor/', 'class': 'glyphicon glyphicon-wrench', 14 | 'display': settings.SQLADVISOR}, 15 | {'key': 'queryapply', 'name': '查询权限管理', 'url': '/queryapplylist/', 'class': 'glyphicon glyphicon-eye-open', 16 | 'display': settings.QUERY} 17 | ) 18 | 19 | leftMenuBtnsSuper = ( 20 | {'key': 'diagnosis', 'name': '主库会话管理', 'url': '/diagnosis_process/', 'class': 'glyphicon glyphicon-scissors', 21 | 'display': True}, 22 | {'key': 'admin', 'name': '后台数据管理', 'url': '/admin/sql/', 'class': 'glyphicon glyphicon-list', 'display': True}, 23 | ) 24 | 25 | leftMenuBtnsDoc = ( 26 | {'key': 'dbaprinciples', 'name': 'SQL审核必读', 'url': '/dbaprinciples/', 'class': 'glyphicon glyphicon-book', 27 | 'display': True}, 28 | {'key': 'charts', 'name': '统计图表展示', 'url': '/charts/', 'class': 'glyphicon glyphicon-file', 'display': True}, 29 | ) 30 | 31 | 32 | def global_info(request): 33 | """存放用户,会话信息等.""" 34 | loginUser = request.session.get('login_username', None) 35 | if loginUser is not None: 36 | user = users.objects.get(username=loginUser) 37 | UserDisplay = user.display 38 | if UserDisplay == '': 39 | UserDisplay = loginUser 40 | if user.is_superuser: 41 | leftMenuBtns = leftMenuBtnsCommon + leftMenuBtnsSuper + leftMenuBtnsDoc 42 | else: 43 | leftMenuBtns = leftMenuBtnsCommon + leftMenuBtnsDoc 44 | # 获取代办数量 45 | try: 46 | todo = Workflow().auditlist(user, 0, 0, 1)['data']['auditlistCount'] 47 | except Exception: 48 | todo = 0 49 | else: 50 | leftMenuBtns = () 51 | UserDisplay = '' 52 | todo = 0 53 | 54 | return { 55 | 'loginUser': loginUser, 56 | 'leftMenuBtns': leftMenuBtns, 57 | 'UserDisplay': UserDisplay, 58 | 'todo': todo 59 | } 60 | -------------------------------------------------------------------------------- /sql/sendmail.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | import traceback 3 | from multiprocessing import Process 4 | import email 5 | from email import encoders 6 | from email.header import Header 7 | from email.mime.text import MIMEText 8 | from email.utils import parseaddr, formataddr 9 | import smtplib 10 | 11 | from django.conf import settings 12 | import logging 13 | 14 | logger = logging.getLogger('default') 15 | 16 | 17 | class MailSender(object): 18 | 19 | def __init__(self): 20 | try: 21 | self.MAIL_REVIEW_SMTP_SERVER = getattr(settings, 'MAIL_REVIEW_SMTP_SERVER') 22 | self.MAIL_REVIEW_SMTP_PORT = int(getattr(settings, 'MAIL_REVIEW_SMTP_PORT')) 23 | self.MAIL_REVIEW_FROM_ADDR = getattr(settings, 'MAIL_REVIEW_FROM_ADDR') 24 | self.MAIL_REVIEW_FROM_PASSWORD = getattr(settings, 'MAIL_REVIEW_FROM_PASSWORD') 25 | self.SSL = getattr(settings, 'MAIL_SSL') 26 | 27 | except AttributeError as a: 28 | print("Error: %s" % a) 29 | except ValueError as v: 30 | print("Error: %s" % v) 31 | 32 | def _format_addr(self, s): 33 | name, addr = parseaddr(s) 34 | return formataddr((Header(name, 'utf-8').encode(), addr)) 35 | 36 | def _add_attachment(self, filename): 37 | ''''' 38 | 添加附件 39 | ''' 40 | file_msg = email.mime.base.MIMEBase('application', 'octet-stream') 41 | file_msg.set_payload(open(filename, 'rb').read()) 42 | # 附件如果有中文会出现乱码问题,加入gbk 43 | file_msg.add_header('Content-Disposition', 'attachment', filename=('gbk', '', filename.split('/')[-1])) 44 | encoders.encode_base64(file_msg) 45 | 46 | return file_msg 47 | 48 | def _send(self, strTitle, strContent, listToAddr, **kwargs): 49 | ''''' 50 | 发送邮件 51 | ''' 52 | # 构造MIMEMultipart对象做为根容器 53 | main_msg = email.mime.multipart.MIMEMultipart() 54 | 55 | # 添加文本内容 56 | text_msg = email.mime.text.MIMEText(strContent, 'plain', 'utf-8') 57 | main_msg.attach(text_msg) 58 | 59 | # 添加附件 60 | filename_list = kwargs.get('filename_list') 61 | if filename_list: 62 | for filename in kwargs['filename_list']: 63 | file_msg = self._add_attachment(filename) 64 | main_msg.attach(file_msg) 65 | 66 | # 收发件人地址和邮件标题: 67 | main_msg['From'] = formataddr(["archer 通知", self.MAIL_REVIEW_FROM_ADDR]) 68 | main_msg['To'] = ','.join(listToAddr) 69 | listCcAddr = kwargs.get('listCcAddr') 70 | if listCcAddr: 71 | main_msg['Cc'] = ', '.join(kwargs['listCcAddr']) 72 | listAddr = listToAddr + listCcAddr 73 | else: 74 | listAddr = listToAddr 75 | main_msg['Subject'] = Header(strTitle, "utf-8").encode() 76 | main_msg['Date'] = email.utils.formatdate() 77 | 78 | if self.SSL: 79 | server = smtplib.SMTP_SSL(self.MAIL_REVIEW_SMTP_SERVER, self.MAIL_REVIEW_SMTP_PORT) # SMTP协议默认SSL端口是465 80 | else: 81 | server = smtplib.SMTP(self.MAIL_REVIEW_SMTP_SERVER, self.MAIL_REVIEW_SMTP_PORT) # SMTP协议默认端口是25 82 | # server.set_debuglevel(1) 83 | 84 | # 如果提供的密码为空,则不需要登录SMTP server 85 | if self.MAIL_REVIEW_FROM_PASSWORD != '': 86 | server.login(self.MAIL_REVIEW_FROM_ADDR, self.MAIL_REVIEW_FROM_PASSWORD) 87 | sendResult = server.sendmail(self.MAIL_REVIEW_FROM_ADDR, listAddr, main_msg.as_string()) 88 | server.quit() 89 | 90 | # 调用方应该调用此方法,采用子进程方式异步阻塞地发送邮件,避免邮件服务挂掉影响archer主服务 91 | def sendEmail(self, strTitle, strContent, listToAddr, **kwargs): 92 | try: 93 | p = Process(target=self._send, args=(strTitle, strContent, listToAddr), kwargs=kwargs) 94 | p.start() 95 | except Exception: 96 | logger.error(traceback.format_exc()) 97 | -------------------------------------------------------------------------------- /sql/sqlreview.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | import simplejson as json 3 | 4 | import time 5 | from threading import Thread 6 | 7 | from django.db import connection 8 | from django.utils import timezone 9 | from django.conf import settings 10 | 11 | from .dao import Dao 12 | from .const import Const, WorkflowDict 13 | from .sendmail import MailSender 14 | from .inception import InceptionDao 15 | from .aes_decryptor import Prpcrypt 16 | from .models import users, workflow, master_config 17 | from .workflow import Workflow 18 | from .permission import role_required, superuser_required 19 | import logging 20 | 21 | logger = logging.getLogger('default') 22 | 23 | dao = Dao() 24 | inceptionDao = InceptionDao() 25 | mailSender = MailSender() 26 | prpCryptor = Prpcrypt() 27 | workflowOb = Workflow() 28 | 29 | 30 | # 获取当前请求url 31 | def getDetailUrl(request): 32 | scheme = request.scheme 33 | host = request.META['HTTP_HOST'] 34 | return "%s://%s/detail/" % (scheme, host) 35 | 36 | 37 | # 根据实例名获取主库连接字符串,并封装成一个dict 38 | def getMasterConnStr(clusterName): 39 | listMasters = master_config.objects.filter(cluster_name=clusterName) 40 | 41 | masterHost = listMasters[0].master_host 42 | masterPort = listMasters[0].master_port 43 | masterUser = listMasters[0].master_user 44 | masterPassword = prpCryptor.decrypt(listMasters[0].master_password) 45 | dictConn = {'masterHost': masterHost, 'masterPort': masterPort, 'masterUser': masterUser, 46 | 'masterPassword': masterPassword} 47 | return dictConn 48 | 49 | 50 | # SQL工单执行回调 51 | def execute_call_back(workflowId, clusterName, url): 52 | workflowDetail = workflow.objects.get(id=workflowId) 53 | # 获取审核人 54 | try: 55 | listAllReviewMen = json.loads(workflowDetail.review_man) 56 | except ValueError: 57 | listAllReviewMen = (workflowDetail.review_man,) 58 | 59 | dictConn = getMasterConnStr(clusterName) 60 | try: 61 | # 交给inception先split,再执行 62 | logger.debug('execute_call_back:' + str(workflowId) + ' executing') 63 | (finalStatus, finalList) = inceptionDao.executeFinal(workflowDetail, dictConn) 64 | 65 | # 封装成JSON格式存进数据库字段里 66 | strJsonResult = json.dumps(finalList) 67 | workflowDetail = workflow.objects.get(id=workflowId) 68 | workflowDetail.execute_result = strJsonResult 69 | workflowDetail.finish_time = timezone.now() 70 | workflowDetail.status = finalStatus 71 | workflowDetail.is_manual = 0 72 | workflowDetail.audit_remark = '' 73 | # 重新获取连接,防止超时 74 | connection.close() 75 | workflowDetail.save() 76 | logger.debug('execute_call_back:' + str(workflowId) + ' finish') 77 | except Exception as e: 78 | logger.error(e) 79 | 80 | # 如果执行完毕了,则根据settings.py里的配置决定是否给提交者和DBA一封邮件提醒.DBA需要知晓审核并执行过的单子 81 | if hasattr(settings, 'MAIL_ON_OFF') == True: 82 | if getattr(settings, 'MAIL_ON_OFF') == "on": 83 | # 给主、副审核人,申请人,DBA各发一封邮件 84 | engineer = workflowDetail.engineer 85 | reviewMen = workflowDetail.review_man 86 | workflowStatus = workflowDetail.status 87 | workflowName = workflowDetail.workflow_name 88 | objEngineer = users.objects.get(username=engineer) 89 | strTitle = "SQL上线工单执行完毕 # " + str(workflowId) 90 | strContent = "发起人:" + engineer + "\n审核人:" + reviewMen + "\n工单地址:" + url + "\n工单名称: " + workflowName + "\n执行结果:" + workflowStatus 91 | reviewManAddr = [email['email'] for email in 92 | users.objects.filter(username__in=listAllReviewMen).values('email')] 93 | dbaAddr = [email['email'] for email in users.objects.filter(role='DBA').values('email')] 94 | listCcAddr = reviewManAddr + dbaAddr 95 | mailSender.sendEmail(strTitle, strContent, [objEngineer.email], listCcAddr=listCcAddr) 96 | 97 | 98 | # 给定时任务执行sql 99 | def execute_job(workflowId, url): 100 | job_id = Const.workflowJobprefix['sqlreview'] + '-' + str(workflowId) 101 | logger.debug('execute_job:' + job_id + ' start') 102 | workflowDetail = workflow.objects.get(id=workflowId) 103 | clusterName = workflowDetail.cluster_name 104 | 105 | # 服务器端二次验证,当前工单状态必须为定时执行过状态 106 | if workflowDetail.status != Const.workflowStatus['timingtask']: 107 | raise Exception('工单不是定时执行状态') 108 | 109 | # 将流程状态修改为执行中,并更新reviewok_time字段 110 | workflowDetail.status = Const.workflowStatus['executing'] 111 | workflowDetail.reviewok_time = timezone.now() 112 | try: 113 | workflowDetail.save() 114 | except Exception: 115 | # 关闭后重新获取连接,防止超时 116 | connection.close() 117 | workflowDetail.save() 118 | logger.debug('execute_job:' + job_id + ' executing') 119 | # 执行之前重新split并check一遍,更新SHA1缓存;因为如果在执行中,其他进程去做这一步操作的话,会导致inception core dump挂掉 120 | splitReviewResult = inceptionDao.sqlautoReview(workflowDetail.sql_content, workflowDetail.cluster_name, 121 | isSplit='yes') 122 | workflowDetail.review_content = json.dumps(splitReviewResult) 123 | try: 124 | workflowDetail.save() 125 | except Exception: 126 | # 关闭后重新获取连接,防止超时 127 | connection.close() 128 | workflowDetail.save() 129 | 130 | # 采取异步回调的方式执行语句,防止出现持续执行中的异常 131 | t = Thread(target=execute_call_back, args=(workflowId, clusterName, url)) 132 | t.start() 133 | -------------------------------------------------------------------------------- /sql/static/ace/snippets/sql.js: -------------------------------------------------------------------------------- 1 | define("ace/snippets/sql",["require","exports","module"], function(require, exports, module) { 2 | "use strict"; 3 | 4 | exports.snippetText = "snippet tbl\n\ 5 | create table ${1:table} (\n\ 6 | ${2:columns}\n\ 7 | );\n\ 8 | snippet col\n\ 9 | ${1:name} ${2:type} ${3:default ''} ${4:not null}\n\ 10 | snippet ccol\n\ 11 | ${1:name} varchar2(${2:size}) ${3:default ''} ${4:not null}\n\ 12 | snippet ncol\n\ 13 | ${1:name} number ${3:default 0} ${4:not null}\n\ 14 | snippet dcol\n\ 15 | ${1:name} date ${3:default sysdate} ${4:not null}\n\ 16 | snippet ind\n\ 17 | create index ${3:$1_$2} on ${1:table}(${2:column});\n\ 18 | snippet uind\n\ 19 | create unique index ${1:name} on ${2:table}(${3:column});\n\ 20 | snippet tblcom\n\ 21 | comment on table ${1:table} is '${2:comment}';\n\ 22 | snippet colcom\n\ 23 | comment on column ${1:table}.${2:column} is '${3:comment}';\n\ 24 | snippet addcol\n\ 25 | alter table ${1:table} add (${2:column} ${3:type});\n\ 26 | snippet seq\n\ 27 | create sequence ${1:name} start with ${2:1} increment by ${3:1} minvalue ${4:1};\n\ 28 | snippet s*\n\ 29 | select * from ${1:table}\n\ 30 | "; 31 | exports.scope = "sql"; 32 | 33 | }); 34 | -------------------------------------------------------------------------------- /sql/static/ace/theme-github.js: -------------------------------------------------------------------------------- 1 | define("ace/theme/github",["require","exports","module","ace/lib/dom"], function(require, exports, module) { 2 | 3 | exports.isDark = false; 4 | exports.cssClass = "ace-github"; 5 | exports.cssText = "\ 6 | .ace-github .ace_gutter {\ 7 | background: #e8e8e8;\ 8 | color: #AAA;\ 9 | }\ 10 | .ace-github {\ 11 | background: #fff;\ 12 | color: #000;\ 13 | }\ 14 | .ace-github .ace_keyword {\ 15 | font-weight: bold;\ 16 | }\ 17 | .ace-github .ace_string {\ 18 | color: #D14;\ 19 | }\ 20 | .ace-github .ace_variable.ace_class {\ 21 | color: teal;\ 22 | }\ 23 | .ace-github .ace_constant.ace_numeric {\ 24 | color: #099;\ 25 | }\ 26 | .ace-github .ace_constant.ace_buildin {\ 27 | color: #0086B3;\ 28 | }\ 29 | .ace-github .ace_support.ace_function {\ 30 | color: #0086B3;\ 31 | }\ 32 | .ace-github .ace_comment {\ 33 | color: #998;\ 34 | font-style: italic;\ 35 | }\ 36 | .ace-github .ace_variable.ace_language {\ 37 | color: #0086B3;\ 38 | }\ 39 | .ace-github .ace_paren {\ 40 | font-weight: bold;\ 41 | }\ 42 | .ace-github .ace_boolean {\ 43 | font-weight: bold;\ 44 | }\ 45 | .ace-github .ace_string.ace_regexp {\ 46 | color: #009926;\ 47 | font-weight: normal;\ 48 | }\ 49 | .ace-github .ace_variable.ace_instance {\ 50 | color: teal;\ 51 | }\ 52 | .ace-github .ace_constant.ace_language {\ 53 | font-weight: bold;\ 54 | }\ 55 | .ace-github .ace_cursor {\ 56 | color: black;\ 57 | }\ 58 | .ace-github.ace_focus .ace_marker-layer .ace_active-line {\ 59 | background: rgb(255, 255, 204);\ 60 | }\ 61 | .ace-github .ace_marker-layer .ace_active-line {\ 62 | background: rgb(245, 245, 245);\ 63 | }\ 64 | .ace-github .ace_marker-layer .ace_selection {\ 65 | background: rgb(181, 213, 255);\ 66 | }\ 67 | .ace-github.ace_multiselect .ace_selection.ace_start {\ 68 | box-shadow: 0 0 3px 0px white;\ 69 | }\ 70 | .ace-github.ace_nobold .ace_line > span {\ 71 | font-weight: normal !important;\ 72 | }\ 73 | .ace-github .ace_marker-layer .ace_step {\ 74 | background: rgb(252, 255, 0);\ 75 | }\ 76 | .ace-github .ace_marker-layer .ace_stack {\ 77 | background: rgb(164, 229, 101);\ 78 | }\ 79 | .ace-github .ace_marker-layer .ace_bracket {\ 80 | margin: -1px 0 0 -1px;\ 81 | border: 1px solid rgb(192, 192, 192);\ 82 | }\ 83 | .ace-github .ace_gutter-active-line {\ 84 | background-color : rgba(0, 0, 0, 0.07);\ 85 | }\ 86 | .ace-github .ace_marker-layer .ace_selected-word {\ 87 | background: rgb(250, 250, 255);\ 88 | border: 1px solid rgb(200, 200, 250);\ 89 | }\ 90 | .ace-github .ace_invisible {\ 91 | color: #BFBFBF\ 92 | }\ 93 | .ace-github .ace_print-margin {\ 94 | width: 1px;\ 95 | background: #e8e8e8;\ 96 | }\ 97 | .ace-github .ace_indent-guide {\ 98 | background: url(\"\") right repeat-y;\ 99 | }"; 100 | 101 | var dom = require("../lib/dom"); 102 | dom.importCssString(exports.cssText, exports.cssClass); 103 | }); 104 | -------------------------------------------------------------------------------- /sql/static/ace/theme-textmate.js: -------------------------------------------------------------------------------- 1 | define("ace/theme/textmate",["require","exports","module","ace/lib/dom"], function(require, exports, module) { 2 | "use strict"; 3 | 4 | exports.isDark = false; 5 | exports.cssClass = "ace-tm"; 6 | exports.cssText = ".ace-tm .ace_gutter {\ 7 | background: #f0f0f0;\ 8 | color: #333;\ 9 | }\ 10 | .ace-tm .ace_print-margin {\ 11 | width: 1px;\ 12 | background: #e8e8e8;\ 13 | }\ 14 | .ace-tm .ace_fold {\ 15 | background-color: #6B72E6;\ 16 | }\ 17 | .ace-tm {\ 18 | background-color: #FFFFFF;\ 19 | color: black;\ 20 | }\ 21 | .ace-tm .ace_cursor {\ 22 | color: black;\ 23 | }\ 24 | .ace-tm .ace_invisible {\ 25 | color: rgb(191, 191, 191);\ 26 | }\ 27 | .ace-tm .ace_storage,\ 28 | .ace-tm .ace_keyword {\ 29 | color: blue;\ 30 | }\ 31 | .ace-tm .ace_constant {\ 32 | color: rgb(197, 6, 11);\ 33 | }\ 34 | .ace-tm .ace_constant.ace_buildin {\ 35 | color: rgb(88, 72, 246);\ 36 | }\ 37 | .ace-tm .ace_constant.ace_language {\ 38 | color: rgb(88, 92, 246);\ 39 | }\ 40 | .ace-tm .ace_constant.ace_library {\ 41 | color: rgb(6, 150, 14);\ 42 | }\ 43 | .ace-tm .ace_invalid {\ 44 | background-color: rgba(255, 0, 0, 0.1);\ 45 | color: red;\ 46 | }\ 47 | .ace-tm .ace_support.ace_function {\ 48 | color: rgb(60, 76, 114);\ 49 | }\ 50 | .ace-tm .ace_support.ace_constant {\ 51 | color: rgb(6, 150, 14);\ 52 | }\ 53 | .ace-tm .ace_support.ace_type,\ 54 | .ace-tm .ace_support.ace_class {\ 55 | color: rgb(109, 121, 222);\ 56 | }\ 57 | .ace-tm .ace_keyword.ace_operator {\ 58 | color: rgb(104, 118, 135);\ 59 | }\ 60 | .ace-tm .ace_string {\ 61 | color: rgb(3, 106, 7);\ 62 | }\ 63 | .ace-tm .ace_comment {\ 64 | color: rgb(76, 136, 107);\ 65 | }\ 66 | .ace-tm .ace_comment.ace_doc {\ 67 | color: rgb(0, 102, 255);\ 68 | }\ 69 | .ace-tm .ace_comment.ace_doc.ace_tag {\ 70 | color: rgb(128, 159, 191);\ 71 | }\ 72 | .ace-tm .ace_constant.ace_numeric {\ 73 | color: rgb(0, 0, 205);\ 74 | }\ 75 | .ace-tm .ace_variable {\ 76 | color: rgb(49, 132, 149);\ 77 | }\ 78 | .ace-tm .ace_xml-pe {\ 79 | color: rgb(104, 104, 91);\ 80 | }\ 81 | .ace-tm .ace_entity.ace_name.ace_function {\ 82 | color: #0000A2;\ 83 | }\ 84 | .ace-tm .ace_heading {\ 85 | color: rgb(12, 7, 255);\ 86 | }\ 87 | .ace-tm .ace_list {\ 88 | color:rgb(185, 6, 144);\ 89 | }\ 90 | .ace-tm .ace_meta.ace_tag {\ 91 | color:rgb(0, 22, 142);\ 92 | }\ 93 | .ace-tm .ace_string.ace_regex {\ 94 | color: rgb(255, 0, 0)\ 95 | }\ 96 | .ace-tm .ace_marker-layer .ace_selection {\ 97 | background: rgb(181, 213, 255);\ 98 | }\ 99 | .ace-tm.ace_multiselect .ace_selection.ace_start {\ 100 | box-shadow: 0 0 3px 0px white;\ 101 | }\ 102 | .ace-tm .ace_marker-layer .ace_step {\ 103 | background: rgb(252, 255, 0);\ 104 | }\ 105 | .ace-tm .ace_marker-layer .ace_stack {\ 106 | background: rgb(164, 229, 101);\ 107 | }\ 108 | .ace-tm .ace_marker-layer .ace_bracket {\ 109 | margin: -1px 0 0 -1px;\ 110 | border: 1px solid rgb(192, 192, 192);\ 111 | }\ 112 | .ace-tm .ace_marker-layer .ace_active-line {\ 113 | background: rgba(0, 0, 0, 0.07);\ 114 | }\ 115 | .ace-tm .ace_gutter-active-line {\ 116 | background-color : #dcdcdc;\ 117 | }\ 118 | .ace-tm .ace_marker-layer .ace_selected-word {\ 119 | background: rgb(250, 250, 255);\ 120 | border: 1px solid rgb(200, 200, 250);\ 121 | }\ 122 | .ace-tm .ace_indent-guide {\ 123 | background: url(\"\") right repeat-y;\ 124 | }\ 125 | "; 126 | 127 | var dom = require("../lib/dom"); 128 | dom.importCssString(exports.cssText, exports.cssClass); 129 | }); 130 | -------------------------------------------------------------------------------- /sql/static/admin/css/dashboard.css: -------------------------------------------------------------------------------- 1 | /* DASHBOARD */ 2 | 3 | .dashboard .module table th { 4 | width: 100%; 5 | } 6 | 7 | .dashboard .module table td { 8 | white-space: nowrap; 9 | } 10 | 11 | .dashboard .module table td a { 12 | display: block; 13 | padding-right: .6em; 14 | } 15 | 16 | /* RECENT ACTIONS MODULE */ 17 | 18 | .module ul.actionlist { 19 | margin-left: 0; 20 | } 21 | 22 | ul.actionlist li { 23 | list-style-type: none; 24 | } 25 | 26 | ul.actionlist li { 27 | overflow: hidden; 28 | text-overflow: ellipsis; 29 | -o-text-overflow: ellipsis; 30 | } 31 | -------------------------------------------------------------------------------- /sql/static/admin/css/ie.css: -------------------------------------------------------------------------------- 1 | /* IE 6 & 7 */ 2 | 3 | /* Proper fixed width for dashboard in IE6 */ 4 | 5 | .dashboard #content { 6 | *width: 768px; 7 | } 8 | 9 | .dashboard #content-main { 10 | *width: 535px; 11 | } 12 | 13 | /* IE 6 ONLY */ 14 | 15 | /* Keep header from flowing off the page */ 16 | 17 | #container { 18 | _position: static; 19 | } 20 | 21 | /* Put the right sidebars back on the page */ 22 | 23 | .colMS #content-related { 24 | _margin-right: 0; 25 | _margin-left: 10px; 26 | _position: static; 27 | } 28 | 29 | /* Put the left sidebars back on the page */ 30 | 31 | .colSM #content-related { 32 | _margin-right: 10px; 33 | _margin-left: -115px; 34 | _position: static; 35 | } 36 | 37 | .form-row { 38 | _height: 1%; 39 | } 40 | 41 | /* Fix right margin for changelist filters in IE6 */ 42 | 43 | #changelist-filter ul { 44 | _margin-right: -10px; 45 | } 46 | 47 | /* IE ignores min-height, but treats height as if it were min-height */ 48 | 49 | .change-list .filtered { 50 | _height: 400px; 51 | } 52 | 53 | /* IE doesn't know alpha transparency in PNGs */ 54 | 55 | .inline-deletelink { 56 | background: transparent url(../img/inline-delete-8bit.png) no-repeat; 57 | } 58 | 59 | /* IE7 doesn't support inline-block */ 60 | .change-list ul.toplinks li { 61 | zoom: 1; 62 | *display: inline; 63 | } 64 | -------------------------------------------------------------------------------- /sql/static/admin/css/login.css: -------------------------------------------------------------------------------- 1 | /* LOGIN FORM */ 2 | 3 | body.login { 4 | background: #eee; 5 | } 6 | 7 | .login #container { 8 | background: white; 9 | border: 1px solid #ccc; 10 | width: 28em; 11 | min-width: 300px; 12 | margin-left: auto; 13 | margin-right: auto; 14 | margin-top: 100px; 15 | } 16 | 17 | .login #content-main { 18 | width: 100%; 19 | } 20 | 21 | .login form { 22 | margin-top: 1em; 23 | } 24 | 25 | .login .form-row { 26 | padding: 4px 0; 27 | float: left; 28 | width: 100%; 29 | } 30 | 31 | .login .form-row label { 32 | padding-right: 0.5em; 33 | line-height: 2em; 34 | font-size: 1em; 35 | clear: both; 36 | color: #333; 37 | } 38 | 39 | .login .form-row #id_username, .login .form-row #id_password { 40 | clear: both; 41 | padding: 6px; 42 | width: 100%; 43 | -webkit-box-sizing: border-box; 44 | -moz-box-sizing: border-box; 45 | box-sizing: border-box; 46 | } 47 | 48 | .login span.help { 49 | font-size: 10px; 50 | display: block; 51 | } 52 | 53 | .login .submit-row { 54 | clear: both; 55 | padding: 1em 0 0 9.4em; 56 | } 57 | 58 | .login .password-reset-link { 59 | text-align: center; 60 | } 61 | -------------------------------------------------------------------------------- /sql/static/admin/css/rtl.css: -------------------------------------------------------------------------------- 1 | body { 2 | direction: rtl; 3 | } 4 | 5 | /* LOGIN */ 6 | 7 | .login .form-row { 8 | float: right; 9 | } 10 | 11 | .login .form-row label { 12 | float: right; 13 | padding-left: 0.5em; 14 | padding-right: 0; 15 | text-align: left; 16 | } 17 | 18 | .login .submit-row { 19 | clear: both; 20 | padding: 1em 9.4em 0 0; 21 | } 22 | 23 | /* GLOBAL */ 24 | 25 | th { 26 | text-align: right; 27 | } 28 | 29 | .module h2, .module caption { 30 | text-align: right; 31 | } 32 | 33 | .addlink, .changelink { 34 | padding-left: 0px; 35 | padding-right: 12px; 36 | background-position: 100% 0.2em; 37 | } 38 | 39 | .deletelink { 40 | padding-left: 0px; 41 | padding-right: 12px; 42 | background-position: 100% 0.25em; 43 | } 44 | 45 | .object-tools { 46 | float: left; 47 | } 48 | 49 | thead th:first-child, 50 | tfoot td:first-child { 51 | border-left: 1px solid #ddd !important; 52 | } 53 | 54 | /* LAYOUT */ 55 | 56 | #user-tools { 57 | right: auto; 58 | left: 0; 59 | text-align: left; 60 | } 61 | 62 | div.breadcrumbs { 63 | text-align: right; 64 | } 65 | 66 | #content-main { 67 | float: right; 68 | } 69 | 70 | #content-related { 71 | float: left; 72 | margin-left: -19em; 73 | margin-right: auto; 74 | } 75 | 76 | .colMS { 77 | margin-left: 20em !important; 78 | margin-right: 10px !important; 79 | } 80 | 81 | /* SORTABLE TABLES */ 82 | 83 | table thead th.sorted .sortoptions { 84 | float: left; 85 | } 86 | 87 | thead th.sorted .text { 88 | padding-right: 0; 89 | padding-left: 42px; 90 | } 91 | 92 | /* dashboard styles */ 93 | 94 | .dashboard .module table td a { 95 | padding-left: .6em; 96 | padding-right: 12px; 97 | } 98 | 99 | /* changelists styles */ 100 | 101 | .change-list .filtered { 102 | background: white url(../img/changelist-bg_rtl.gif) top left repeat-y !important; 103 | } 104 | 105 | .change-list .filtered table { 106 | border-left: 1px solid #ddd; 107 | border-right: 0px none; 108 | } 109 | 110 | #changelist-filter { 111 | right: auto; 112 | left: 0; 113 | border-left: 0px none; 114 | border-right: 1px solid #ddd; 115 | } 116 | 117 | .change-list .filtered .results, .change-list .filtered .paginator, .filtered #toolbar, .filtered div.xfull { 118 | margin-right: 0px !important; 119 | margin-left: 160px !important; 120 | } 121 | 122 | #changelist-filter li.selected { 123 | border-left: 0px none; 124 | padding-left: 0px; 125 | margin-left: 0; 126 | border-right: 5px solid #ccc; 127 | padding-right: 5px; 128 | margin-right: -10px; 129 | } 130 | 131 | .filtered .actions { 132 | border-left:1px solid #DDDDDD; 133 | margin-left:160px !important; 134 | border-right: 0 none; 135 | margin-right:0 !important; 136 | } 137 | 138 | #changelist table tbody td:first-child, #changelist table tbody th:first-child { 139 | border-right: 0; 140 | border-left: 1px solid #ddd; 141 | } 142 | 143 | /* FORMS */ 144 | 145 | .aligned label { 146 | padding: 0 0 3px 1em; 147 | float: right; 148 | } 149 | 150 | .submit-row { 151 | text-align: left 152 | } 153 | 154 | .submit-row p.deletelink-box { 155 | float: right; 156 | } 157 | 158 | .submit-row .deletelink { 159 | background: url(../img/icon_deletelink.gif) 0 50% no-repeat; 160 | padding-right: 14px; 161 | } 162 | 163 | .vDateField, .vTimeField { 164 | margin-left: 2px; 165 | } 166 | 167 | form ul.inline li { 168 | float: right; 169 | padding-right: 0; 170 | padding-left: 7px; 171 | } 172 | 173 | input[type=submit].default, .submit-row input.default { 174 | float: left; 175 | } 176 | 177 | fieldset .field-box { 178 | float: right; 179 | margin-left: 20px; 180 | margin-right: 0; 181 | } 182 | 183 | .errorlist li { 184 | background-position: 100% .3em; 185 | padding: 4px 25px 4px 5px; 186 | } 187 | 188 | .errornote { 189 | background-position: 100% .3em; 190 | padding: 4px 25px 4px 5px; 191 | } 192 | 193 | /* WIDGETS */ 194 | 195 | .calendarnav-previous { 196 | top: 0; 197 | left: auto; 198 | right: 0; 199 | } 200 | 201 | .calendarnav-next { 202 | top: 0; 203 | right: auto; 204 | left: 0; 205 | } 206 | 207 | .calendar caption, .calendarbox h2 { 208 | text-align: center; 209 | } 210 | 211 | .selector { 212 | float: right; 213 | } 214 | 215 | .selector .selector-filter { 216 | text-align: right; 217 | } 218 | 219 | .inline-deletelink { 220 | float: left; 221 | } 222 | 223 | /* MISC */ 224 | 225 | .inline-related h2, .inline-group h2 { 226 | text-align: right 227 | } 228 | 229 | .inline-related h3 span.delete { 230 | padding-right: 20px; 231 | padding-left: inherit; 232 | left: 10px; 233 | right: inherit; 234 | float:left; 235 | } 236 | 237 | .inline-related h3 span.delete label { 238 | margin-left: inherit; 239 | margin-right: 2px; 240 | } 241 | 242 | /* IE7 specific bug fixes */ 243 | 244 | div.colM { 245 | position: relative; 246 | } 247 | 248 | .submit-row input { 249 | float: left; 250 | } 251 | -------------------------------------------------------------------------------- /sql/static/admin/img/changelist-bg.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/changelist-bg.gif -------------------------------------------------------------------------------- /sql/static/admin/img/changelist-bg_rtl.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/changelist-bg_rtl.gif -------------------------------------------------------------------------------- /sql/static/admin/img/default-bg-reverse.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/default-bg-reverse.gif -------------------------------------------------------------------------------- /sql/static/admin/img/default-bg.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/default-bg.gif -------------------------------------------------------------------------------- /sql/static/admin/img/deleted-overlay.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/deleted-overlay.gif -------------------------------------------------------------------------------- /sql/static/admin/img/gis/move_vertex_off.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/gis/move_vertex_off.png -------------------------------------------------------------------------------- /sql/static/admin/img/gis/move_vertex_on.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/gis/move_vertex_on.png -------------------------------------------------------------------------------- /sql/static/admin/img/icon-no.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon-no.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon-unknown.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon-unknown.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon-yes.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon-yes.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon_addlink.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon_addlink.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon_alert.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon_alert.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon_calendar.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon_calendar.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon_changelink.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon_changelink.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon_clock.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon_clock.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon_deletelink.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon_deletelink.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon_error.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon_error.gif -------------------------------------------------------------------------------- /sql/static/admin/img/icon_searchbox.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon_searchbox.png -------------------------------------------------------------------------------- /sql/static/admin/img/icon_success.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/icon_success.gif -------------------------------------------------------------------------------- /sql/static/admin/img/inline-delete-8bit.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/inline-delete-8bit.png -------------------------------------------------------------------------------- /sql/static/admin/img/inline-delete.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/inline-delete.png -------------------------------------------------------------------------------- /sql/static/admin/img/inline-restore-8bit.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/inline-restore-8bit.png -------------------------------------------------------------------------------- /sql/static/admin/img/inline-restore.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/inline-restore.png -------------------------------------------------------------------------------- /sql/static/admin/img/inline-splitter-bg.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/inline-splitter-bg.gif -------------------------------------------------------------------------------- /sql/static/admin/img/nav-bg-grabber.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/nav-bg-grabber.gif -------------------------------------------------------------------------------- /sql/static/admin/img/nav-bg-reverse.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/nav-bg-reverse.gif -------------------------------------------------------------------------------- /sql/static/admin/img/nav-bg-selected.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/nav-bg-selected.gif -------------------------------------------------------------------------------- /sql/static/admin/img/nav-bg.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/nav-bg.gif -------------------------------------------------------------------------------- /sql/static/admin/img/selector-icons.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/selector-icons.gif -------------------------------------------------------------------------------- /sql/static/admin/img/selector-search.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/selector-search.gif -------------------------------------------------------------------------------- /sql/static/admin/img/sorting-icons.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/sorting-icons.gif -------------------------------------------------------------------------------- /sql/static/admin/img/tooltag-add.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/tooltag-add.png -------------------------------------------------------------------------------- /sql/static/admin/img/tooltag-arrowright.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/admin/img/tooltag-arrowright.png -------------------------------------------------------------------------------- /sql/static/admin/js/LICENSE-JQUERY.txt: -------------------------------------------------------------------------------- 1 | Copyright (c) 2010 John Resig, http://jquery.com/ 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. -------------------------------------------------------------------------------- /sql/static/admin/js/SelectBox.js: -------------------------------------------------------------------------------- 1 | var SelectBox = { 2 | cache: new Object(), 3 | init: function(id) { 4 | var box = document.getElementById(id); 5 | var node; 6 | SelectBox.cache[id] = new Array(); 7 | var cache = SelectBox.cache[id]; 8 | for (var i = 0; (node = box.options[i]); i++) { 9 | cache.push({value: node.value, text: node.text, displayed: 1}); 10 | } 11 | }, 12 | redisplay: function(id) { 13 | // Repopulate HTML select box from cache 14 | var box = document.getElementById(id); 15 | box.options.length = 0; // clear all options 16 | for (var i = 0, j = SelectBox.cache[id].length; i < j; i++) { 17 | var node = SelectBox.cache[id][i]; 18 | if (node.displayed) { 19 | var new_option = new Option(node.text, node.value, false, false); 20 | // Shows a tooltip when hovering over the option 21 | new_option.setAttribute("title", node.text); 22 | box.options[box.options.length] = new_option; 23 | } 24 | } 25 | }, 26 | filter: function(id, text) { 27 | // Redisplay the HTML select box, displaying only the choices containing ALL 28 | // the words in text. (It's an AND search.) 29 | var tokens = text.toLowerCase().split(/\s+/); 30 | var node, token; 31 | for (var i = 0; (node = SelectBox.cache[id][i]); i++) { 32 | node.displayed = 1; 33 | for (var j = 0; (token = tokens[j]); j++) { 34 | if (node.text.toLowerCase().indexOf(token) == -1) { 35 | node.displayed = 0; 36 | } 37 | } 38 | } 39 | SelectBox.redisplay(id); 40 | }, 41 | delete_from_cache: function(id, value) { 42 | var node, delete_index = null; 43 | for (var i = 0; (node = SelectBox.cache[id][i]); i++) { 44 | if (node.value == value) { 45 | delete_index = i; 46 | break; 47 | } 48 | } 49 | var j = SelectBox.cache[id].length - 1; 50 | for (var i = delete_index; i < j; i++) { 51 | SelectBox.cache[id][i] = SelectBox.cache[id][i+1]; 52 | } 53 | SelectBox.cache[id].length--; 54 | }, 55 | add_to_cache: function(id, option) { 56 | SelectBox.cache[id].push({value: option.value, text: option.text, displayed: 1}); 57 | }, 58 | cache_contains: function(id, value) { 59 | // Check if an item is contained in the cache 60 | var node; 61 | for (var i = 0; (node = SelectBox.cache[id][i]); i++) { 62 | if (node.value == value) { 63 | return true; 64 | } 65 | } 66 | return false; 67 | }, 68 | move: function(from, to) { 69 | var from_box = document.getElementById(from); 70 | var to_box = document.getElementById(to); 71 | var option; 72 | for (var i = 0; (option = from_box.options[i]); i++) { 73 | if (option.selected && SelectBox.cache_contains(from, option.value)) { 74 | SelectBox.add_to_cache(to, {value: option.value, text: option.text, displayed: 1}); 75 | SelectBox.delete_from_cache(from, option.value); 76 | } 77 | } 78 | SelectBox.redisplay(from); 79 | SelectBox.redisplay(to); 80 | }, 81 | move_all: function(from, to) { 82 | var from_box = document.getElementById(from); 83 | var to_box = document.getElementById(to); 84 | var option; 85 | for (var i = 0; (option = from_box.options[i]); i++) { 86 | if (SelectBox.cache_contains(from, option.value)) { 87 | SelectBox.add_to_cache(to, {value: option.value, text: option.text, displayed: 1}); 88 | SelectBox.delete_from_cache(from, option.value); 89 | } 90 | } 91 | SelectBox.redisplay(from); 92 | SelectBox.redisplay(to); 93 | }, 94 | sort: function(id) { 95 | SelectBox.cache[id].sort( function(a, b) { 96 | a = a.text.toLowerCase(); 97 | b = b.text.toLowerCase(); 98 | try { 99 | if (a > b) return 1; 100 | if (a < b) return -1; 101 | } 102 | catch (e) { 103 | // silently fail on IE 'unknown' exception 104 | } 105 | return 0; 106 | } ); 107 | }, 108 | select_all: function(id) { 109 | var box = document.getElementById(id); 110 | for (var i = 0; i < box.options.length; i++) { 111 | box.options[i].selected = 'selected'; 112 | } 113 | } 114 | } 115 | -------------------------------------------------------------------------------- /sql/static/admin/js/actions.min.js: -------------------------------------------------------------------------------- 1 | (function(a){var f;a.fn.actions=function(q){var b=a.extend({},a.fn.actions.defaults,q),g=a(this),e=!1,m=function(c){c?k():l();a(g).prop("checked",c).parent().parent().toggleClass(b.selectedClass,c)},h=function(){var c=a(g).filter(":checked").length;a(b.counterContainer).html(interpolate(ngettext("%(sel)s of %(cnt)s selected","%(sel)s of %(cnt)s selected",c),{sel:c,cnt:_actions_icnt},!0));a(b.allToggle).prop("checked",function(){var a;c==g.length?(a=!0,k()):(a=!1,n());return a})},k=function(){a(b.acrossClears).hide(); 2 | a(b.acrossQuestions).show();a(b.allContainer).hide()},p=function(){a(b.acrossClears).show();a(b.acrossQuestions).hide();a(b.actionContainer).toggleClass(b.selectedClass);a(b.allContainer).show();a(b.counterContainer).hide()},l=function(){a(b.acrossClears).hide();a(b.acrossQuestions).hide();a(b.allContainer).hide();a(b.counterContainer).show()},n=function(){l();a(b.acrossInput).val(0);a(b.actionContainer).removeClass(b.selectedClass)};a(b.counterContainer).show();a(this).filter(":checked").each(function(c){a(this).parent().parent().toggleClass(b.selectedClass); 3 | h();1==a(b.acrossInput).val()&&p()});a(b.allToggle).show().click(function(){m(a(this).prop("checked"));h()});a("a",b.acrossQuestions).click(function(c){c.preventDefault();a(b.acrossInput).val(1);p()});a("a",b.acrossClears).click(function(c){c.preventDefault();a(b.allToggle).prop("checked",!1);n();m(0);h()});f=null;a(g).click(function(c){c||(c=window.event);var d=c.target?c.target:c.srcElement;if(f&&a.data(f)!=a.data(d)&&!0===c.shiftKey){var e=!1;a(f).prop("checked",d.checked).parent().parent().toggleClass(b.selectedClass, 4 | d.checked);a(g).each(function(){if(a.data(this)==a.data(f)||a.data(this)==a.data(d))e=e?!1:!0;e&&a(this).prop("checked",d.checked).parent().parent().toggleClass(b.selectedClass,d.checked)})}a(d).parent().parent().toggleClass(b.selectedClass,d.checked);f=d;h()});a("form#changelist-form table#result_list tr").find("td:gt(0) :input").change(function(){e=!0});a('form#changelist-form button[name="index"]').click(function(a){if(e)return confirm(gettext("You have unsaved changes on individual editable fields. If you run an action, your unsaved changes will be lost."))}); 5 | a('form#changelist-form input[name="_save"]').click(function(c){var d=!1;a("select option:selected",b.actionContainer).each(function(){a(this).val()&&(d=!0)});if(d)return e?confirm(gettext("You have selected an action, but you haven't saved your changes to individual fields yet. Please click OK to save. You'll need to re-run the action.")):confirm(gettext("You have selected an action, and you haven't made any changes on individual fields. You're probably looking for the Go button rather than the Save button."))})}; 6 | a.fn.actions.defaults={actionContainer:"div.actions",counterContainer:"span.action-counter",allContainer:"div.actions span.all",acrossInput:"div.actions input.select-across",acrossQuestions:"div.actions span.question",acrossClears:"div.actions span.clear",allToggle:"#action-toggle",selectedClass:"selected"}})(django.jQuery); 7 | -------------------------------------------------------------------------------- /sql/static/admin/js/admin/RelatedObjectLookups.js: -------------------------------------------------------------------------------- 1 | // Handles related-objects functionality: lookup link for raw_id_fields 2 | // and Add Another links. 3 | 4 | function html_unescape(text) { 5 | // Unescape a string that was escaped using django.utils.html.escape. 6 | text = text.replace(/</g, '<'); 7 | text = text.replace(/>/g, '>'); 8 | text = text.replace(/"/g, '"'); 9 | text = text.replace(/'/g, "'"); 10 | text = text.replace(/&/g, '&'); 11 | return text; 12 | } 13 | 14 | // IE doesn't accept periods or dashes in the window name, but the element IDs 15 | // we use to generate popup window names may contain them, therefore we map them 16 | // to allowed characters in a reversible way so that we can locate the correct 17 | // element when the popup window is dismissed. 18 | function id_to_windowname(text) { 19 | text = text.replace(/\./g, '__dot__'); 20 | text = text.replace(/\-/g, '__dash__'); 21 | return text; 22 | } 23 | 24 | function windowname_to_id(text) { 25 | text = text.replace(/__dot__/g, '.'); 26 | text = text.replace(/__dash__/g, '-'); 27 | return text; 28 | } 29 | 30 | function showAdminPopup(triggeringLink, name_regexp) { 31 | var name = triggeringLink.id.replace(name_regexp, ''); 32 | name = id_to_windowname(name); 33 | var href = triggeringLink.href; 34 | if (href.indexOf('?') == -1) { 35 | href += '?_popup=1'; 36 | } else { 37 | href += '&_popup=1'; 38 | } 39 | var win = window.open(href, name, 'height=500,width=800,resizable=yes,scrollbars=yes'); 40 | win.focus(); 41 | return false; 42 | } 43 | 44 | function showRelatedObjectLookupPopup(triggeringLink) { 45 | return showAdminPopup(triggeringLink, /^lookup_/); 46 | } 47 | 48 | function dismissRelatedLookupPopup(win, chosenId) { 49 | var name = windowname_to_id(win.name); 50 | var elem = document.getElementById(name); 51 | if (elem.className.indexOf('vManyToManyRawIdAdminField') != -1 && elem.value) { 52 | elem.value += ',' + chosenId; 53 | } else { 54 | document.getElementById(name).value = chosenId; 55 | } 56 | win.close(); 57 | } 58 | 59 | function showRelatedObjectPopup(triggeringLink) { 60 | var name = triggeringLink.id.replace(/^(change|add|delete)_/, ''); 61 | name = id_to_windowname(name); 62 | var href = triggeringLink.href; 63 | var win = window.open(href, name, 'height=500,width=800,resizable=yes,scrollbars=yes'); 64 | win.focus(); 65 | return false; 66 | } 67 | 68 | function dismissAddRelatedObjectPopup(win, newId, newRepr) { 69 | // newId and newRepr are expected to have previously been escaped by 70 | // django.utils.html.escape. 71 | newId = html_unescape(newId); 72 | newRepr = html_unescape(newRepr); 73 | var name = windowname_to_id(win.name); 74 | var elem = document.getElementById(name); 75 | var o; 76 | if (elem) { 77 | var elemName = elem.nodeName.toUpperCase(); 78 | if (elemName == 'SELECT') { 79 | o = new Option(newRepr, newId); 80 | elem.options[elem.options.length] = o; 81 | o.selected = true; 82 | } else if (elemName == 'INPUT') { 83 | if (elem.className.indexOf('vManyToManyRawIdAdminField') != -1 && elem.value) { 84 | elem.value += ',' + newId; 85 | } else { 86 | elem.value = newId; 87 | } 88 | } 89 | // Trigger a change event to update related links if required. 90 | django.jQuery(elem).trigger('change'); 91 | } else { 92 | var toId = name + "_to"; 93 | o = new Option(newRepr, newId); 94 | SelectBox.add_to_cache(toId, o); 95 | SelectBox.redisplay(toId); 96 | } 97 | win.close(); 98 | } 99 | 100 | function dismissChangeRelatedObjectPopup(win, objId, newRepr, newId) { 101 | objId = html_unescape(objId); 102 | newRepr = html_unescape(newRepr); 103 | var id = windowname_to_id(win.name).replace(/^edit_/, ''); 104 | var selectsSelector = interpolate('#%s, #%s_from, #%s_to', [id, id, id]); 105 | var selects = django.jQuery(selectsSelector); 106 | selects.find('option').each(function() { 107 | if (this.value == objId) { 108 | this.textContent = newRepr; 109 | this.value = newId; 110 | } 111 | }); 112 | win.close(); 113 | }; 114 | 115 | function dismissDeleteRelatedObjectPopup(win, objId) { 116 | objId = html_unescape(objId); 117 | var id = windowname_to_id(win.name).replace(/^delete_/, ''); 118 | var selectsSelector = interpolate('#%s, #%s_from, #%s_to', [id, id, id]); 119 | var selects = django.jQuery(selectsSelector); 120 | selects.find('option').each(function() { 121 | if (this.value == objId) { 122 | django.jQuery(this).remove(); 123 | } 124 | }).trigger('change'); 125 | win.close(); 126 | }; 127 | 128 | // Kept for backward compatibility 129 | showAddAnotherPopup = showRelatedObjectPopup; 130 | dismissAddAnotherPopup = dismissAddRelatedObjectPopup; 131 | 132 | django.jQuery(function($){ 133 | $(document).ready(function() { 134 | $('.related-lookup').click(function(e) { 135 | e.preventDefault(); 136 | showRelatedObjectLookupPopup(this); 137 | }); 138 | }); 139 | }); 140 | -------------------------------------------------------------------------------- /sql/static/admin/js/collapse.js: -------------------------------------------------------------------------------- 1 | (function($) { 2 | $(document).ready(function() { 3 | // Add anchor tag for Show/Hide link 4 | $("fieldset.collapse").each(function(i, elem) { 5 | // Don't hide if fields in this fieldset have errors 6 | if ($(elem).find("div.errors").length == 0) { 7 | $(elem).addClass("collapsed").find("h2").first().append(' (' + gettext("Show") + 9 | ')'); 10 | } 11 | }); 12 | // Add toggle to anchor tag 13 | $("fieldset.collapse a.collapse-toggle").click(function(ev) { 14 | if ($(this).closest("fieldset").hasClass("collapsed")) { 15 | // Show 16 | $(this).text(gettext("Hide")).closest("fieldset").removeClass("collapsed").trigger("show.fieldset", [$(this).attr("id")]); 17 | } else { 18 | // Hide 19 | $(this).text(gettext("Show")).closest("fieldset").addClass("collapsed").trigger("hide.fieldset", [$(this).attr("id")]); 20 | } 21 | return false; 22 | }); 23 | }); 24 | })(django.jQuery); 25 | -------------------------------------------------------------------------------- /sql/static/admin/js/collapse.min.js: -------------------------------------------------------------------------------- 1 | (function(a){a(document).ready(function(){a("fieldset.collapse").each(function(c,b){a(b).find("div.errors").length==0&&a(b).addClass("collapsed").find("h2").first().append(' ('+gettext("Show")+")")});a("fieldset.collapse a.collapse-toggle").click(function(){a(this).closest("fieldset").hasClass("collapsed")?a(this).text(gettext("Hide")).closest("fieldset").removeClass("collapsed").trigger("show.fieldset",[a(this).attr("id")]):a(this).text(gettext("Show")).closest("fieldset").addClass("collapsed").trigger("hide.fieldset", 2 | [a(this).attr("id")]);return false})})})(django.jQuery); 3 | -------------------------------------------------------------------------------- /sql/static/admin/js/inlines.min.js: -------------------------------------------------------------------------------- 1 | (function(a){a.fn.formset=function(g){var b=a.extend({},a.fn.formset.defaults,g),i=a(this);g=i.parent();var m=function(e,k,h){var j=RegExp("("+k+"-(\\d+|__prefix__))");k=k+"-"+h;a(e).prop("for")&&a(e).prop("for",a(e).prop("for").replace(j,k));if(e.id)e.id=e.id.replace(j,k);if(e.name)e.name=e.name.replace(j,k)},l=a("#id_"+b.prefix+"-TOTAL_FORMS").prop("autocomplete","off"),d=parseInt(l.val(),10),c=a("#id_"+b.prefix+"-MAX_NUM_FORMS").prop("autocomplete","off");l=c.val()===""||c.val()-l.val()>0;i.each(function(){a(this).not("."+ 2 | b.emptyCssClass).addClass(b.formCssClass)});if(i.length&&l){var f;if(i.prop("tagName")=="TR"){i=this.eq(-1).children().length;g.append(''+b.addText+"");f=g.find("tr:last a")}else{i.filter(":last").after('
'+b.addText+"
");f=i.filter(":last").next().find("a")}f.click(function(e){e.preventDefault();var k=a("#id_"+b.prefix+"-TOTAL_FORMS");e=a("#"+ 3 | b.prefix+"-empty");var h=e.clone(true);h.removeClass(b.emptyCssClass).addClass(b.formCssClass).attr("id",b.prefix+"-"+d);if(h.is("tr"))h.children(":last").append('
'+b.deleteText+"
");else h.is("ul")||h.is("ol")?h.append('
  • '+b.deleteText+"
  • "):h.children(":first").append(''+b.deleteText+""); 4 | h.find("*").each(function(){m(this,b.prefix,k.val())});h.insertBefore(a(e));a(k).val(parseInt(k.val(),10)+1);d+=1;c.val()!==""&&c.val()-k.val()<=0&&f.parent().hide();h.find("a."+b.deleteCssClass).click(function(j){j.preventDefault();j=a(this).parents("."+b.formCssClass);j.remove();d-=1;b.removed&&b.removed(j);j=a("."+b.formCssClass);a("#id_"+b.prefix+"-TOTAL_FORMS").val(j.length);if(c.val()===""||c.val()-j.length>0)f.parent().show();for(var n=0,o=j.length;n 0) { 23 | values.push(field.val()); 24 | } 25 | }); 26 | prepopulatedField.val(URLify(values.join(' '), maxLength)); 27 | }; 28 | 29 | prepopulatedField.data('_changed', false); 30 | prepopulatedField.change(function() { 31 | prepopulatedField.data('_changed', true); 32 | }); 33 | 34 | if (!prepopulatedField.val()) { 35 | $(dependencies.join(',')).keyup(populate).change(populate).focus(populate); 36 | } 37 | }); 38 | }; 39 | })(django.jQuery); 40 | -------------------------------------------------------------------------------- /sql/static/admin/js/prepopulate.min.js: -------------------------------------------------------------------------------- 1 | (function(b){b.fn.prepopulate=function(e,g){return this.each(function(){var a=b(this),d=function(){if(!a.data("_changed")){var f=[];b.each(e,function(h,c){c=b(c);c.val().length>0&&f.push(c.val())});a.val(URLify(f.join(" "),g))}};a.data("_changed",false);a.change(function(){a.data("_changed",true)});a.val()||b(e.join(",")).keyup(d).change(d).focus(d)})}})(django.jQuery); 2 | -------------------------------------------------------------------------------- /sql/static/admin/js/related-widget-wrapper.js: -------------------------------------------------------------------------------- 1 | django.jQuery(function($){ 2 | function updateLinks() { 3 | var $this = $(this); 4 | var siblings = $this.nextAll('.change-related, .delete-related'); 5 | if (!siblings.length) return; 6 | var value = $this.val(); 7 | if (value) { 8 | siblings.each(function(){ 9 | var elm = $(this); 10 | elm.attr('href', elm.attr('data-href-template').replace('__fk__', value)); 11 | }); 12 | } else siblings.removeAttr('href'); 13 | } 14 | var container = $(document); 15 | container.on('change', '.related-widget-wrapper select', updateLinks); 16 | container.find('.related-widget-wrapper select').each(updateLinks); 17 | container.on('click', '.related-widget-wrapper-link', function(event){ 18 | if (this.href) { 19 | showRelatedObjectPopup(this); 20 | } 21 | event.preventDefault(); 22 | }); 23 | }); 24 | -------------------------------------------------------------------------------- /sql/static/admin/js/timeparse.js: -------------------------------------------------------------------------------- 1 | var timeParsePatterns = [ 2 | // 9 3 | { re: /^\d{1,2}$/i, 4 | handler: function(bits) { 5 | if (bits[0].length == 1) { 6 | return '0' + bits[0] + ':00'; 7 | } else { 8 | return bits[0] + ':00'; 9 | } 10 | } 11 | }, 12 | // 13:00 13 | { re: /^\d{2}[:.]\d{2}$/i, 14 | handler: function(bits) { 15 | return bits[0].replace('.', ':'); 16 | } 17 | }, 18 | // 9:00 19 | { re: /^\d[:.]\d{2}$/i, 20 | handler: function(bits) { 21 | return '0' + bits[0].replace('.', ':'); 22 | } 23 | }, 24 | // 3 am / 3 a.m. / 3am 25 | { re: /^(\d+)\s*([ap])(?:.?m.?)?$/i, 26 | handler: function(bits) { 27 | var hour = parseInt(bits[1]); 28 | if (hour == 12) { 29 | hour = 0; 30 | } 31 | if (bits[2].toLowerCase() == 'p') { 32 | if (hour == 12) { 33 | hour = 0; 34 | } 35 | return (hour + 12) + ':00'; 36 | } else { 37 | if (hour < 10) { 38 | return '0' + hour + ':00'; 39 | } else { 40 | return hour + ':00'; 41 | } 42 | } 43 | } 44 | }, 45 | // 3.30 am / 3:15 a.m. / 3.00am 46 | { re: /^(\d+)[.:](\d{2})\s*([ap]).?m.?$/i, 47 | handler: function(bits) { 48 | var hour = parseInt(bits[1]); 49 | var mins = parseInt(bits[2]); 50 | if (mins < 10) { 51 | mins = '0' + mins; 52 | } 53 | if (hour == 12) { 54 | hour = 0; 55 | } 56 | if (bits[3].toLowerCase() == 'p') { 57 | if (hour == 12) { 58 | hour = 0; 59 | } 60 | return (hour + 12) + ':' + mins; 61 | } else { 62 | if (hour < 10) { 63 | return '0' + hour + ':' + mins; 64 | } else { 65 | return hour + ':' + mins; 66 | } 67 | } 68 | } 69 | }, 70 | // noon 71 | { re: /^no/i, 72 | handler: function(bits) { 73 | return '12:00'; 74 | } 75 | }, 76 | // midnight 77 | { re: /^mid/i, 78 | handler: function(bits) { 79 | return '00:00'; 80 | } 81 | } 82 | ]; 83 | 84 | function parseTimeString(s) { 85 | for (var i = 0; i < timeParsePatterns.length; i++) { 86 | var re = timeParsePatterns[i].re; 87 | var handler = timeParsePatterns[i].handler; 88 | var bits = re.exec(s); 89 | if (bits) { 90 | return handler(bits); 91 | } 92 | } 93 | return s; 94 | } 95 | -------------------------------------------------------------------------------- /sql/static/bootstrap-3.3.7-dist/css/dashboard.css: -------------------------------------------------------------------------------- 1 | /* 2 | * Base structure 3 | */ 4 | 5 | /* Move down content because we have a fixed navbar that is 50px tall */ 6 | body { 7 | padding-top: 50px; 8 | } 9 | 10 | 11 | /* 12 | * Global add-ons 13 | */ 14 | 15 | .sub-header { 16 | padding-bottom: 10px; 17 | border-bottom: 1px solid #eee; 18 | } 19 | 20 | /* 21 | * Top navigation 22 | * Hide default border to remove 1px line. 23 | */ 24 | .navbar-fixed-top { 25 | border: 0; 26 | } 27 | 28 | /* 29 | * Sidebar 30 | */ 31 | 32 | /* Hide for mobile, show later */ 33 | .sidebar { 34 | display: none; 35 | } 36 | @media (min-width: 768px) { 37 | .sidebar { 38 | position: fixed; 39 | top: 51px; 40 | bottom: 0; 41 | left: 0; 42 | z-index: 1000; 43 | display: block; 44 | padding: 20px; 45 | overflow-x: hidden; 46 | overflow-y: auto; /* Scrollable contents if viewport is shorter than content. */ 47 | background-color: #f5f5f5; 48 | border-right: 1px solid #eee; 49 | } 50 | } 51 | 52 | /* Sidebar navigation */ 53 | .nav-sidebar { 54 | margin-right: -21px; /* 20px padding + 1px border */ 55 | margin-bottom: 20px; 56 | margin-left: -20px; 57 | } 58 | .nav-sidebar > li > a { 59 | padding-right: 20px; 60 | padding-left: 20px; 61 | } 62 | .nav-sidebar > .active > a, 63 | .nav-sidebar > .active > a:hover, 64 | .nav-sidebar > .active > a:focus { 65 | color: #fff; 66 | background-color: #428bca; 67 | } 68 | 69 | 70 | /* 71 | * Main content 72 | */ 73 | 74 | .main { 75 | padding: 20px; 76 | } 77 | @media (min-width: 768px) { 78 | .main { 79 | padding-right: 40px; 80 | padding-left: 40px; 81 | } 82 | } 83 | .main .page-header { 84 | margin-top: 0; 85 | } 86 | 87 | 88 | /* 89 | * Placeholder dashboard ideas 90 | */ 91 | 92 | .placeholders { 93 | margin-bottom: 30px; 94 | text-align: center; 95 | } 96 | .placeholders h4 { 97 | margin-bottom: 0; 98 | } 99 | .placeholder { 100 | margin-bottom: 20px; 101 | } 102 | .placeholder img { 103 | display: inline-block; 104 | border-radius: 50%; 105 | } 106 | -------------------------------------------------------------------------------- /sql/static/bootstrap-3.3.7-dist/fonts/glyphicons-halflings-regular.eot: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/bootstrap-3.3.7-dist/fonts/glyphicons-halflings-regular.eot -------------------------------------------------------------------------------- /sql/static/bootstrap-3.3.7-dist/fonts/glyphicons-halflings-regular.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/bootstrap-3.3.7-dist/fonts/glyphicons-halflings-regular.ttf -------------------------------------------------------------------------------- /sql/static/bootstrap-3.3.7-dist/fonts/glyphicons-halflings-regular.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/bootstrap-3.3.7-dist/fonts/glyphicons-halflings-regular.woff -------------------------------------------------------------------------------- /sql/static/bootstrap-3.3.7-dist/fonts/glyphicons-halflings-regular.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/bootstrap-3.3.7-dist/fonts/glyphicons-halflings-regular.woff2 -------------------------------------------------------------------------------- /sql/static/bootstrap-3.3.7-dist/js/npm.js: -------------------------------------------------------------------------------- 1 | // This file is autogenerated via the `commonjs` Grunt task. You can require() this file in a CommonJS environment. 2 | require('../../js/transition.js') 3 | require('../../js/alert.js') 4 | require('../../js/button.js') 5 | require('../../js/carousel.js') 6 | require('../../js/collapse.js') 7 | require('../../js/dropdown.js') 8 | require('../../js/modal.js') 9 | require('../../js/tooltip.js') 10 | require('../../js/popover.js') 11 | require('../../js/scrollspy.js') 12 | require('../../js/tab.js') 13 | require('../../js/affix.js') -------------------------------------------------------------------------------- /sql/static/bootstrap-table/css/bootstrap-table.min.css: -------------------------------------------------------------------------------- 1 | .fixed-table-container .bs-checkbox,.fixed-table-container .no-records-found{text-align:center}.fixed-table-body thead th .th-inner,.table td,.table th{box-sizing:border-box}.bootstrap-table .table{margin-bottom:0!important;border-bottom:1px solid #ddd;border-collapse:collapse!important;border-radius:1px}.bootstrap-table .table:not(.table-condensed),.bootstrap-table .table:not(.table-condensed)>tbody>tr>td,.bootstrap-table .table:not(.table-condensed)>tbody>tr>th,.bootstrap-table .table:not(.table-condensed)>tfoot>tr>td,.bootstrap-table .table:not(.table-condensed)>tfoot>tr>th,.bootstrap-table .table:not(.table-condensed)>thead>tr>td{padding:8px}.bootstrap-table .table.table-no-bordered>tbody>tr>td,.bootstrap-table .table.table-no-bordered>thead>tr>th{border-right:2px solid transparent}.bootstrap-table .table.table-no-bordered>tbody>tr>td:last-child{border-right:none}.fixed-table-container{position:relative;clear:both;border:1px solid #ddd;border-radius:4px;-webkit-border-radius:4px;-moz-border-radius:4px}.fixed-table-container.table-no-bordered{border:1px solid transparent}.fixed-table-footer,.fixed-table-header{overflow:hidden}.fixed-table-footer{border-top:1px solid #ddd}.fixed-table-body{overflow-x:auto;overflow-y:auto;height:100%}.fixed-table-container table{width:100%}.fixed-table-container thead th{height:0;padding:0;margin:0;border-left:1px solid #ddd}.fixed-table-container thead th:focus{outline:transparent solid 0}.fixed-table-container thead th:first-child{border-left:none;border-top-left-radius:4px;-webkit-border-top-left-radius:4px;-moz-border-radius-topleft:4px}.fixed-table-container tbody td .th-inner,.fixed-table-container thead th .th-inner{padding:8px;line-height:24px;vertical-align:top;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.fixed-table-container thead th .sortable{cursor:pointer;background-position:right;background-repeat:no-repeat;padding-right:30px}.fixed-table-container thead th .both{background-image:url(' QMQ5AQBCF4dWQSJxC5wwax1Cq1e7BAdxD5SL+Tq/QCM1oNiJidwox0355mXnG/DrEtIQ6azioNZQxI0ykPhTQIwhCR+BmBYtlK7kLJYwWCcJA9M4qdrZrd8pPjZWPtOqdRQy320YSV17OatFC4euts6z39GYMKRPCTKY9UnPQ6P+GtMRfGtPnBCiqhAeJPmkqAAAAAElFTkSuQmCC')}.fixed-table-container thead th .asc{background-image:url()}.fixed-table-container thead th .desc{background-image:url()}.fixed-table-container th.detail{width:30px}.fixed-table-container tbody td{border-left:1px solid #ddd}.fixed-table-container tbody tr:first-child td{border-top:none}.fixed-table-container tbody td:first-child{border-left:none}.fixed-table-container tbody .selected td{background-color:#f5f5f5}.fixed-table-container .bs-checkbox .th-inner{padding:8px 0}.fixed-table-container input[type=radio],.fixed-table-container input[type=checkbox]{margin:0 auto!important}.fixed-table-pagination .pagination-detail,.fixed-table-pagination div.pagination{margin-top:10px;margin-bottom:10px}.fixed-table-pagination div.pagination .pagination{margin:0}.fixed-table-pagination .pagination a{padding:6px 12px;line-height:1.428571429}.fixed-table-pagination .pagination-info{line-height:34px;margin-right:5px}.fixed-table-pagination .btn-group{position:relative;display:inline-block;vertical-align:middle}.fixed-table-pagination .dropup .dropdown-menu{margin-bottom:0}.fixed-table-pagination .page-list{display:inline-block}.fixed-table-toolbar .columns-left{margin-right:5px}.fixed-table-toolbar .columns-right{margin-left:5px}.fixed-table-toolbar .columns label{display:block;padding:3px 20px;clear:both;font-weight:400;line-height:1.428571429}.fixed-table-toolbar .bs-bars,.fixed-table-toolbar .columns,.fixed-table-toolbar .search{position:relative;margin-top:10px;margin-bottom:10px;line-height:34px}.fixed-table-pagination li.disabled a{pointer-events:none;cursor:default}.fixed-table-loading{display:none;position:absolute;top:42px;right:0;bottom:0;left:0;z-index:99;background-color:#fff;text-align:center}.fixed-table-body .card-view .title{font-weight:700;display:inline-block;min-width:30%;text-align:left!important}.table td,.table th{vertical-align:middle}.fixed-table-toolbar .dropdown-menu{text-align:left;max-height:300px;overflow:auto}.fixed-table-toolbar .btn-group>.btn-group{display:inline-block;margin-left:-1px!important}.fixed-table-toolbar .btn-group>.btn-group>.btn{border-radius:0}.fixed-table-toolbar .btn-group>.btn-group:first-child>.btn{border-top-left-radius:4px;border-bottom-left-radius:4px}.fixed-table-toolbar .btn-group>.btn-group:last-child>.btn{border-top-right-radius:4px;border-bottom-right-radius:4px}.bootstrap-table .table>thead>tr>th{vertical-align:bottom;border-bottom:1px solid #ddd}.bootstrap-table .table thead>tr>th{padding:0;margin:0}.bootstrap-table .fixed-table-footer tbody>tr>td{padding:0!important}.bootstrap-table .fixed-table-footer .table{border-bottom:none;border-radius:0;padding:0!important}.pull-right .dropdown-menu{right:0;left:auto}p.fixed-table-scroll-inner{width:100%;height:200px}div.fixed-table-scroll-outer{top:0;left:0;visibility:hidden;width:200px;height:150px;overflow:hidden} -------------------------------------------------------------------------------- /sql/static/bootstrap-table/js/bootstrap-table-export.min.js: -------------------------------------------------------------------------------- 1 | /* 2 | * bootstrap-table - v1.11.1 - 2017-02-22 3 | * https://github.com/wenzhixin/bootstrap-table 4 | * Copyright (c) 2017 zhixin wen 5 | * Licensed MIT License 6 | */ 7 | !function(a){"use strict";var b=a.fn.bootstrapTable.utils.sprintf,c={json:"JSON",xml:"XML",png:"PNG",csv:"CSV",txt:"TXT",sql:"SQL",doc:"MS-Word",excel:"MS-Excel",xlsx:"MS-Excel (OpenXML)",powerpoint:"MS-Powerpoint",pdf:"PDF"};a.extend(a.fn.bootstrapTable.defaults,{showExport:!1,exportDataType:"basic",exportTypes:["json","xml","csv","txt","sql","excel"],exportOptions:{}}),a.extend(a.fn.bootstrapTable.defaults.icons,{"export":"glyphicon-export icon-share"}),a.extend(a.fn.bootstrapTable.locales,{formatExport:function(){return"Export data"}}),a.extend(a.fn.bootstrapTable.defaults,a.fn.bootstrapTable.locales);var d=a.fn.bootstrapTable.Constructor,e=d.prototype.initToolbar;d.prototype.initToolbar=function(){if(this.showToolbar=this.options.showExport,e.apply(this,Array.prototype.slice.apply(arguments)),this.options.showExport){var d=this,f=this.$toolbar.find(">.btn-group"),g=f.find("div.export");if(!g.length){g=a(['
    ','",'","
    "].join("")).appendTo(f);var h=g.find(".dropdown-menu"),i=this.options.exportTypes;if("string"==typeof this.options.exportTypes){var j=this.options.exportTypes.slice(1,-1).replace(/ /g,"").split(",");i=[],a.each(j,function(a,b){i.push(b.slice(1,-1))})}a.each(i,function(a,b){c.hasOwnProperty(b)&&h.append(['
  • ','',c[b],"","
  • "].join(""))}),h.find("li").click(function(){var b=a(this).data("type"),c=function(){d.$el.tableExport(a.extend({},d.options.exportOptions,{type:b,escape:!1}))};if("all"===d.options.exportDataType&&d.options.pagination)d.$el.one("server"===d.options.sidePagination?"post-body.bs.table":"page-change.bs.table",function(){c(),d.togglePagination()}),d.togglePagination();else if("selected"===d.options.exportDataType){var e=d.getData(),f=d.getAllSelections();"server"===d.options.sidePagination&&(e={total:d.options.totalRows},e[d.options.dataField]=d.getData(),f={total:d.options.totalRows},f[d.options.dataField]=d.getAllSelections()),d.load(f),c(),d.load(e)}else c()})}}}}(jQuery); -------------------------------------------------------------------------------- /sql/static/bootstrap-table/js/bootstrap-table-zh-CN.min.js: -------------------------------------------------------------------------------- 1 | /* 2 | * bootstrap-table - v1.11.0 - 2016-07-02 3 | * https://github.com/wenzhixin/bootstrap-table 4 | * Copyright (c) 2016 zhixin wen 5 | * Licensed MIT License 6 | */ 7 | !function(a){"use strict";a.fn.bootstrapTable.locales["zh-CN"]={formatLoadingMessage:function(){return"正在努力地加载数据中,请稍候……"},formatRecordsPerPage:function(a){return"每页显示 "+a+" 条记录"},formatShowingRows:function(a,b,c){return"显示第 "+a+" 到第 "+b+" 条记录,总共 "+c+" 条记录"},formatSearch:function(){return"搜索"},formatNoMatches:function(){return"没有找到匹配的记录"},formatPaginationSwitch:function(){return"隐藏/显示分页"},formatRefresh:function(){return"刷新"},formatToggle:function(){return"切换"},formatColumns:function(){return"列"},formatExport:function(){return"导出数据"},formatClearFilters:function(){return"清空过滤"}},a.extend(a.fn.bootstrapTable.defaults,a.fn.bootstrapTable.locales["zh-CN"])}(jQuery); -------------------------------------------------------------------------------- /sql/static/charts.html: -------------------------------------------------------------------------------- 1 | {% extends "base.html" %} 2 | 3 | {% block content %} 4 |
    5 | 6 |
    7 | 8 |
    9 | {% endblock content %} 10 | 11 | {% block js %} 12 | {% load staticfiles %} 13 | 14 | 15 | {% endblock %} 16 | 17 | 18 | -------------------------------------------------------------------------------- /sql/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 | -------------------------------------------------------------------------------- /sql/static/dist/css/bootstrap-select.min.css: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */.bootstrap-select{width:220px \0}.bootstrap-select>.dropdown-toggle{width:100%;padding-right:25px}.error .bootstrap-select .dropdown-toggle,.has-error .bootstrap-select .dropdown-toggle{border-color:#b94a48}.bootstrap-select.fit-width{width:auto!important}.bootstrap-select:not([class*=col-]):not([class*=form-control]):not(.input-group-btn){width:220px}.bootstrap-select .dropdown-toggle:focus{outline:thin dotted #333!important;outline:5px auto -webkit-focus-ring-color!important;outline-offset:-2px}.bootstrap-select.form-control{margin-bottom:0;padding:0;border:none}.bootstrap-select.form-control:not([class*=col-]){width:100%}.bootstrap-select.form-control.input-group-btn{z-index:auto}.bootstrap-select.btn-group:not(.input-group-btn),.bootstrap-select.btn-group[class*=col-]{float:none;display:inline-block;margin-left:0}.bootstrap-select.btn-group.dropdown-menu-right,.bootstrap-select.btn-group[class*=col-].dropdown-menu-right,.row .bootstrap-select.btn-group[class*=col-].dropdown-menu-right{float:right}.form-group .bootstrap-select.btn-group,.form-horizontal .bootstrap-select.btn-group,.form-inline .bootstrap-select.btn-group{margin-bottom:0}.form-group-lg .bootstrap-select.btn-group.form-control,.form-group-sm .bootstrap-select.btn-group.form-control{padding:0}.form-inline .bootstrap-select.btn-group .form-control{width:100%}.bootstrap-select.btn-group.disabled,.bootstrap-select.btn-group>.disabled{cursor:not-allowed}.bootstrap-select.btn-group.disabled:focus,.bootstrap-select.btn-group>.disabled:focus{outline:0!important}.bootstrap-select.btn-group .dropdown-toggle .filter-option{display:inline-block;overflow:hidden;width:100%;text-align:left}.bootstrap-select.btn-group .dropdown-toggle .caret{position:absolute;top:50%;right:12px;margin-top:-2px;vertical-align:middle}.bootstrap-select.btn-group[class*=col-] .dropdown-toggle{width:100%}.bootstrap-select.btn-group .dropdown-menu{min-width:100%;z-index:1035;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.bootstrap-select.btn-group .dropdown-menu.inner{position:static;float:none;border:0;padding:0;margin:0;border-radius:0;-webkit-box-shadow:none;box-shadow:none}.bootstrap-select.btn-group .dropdown-menu li{position:relative}.bootstrap-select.btn-group .dropdown-menu li.active small{color:#fff}.bootstrap-select.btn-group .dropdown-menu li.disabled a{cursor:not-allowed}.bootstrap-select.btn-group .dropdown-menu li a{cursor:pointer}.bootstrap-select.btn-group .dropdown-menu li a.opt{position:relative;padding-left:2.25em}.bootstrap-select.btn-group .dropdown-menu li a span.check-mark{display:none}.bootstrap-select.btn-group .dropdown-menu li a span.text{display:inline-block}.bootstrap-select.btn-group .dropdown-menu li small{padding-left:.5em}.bootstrap-select.btn-group .dropdown-menu .notify{position:absolute;bottom:5px;width:96%;margin:0 2%;min-height:26px;padding:3px 5px;background:#f5f5f5;border:1px solid #e3e3e3;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.05);box-shadow:inset 0 1px 1px rgba(0,0,0,.05);pointer-events:none;opacity:.9;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.bootstrap-select.btn-group .no-results{padding:3px;background:#f5f5f5;margin:0 5px;white-space:nowrap}.bootstrap-select.btn-group.fit-width .dropdown-toggle .filter-option{position:static}.bootstrap-select.btn-group.fit-width .dropdown-toggle .caret{position:static;top:auto;margin-top:-1px}.bootstrap-select.btn-group.show-tick .dropdown-menu li.selected a span.check-mark{position:absolute;display:inline-block;right:15px;margin-top:5px}.bootstrap-select.btn-group.show-tick .dropdown-menu li a span.text{margin-right:34px}.bootstrap-select.show-menu-arrow.open>.dropdown-toggle{z-index:1036}.bootstrap-select.show-menu-arrow .dropdown-toggle:before{content:'';border-left:7px solid transparent;border-right:7px solid transparent;border-bottom:7px solid rgba(204,204,204,.2);position:absolute;bottom:-4px;left:9px;display:none}.bootstrap-select.show-menu-arrow .dropdown-toggle:after{content:'';border-left:6px solid transparent;border-right:6px solid transparent;border-bottom:6px solid #fff;position:absolute;bottom:-4px;left:10px;display:none}.bootstrap-select.show-menu-arrow.dropup .dropdown-toggle:before{bottom:auto;top:-3px;border-top:7px solid rgba(204,204,204,.2);border-bottom:0}.bootstrap-select.show-menu-arrow.dropup .dropdown-toggle:after{bottom:auto;top:-3px;border-top:6px solid #fff;border-bottom:0}.bootstrap-select.show-menu-arrow.pull-right .dropdown-toggle:before{right:12px;left:auto}.bootstrap-select.show-menu-arrow.pull-right .dropdown-toggle:after{right:13px;left:auto}.bootstrap-select.show-menu-arrow.open>.dropdown-toggle:after,.bootstrap-select.show-menu-arrow.open>.dropdown-toggle:before{display:block}.bs-actionsbox,.bs-donebutton,.bs-searchbox{padding:4px 8px}.bs-actionsbox{float:left;width:100%;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.bs-actionsbox .btn-group button{width:50%}.bs-donebutton{float:left;width:100%;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.bs-donebutton .btn-group button{width:100%}.bs-searchbox+.bs-actionsbox{padding:0 8px 4px}.bs-searchbox .form-control{margin-bottom:0;width:100%}select.bs-select-hidden,select.selectpicker{display:none!important}select.mobile-device{position:absolute!important;top:0;left:0;display:block!important;width:100%;height:100%!important;opacity:0} -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-bg_BG.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Нищо избрано', 27 | noneResultsText: 'Няма резултат за {0}', 28 | countSelectedText: function (numSelected, numTotal) { 29 | return (numSelected == 1) ? "{0} избран елемент" : "{0} избрани елемента"; 30 | }, 31 | maxOptionsText: function (numAll, numGroup) { 32 | return [ 33 | (numAll == 1) ? 'Лимита е достигнат ({n} елемент максимум)' : 'Лимита е достигнат ({n} елемента максимум)', 34 | (numGroup == 1) ? 'Груповия лимит е достигнат ({n} елемент максимум)' : 'Груповия лимит е достигнат ({n} елемента максимум)' 35 | ]; 36 | }, 37 | selectAllText: 'Избери всички', 38 | deselectAllText: 'Размаркирай всички', 39 | multipleSeparator: ', ' 40 | }; 41 | })(jQuery); 42 | 43 | 44 | })); 45 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-bg_BG.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Нищо избрано",noneResultsText:"Няма резултат за {0}",countSelectedText:function(a,b){return 1==a?"{0} избран елемент":"{0} избрани елемента"},maxOptionsText:function(a,b){return[1==a?"Лимита е достигнат ({n} елемент максимум)":"Лимита е достигнат ({n} елемента максимум)",1==b?"Груповия лимит е достигнат ({n} елемент максимум)":"Груповия лимит е достигнат ({n} елемента максимум)"]},selectAllText:"Избери всички",deselectAllText:"Размаркирай всички",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-cs_CZ.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Nic není vybráno', 27 | noneResultsText: 'Žádné výsledky {0}', 28 | countSelectedText: 'Označeno {0} z {1}', 29 | maxOptionsText: ['Limit překročen ({n} {var} max)', 'Limit skupiny překročen ({n} {var} max)', ['položek', 'položka']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-cs_CZ.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Nic není vybráno",noneResultsText:"Žádné výsledky {0}",countSelectedText:"Označeno {0} z {1}",maxOptionsText:["Limit překročen ({n} {var} max)","Limit skupiny překročen ({n} {var} max)",["položek","položka"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-da_DK.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Intet valgt', 27 | noneResultsText: 'Ingen resultater fundet {0}', 28 | countSelectedText: function (numSelected, numTotal) { 29 | return (numSelected == 1) ? "{0} valgt" : "{0} valgt"; 30 | }, 31 | maxOptionsText: function (numAll, numGroup) { 32 | return [ 33 | (numAll == 1) ? 'Begrænsning nået (max {n} valgt)' : 'Begrænsning nået (max {n} valgte)', 34 | (numGroup == 1) ? 'Gruppe-begrænsning nået (max {n} valgt)' : 'Gruppe-begrænsning nået (max {n} valgte)' 35 | ]; 36 | }, 37 | selectAllText: 'Markér alle', 38 | deselectAllText: 'Afmarkér alle', 39 | multipleSeparator: ', ' 40 | }; 41 | })(jQuery); 42 | 43 | 44 | })); 45 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-da_DK.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Intet valgt",noneResultsText:"Ingen resultater fundet {0}",countSelectedText:function(a,b){return"{0} valgt"},maxOptionsText:function(a,b){return[1==a?"Begrænsning nået (max {n} valgt)":"Begrænsning nået (max {n} valgte)",1==b?"Gruppe-begrænsning nået (max {n} valgt)":"Gruppe-begrænsning nået (max {n} valgte)"]},selectAllText:"Markér alle",deselectAllText:"Afmarkér alle",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-de_DE.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Bitte wählen...', 27 | noneResultsText: 'Keine Ergebnisse für {0}', 28 | countSelectedText: '{0} von {1} ausgewählt', 29 | maxOptionsText: ['Limit erreicht ({n} {var} max.)', 'Gruppen-Limit erreicht ({n} {var} max.)', ['Eintrag', 'Einträge']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-de_DE.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Bitte wählen...",noneResultsText:"Keine Ergebnisse für {0}",countSelectedText:"{0} von {1} ausgewählt",maxOptionsText:["Limit erreicht ({n} {var} max.)","Gruppen-Limit erreicht ({n} {var} max.)",["Eintrag","Einträge"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-en_US.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Nothing selected', 27 | noneResultsText: 'No results match {0}', 28 | countSelectedText: function (numSelected, numTotal) { 29 | return (numSelected == 1) ? "{0} item selected" : "{0} items selected"; 30 | }, 31 | maxOptionsText: function (numAll, numGroup) { 32 | return [ 33 | (numAll == 1) ? 'Limit reached ({n} item max)' : 'Limit reached ({n} items max)', 34 | (numGroup == 1) ? 'Group limit reached ({n} item max)' : 'Group limit reached ({n} items max)' 35 | ]; 36 | }, 37 | selectAllText: 'Select All', 38 | deselectAllText: 'Deselect All', 39 | multipleSeparator: ', ' 40 | }; 41 | })(jQuery); 42 | 43 | 44 | })); 45 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-en_US.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Nothing selected",noneResultsText:"No results match {0}",countSelectedText:function(a,b){return 1==a?"{0} item selected":"{0} items selected"},maxOptionsText:function(a,b){return[1==a?"Limit reached ({n} item max)":"Limit reached ({n} items max)",1==b?"Group limit reached ({n} item max)":"Group limit reached ({n} items max)"]},selectAllText:"Select All",deselectAllText:"Deselect All",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-es_CL.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'No hay selección', 27 | noneResultsText: 'No hay resultados {0}', 28 | countSelectedText: 'Seleccionados {0} de {1}', 29 | maxOptionsText: ['Límite alcanzado ({n} {var} max)', 'Límite del grupo alcanzado({n} {var} max)', ['elementos', 'element']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-es_CL.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"No hay selección",noneResultsText:"No hay resultados {0}",countSelectedText:"Seleccionados {0} de {1}",maxOptionsText:["Límite alcanzado ({n} {var} max)","Límite del grupo alcanzado({n} {var} max)",["elementos","element"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-eu.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Hautapenik ez', 27 | noneResultsText: 'Emaitzarik ez {0}', 28 | countSelectedText: '{1}(e)tik {0} hautatuta', 29 | maxOptionsText: ['Mugara iritsita ({n} {var} gehienez)', 'Taldearen mugara iritsita ({n} {var} gehienez)', ['elementu', 'elementu']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-eu.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Hautapenik ez",noneResultsText:"Emaitzarik ez {0}",countSelectedText:"{1}(e)tik {0} hautatuta",maxOptionsText:["Mugara iritsita ({n} {var} gehienez)","Taldearen mugara iritsita ({n} {var} gehienez)",["elementu","elementu"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-fa_IR.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'چیزی انتخاب نشده است', 27 | noneResultsText: 'هیج مشابهی برای {0} پیدا نشد', 28 | countSelectedText: "{0} از {1} مورد انتخاب شده", 29 | maxOptionsText: ['بیشتر ممکن نیست {حداکثر {n} عدد}', 'بیشتر ممکن نیست {حداکثر {n} عدد}'], 30 | selectAllText: 'انتخاب همه', 31 | deselectAllText: 'انتخاب هیچ کدام', 32 | multipleSeparator: ', ' 33 | }; 34 | })(jQuery); 35 | 36 | 37 | })); 38 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-fa_IR.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"چیزی انتخاب نشده است",noneResultsText:"هیج مشابهی برای {0} پیدا نشد",countSelectedText:"{0} از {1} مورد انتخاب شده",maxOptionsText:["بیشتر ممکن نیست {حداکثر {n} عدد}","بیشتر ممکن نیست {حداکثر {n} عدد}"],selectAllText:"انتخاب همه",deselectAllText:"انتخاب هیچ کدام",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-fr_FR.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Aucune sélection', 27 | noneResultsText: 'Aucun résultat pour {0}', 28 | countSelectedText: function (numSelected, numTotal) { 29 | return (numSelected > 1) ? "{0} éléments sélectionnés" : "{0} élément sélectionné"; 30 | }, 31 | maxOptionsText: function (numAll, numGroup) { 32 | return [ 33 | (numAll > 1) ? 'Limite atteinte ({n} éléments max)' : 'Limite atteinte ({n} élément max)', 34 | (numGroup > 1) ? 'Limite du groupe atteinte ({n} éléments max)' : 'Limite du groupe atteinte ({n} élément max)' 35 | ]; 36 | }, 37 | multipleSeparator: ', ' 38 | }; 39 | })(jQuery); 40 | 41 | 42 | })); 43 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-fr_FR.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Aucune sélection",noneResultsText:"Aucun résultat pour {0}",countSelectedText:function(a,b){return a>1?"{0} éléments sélectionnés":"{0} élément sélectionné"},maxOptionsText:function(a,b){return[a>1?"Limite atteinte ({n} éléments max)":"Limite atteinte ({n} élément max)",b>1?"Limite du groupe atteinte ({n} éléments max)":"Limite du groupe atteinte ({n} élément max)"]},multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-hu_HU.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Válasszon!', 27 | noneResultsText: 'Nincs találat {0}', 28 | countSelectedText: function (numSelected, numTotal) { 29 | return '{n} elem kiválasztva'; 30 | }, 31 | maxOptionsText: function (numAll, numGroup) { 32 | return [ 33 | 'Legfeljebb {n} elem választható', 34 | 'A csoportban legfeljebb {n} elem választható' 35 | ]; 36 | }, 37 | selectAllText: 'Mind', 38 | deselectAllText: 'Egyik sem', 39 | multipleSeparator: ', ' 40 | }; 41 | })(jQuery); 42 | 43 | 44 | })); 45 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-hu_HU.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Válasszon!",noneResultsText:"Nincs találat {0}",countSelectedText:function(a,b){return"{n} elem kiválasztva"},maxOptionsText:function(a,b){return["Legfeljebb {n} elem választható","A csoportban legfeljebb {n} elem választható"]},selectAllText:"Mind",deselectAllText:"Egyik sem",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-it_IT.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Nessuna selezione', 27 | noneResultsText: 'Nessun risultato per {0}', 28 | countSelectedText: 'Selezionati {0} di {1}', 29 | maxOptionsText: ['Limite raggiunto ({n} {var} max)', 'Limite del gruppo raggiunto ({n} {var} max)', ['elementi', 'elemento']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-it_IT.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Nessuna selezione",noneResultsText:"Nessun risultato per {0}",countSelectedText:"Selezionati {0} di {1}",maxOptionsText:["Limite raggiunto ({n} {var} max)","Limite del gruppo raggiunto ({n} {var} max)",["elementi","elemento"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-ko_KR.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: '항목을 선택해주세요', 27 | noneResultsText: '{0} 검색 결과가 없습니다', 28 | countSelectedText: function (numSelected, numTotal) { 29 | return "{0}개를 선택하였습니다"; 30 | }, 31 | maxOptionsText: function (numAll, numGroup) { 32 | return [ 33 | '{n}개까지 선택 가능합니다', 34 | '해당 그룹은 {n}개까지 선택 가능합니다' 35 | ]; 36 | }, 37 | selectAllText: '전체선택', 38 | deselectAllText: '전체해제', 39 | multipleSeparator: ', ' 40 | }; 41 | })(jQuery); 42 | 43 | 44 | })); 45 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-ko_KR.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"항목을 선택해주세요",noneResultsText:"{0} 검색 결과가 없습니다",countSelectedText:function(a,b){return"{0}개를 선택하였습니다"},maxOptionsText:function(a,b){return["{n}개까지 선택 가능합니다","해당 그룹은 {n}개까지 선택 가능합니다"]},selectAllText:"전체선택",deselectAllText:"전체해제",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-nl_NL.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Niets geselecteerd', 27 | noneResultsText: 'Geen resultaten gevonden voor {0}', 28 | countSelectedText: '{0} van {1} geselecteerd', 29 | maxOptionsText: ['Limiet bereikt ({n} {var} max)', 'Groep limiet bereikt ({n} {var} max)', ['items', 'item']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-nl_NL.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Niets geselecteerd",noneResultsText:"Geen resultaten gevonden voor {0}",countSelectedText:"{0} van {1} geselecteerd",maxOptionsText:["Limiet bereikt ({n} {var} max)","Groep limiet bereikt ({n} {var} max)",["items","item"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-pl_PL.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Nic nie zaznaczono', 27 | noneResultsText: 'Brak wyników wyszukiwania {0}', 28 | countSelectedText: 'Zaznaczono {0} z {1}', 29 | maxOptionsText: ['Osiągnięto limit ({n} {var} max)', 'Limit grupy osiągnięty ({n} {var} max)', ['elementy', 'element']], 30 | selectAll: 'Zaznacz wszystkie', 31 | deselectAll: 'Odznacz wszystkie', 32 | multipleSeparator: ', ' 33 | }; 34 | })(jQuery); 35 | 36 | 37 | })); 38 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-pl_PL.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Nic nie zaznaczono",noneResultsText:"Brak wyników wyszukiwania {0}",countSelectedText:"Zaznaczono {0} z {1}",maxOptionsText:["Osiągnięto limit ({n} {var} max)","Limit grupy osiągnięty ({n} {var} max)",["elementy","element"]],selectAll:"Zaznacz wszystkie",deselectAll:"Odznacz wszystkie",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-pt_BR.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Nada selecionado', 27 | noneResultsText: 'Nada encontrado contendo {0}', 28 | countSelectedText: 'Selecionado {0} de {1}', 29 | maxOptionsText: ['Limite excedido (máx. {n} {var})', 'Limite do grupo excedido (máx. {n} {var})', ['itens', 'item']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-pt_BR.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Nada selecionado",noneResultsText:"Nada encontrado contendo {0}",countSelectedText:"Selecionado {0} de {1}",maxOptionsText:["Limite excedido (máx. {n} {var})","Limite do grupo excedido (máx. {n} {var})",["itens","item"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-pt_PT.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Nenhum seleccionado', 27 | noneResultsText: 'Sem resultados contendo {0}', 28 | countSelectedText: 'Selecionado {0} de {1}', 29 | maxOptionsText: ['Limite ultrapassado (máx. {n} {var})', 'Limite de seleções ultrapassado (máx. {n} {var})', ['itens', 'item']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-pt_PT.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Nenhum seleccionado",noneResultsText:"Sem resultados contendo {0}",countSelectedText:"Selecionado {0} de {1}",maxOptionsText:["Limite ultrapassado (máx. {n} {var})","Limite de seleções ultrapassado (máx. {n} {var})",["itens","item"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-ro_RO.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Nu a fost selectat nimic', 27 | noneResultsText: 'Nu exista niciun rezultat {0}', 28 | countSelectedText: '{0} din {1} selectat(e)', 29 | maxOptionsText: ['Limita a fost atinsa ({n} {var} max)', 'Limita de grup a fost atinsa ({n} {var} max)', ['iteme', 'item']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-ro_RO.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Nu a fost selectat nimic",noneResultsText:"Nu exista niciun rezultat {0}",countSelectedText:"{0} din {1} selectat(e)",maxOptionsText:["Limita a fost atinsa ({n} {var} max)","Limita de grup a fost atinsa ({n} {var} max)",["iteme","item"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-ru_RU.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Ничего не выбрано', 27 | noneResultsText: 'Совпадений не найдено {0}', 28 | countSelectedText: 'Выбрано {0} из {1}', 29 | maxOptionsText: ['Достигнут предел ({n} {var} максимум)', 'Достигнут предел в группе ({n} {var} максимум)', ['items', 'item']], 30 | doneButtonText: 'Закрыть', 31 | multipleSeparator: ', ' 32 | }; 33 | })(jQuery); 34 | 35 | 36 | })); 37 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-ru_RU.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Ничего не выбрано",noneResultsText:"Совпадений не найдено {0}",countSelectedText:"Выбрано {0} из {1}",maxOptionsText:["Достигнут предел ({n} {var} максимум)","Достигнут предел в группе ({n} {var} максимум)",["items","item"]],doneButtonText:"Закрыть",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-sk_SK.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Vyberte zo zoznamu', 27 | noneResultsText: 'Pre výraz {0} neboli nájdené žiadne výsledky', 28 | countSelectedText: 'Vybrané {0} z {1}', 29 | maxOptionsText: ['Limit prekročený ({n} {var} max)', 'Limit skupiny prekročený ({n} {var} max)', ['položiek', 'položka']], 30 | selectAllText: 'Vybrať všetky', 31 | deselectAllText: 'Zrušiť výber', 32 | multipleSeparator: ', ' 33 | }; 34 | })(jQuery); 35 | 36 | 37 | })); 38 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-sk_SK.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Vyberte zo zoznamu",noneResultsText:"Pre výraz {0} neboli nájdené žiadne výsledky",countSelectedText:"Vybrané {0} z {1}",maxOptionsText:["Limit prekročený ({n} {var} max)","Limit skupiny prekročený ({n} {var} max)",["položiek","položka"]],selectAllText:"Vybrať všetky",deselectAllText:"Zrušiť výber",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-sl_SI.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Nič izbranega', 27 | noneResultsText: 'Ni zadetkov za {0}', 28 | countSelectedText: function (numSelected, numTotal) { 29 | "Število izbranih: {0}"; 30 | }, 31 | maxOptionsText: function (numAll, numGroup) { 32 | return [ 33 | 'Omejitev dosežena (max. izbranih: {n})', 34 | 'Omejitev skupine dosežena (max. izbranih: {n})' 35 | ]; 36 | }, 37 | selectAllText: 'Izberi vse', 38 | deselectAllText: 'Počisti izbor', 39 | multipleSeparator: ', ' 40 | }; 41 | })(jQuery); 42 | 43 | 44 | })); 45 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-sl_SI.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Nič izbranega",noneResultsText:"Ni zadetkov za {0}",countSelectedText:function(a,b){"Število izbranih: {0}"},maxOptionsText:function(a,b){return["Omejitev dosežena (max. izbranih: {n})","Omejitev skupine dosežena (max. izbranih: {n})"]},selectAllText:"Izberi vse",deselectAllText:"Počisti izbor",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-sv_SE.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Inget valt', 27 | noneResultsText: 'Inget sökresultat matchar {0}', 28 | countSelectedText: function (numSelected, numTotal) { 29 | return (numSelected === 1) ? "{0} alternativ valt" : "{0} alternativ valda"; 30 | }, 31 | maxOptionsText: function (numAll, numGroup) { 32 | return [ 33 | 'Gräns uppnåd (max {n} alternativ)', 34 | 'Gräns uppnåd (max {n} gruppalternativ)' 35 | ]; 36 | }, 37 | selectAllText: 'Markera alla', 38 | deselectAllText: 'Avmarkera alla', 39 | multipleSeparator: ', ' 40 | }; 41 | })(jQuery); 42 | 43 | 44 | })); 45 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-sv_SE.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Inget valt",noneResultsText:"Inget sökresultat matchar {0}",countSelectedText:function(a,b){return 1===a?"{0} alternativ valt":"{0} alternativ valda"},maxOptionsText:function(a,b){return["Gräns uppnåd (max {n} alternativ)","Gräns uppnåd (max {n} gruppalternativ)"]},selectAllText:"Markera alla",deselectAllText:"Avmarkera alla",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-tr_TR.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Hiçbiri seçilmedi', 27 | noneResultsText: 'Hiçbir sonuç bulunamadı {0}', 28 | countSelectedText: function (numSelected, numTotal) { 29 | return (numSelected == 1) ? "{0} öğe seçildi" : "{0} öğe seçildi"; 30 | }, 31 | maxOptionsText: function (numAll, numGroup) { 32 | return [ 33 | (numAll == 1) ? 'Limit aşıldı (maksimum {n} sayıda öğe )' : 'Limit aşıldı (maksimum {n} sayıda öğe)', 34 | (numGroup == 1) ? 'Grup limiti aşıldı (maksimum {n} sayıda öğe)' : 'Grup limiti aşıldı (maksimum {n} sayıda öğe)' 35 | ]; 36 | }, 37 | selectAllText: 'Tümünü Seç', 38 | deselectAllText: 'Seçiniz', 39 | multipleSeparator: ', ' 40 | }; 41 | })(jQuery); 42 | 43 | 44 | })); 45 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-tr_TR.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Hiçbiri seçilmedi",noneResultsText:"Hiçbir sonuç bulunamadı {0}",countSelectedText:function(a,b){return"{0} öğe seçildi"},maxOptionsText:function(a,b){return[1==a?"Limit aşıldı (maksimum {n} sayıda öğe )":"Limit aşıldı (maksimum {n} sayıda öğe)","Grup limiti aşıldı (maksimum {n} sayıda öğe)"]},selectAllText:"Tümünü Seç",deselectAllText:"Seçiniz",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-ua_UA.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: 'Нічого не вибрано', 27 | noneResultsText: 'Збігів не знайдено {0}', 28 | countSelectedText: 'Вибрано {0} із {1}', 29 | maxOptionsText: ['Досягнута межа ({n} {var} максимум)', 'Досягнута межа в групі ({n} {var} максимум)', ['items', 'item']], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-ua_UA.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"Нічого не вибрано",noneResultsText:"Збігів не знайдено {0}",countSelectedText:"Вибрано {0} із {1}",maxOptionsText:["Досягнута межа ({n} {var} максимум)","Досягнута межа в групі ({n} {var} максимум)",["items","item"]],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-zh_CN.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: '没有选中任何项', 27 | noneResultsText: '没有找到匹配项', 28 | countSelectedText: '选中{1}中的{0}项', 29 | maxOptionsText: ['超出限制 (最多选择{n}项)', '组选择超出限制(最多选择{n}组)'], 30 | multipleSeparator: ', ' 31 | }; 32 | })(jQuery); 33 | 34 | 35 | })); 36 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-zh_CN.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"没有选中任何项",noneResultsText:"没有找到匹配项",countSelectedText:"选中{1}中的{0}项",maxOptionsText:["超出限制 (最多选择{n}项)","组选择超出限制(最多选择{n}组)"],multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-zh_TW.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | 8 | (function (root, factory) { 9 | if (typeof define === 'function' && define.amd) { 10 | // AMD. Register as an anonymous module unless amdModuleId is set 11 | define(["jquery"], function (a0) { 12 | return (factory(a0)); 13 | }); 14 | } else if (typeof exports === 'object') { 15 | // Node. Does not work with strict CommonJS, but 16 | // only CommonJS-like environments that support module.exports, 17 | // like Node. 18 | module.exports = factory(require("jquery")); 19 | } else { 20 | factory(jQuery); 21 | } 22 | }(this, function () { 23 | 24 | (function ($) { 25 | $.fn.selectpicker.defaults = { 26 | noneSelectedText: '沒有選取任何項目', 27 | noneResultsText: '沒有找到符合的結果', 28 | countSelectedText: '已經選取{0}個項目', 29 | maxOptionsText: ['超過限制 (最多選擇{n}項)', '超過限制(最多選擇{n}組)'], 30 | selectAllText: '選取全部', 31 | deselectAllText: '全部取消', 32 | multipleSeparator: ', ' 33 | }; 34 | })(jQuery); 35 | 36 | 37 | })); 38 | -------------------------------------------------------------------------------- /sql/static/dist/js/i18n/defaults-zh_TW.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap-select v1.7.2 (http://silviomoreto.github.io/bootstrap-select) 3 | * 4 | * Copyright 2013-2015 bootstrap-select 5 | * Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE) 6 | */ 7 | !function(a,b){"function"==typeof define&&define.amd?define(["jquery"],function(a){return b(a)}):"object"==typeof exports?module.exports=b(require("jquery")):b(jQuery)}(this,function(){!function(a){a.fn.selectpicker.defaults={noneSelectedText:"沒有選取任何項目",noneResultsText:"沒有找到符合的結果",countSelectedText:"已經選取{0}個項目",maxOptionsText:["超過限制 (最多選擇{n}項)","超過限制(最多選擇{n}組)"],selectAllText:"選取全部",deselectAllText:"全部取消",multipleSeparator:", "}}(jQuery)}); -------------------------------------------------------------------------------- /sql/static/error.html: -------------------------------------------------------------------------------- 1 | {% extends "base.html" %} 2 | 3 | {% block content %} 4 |

    {{errMsg}}

    5 | {% endblock content %} 6 | -------------------------------------------------------------------------------- /sql/static/fileinput/img/loading.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/static/fileinput/img/loading.gif -------------------------------------------------------------------------------- /sql/static/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/static/login.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 自动化运维平台 5 | {% load staticfiles %} 6 | 7 | 8 | 9 | 10 | 11 | 12 | 16 | 17 | 18 | 36 | 37 | 38 | 57 | 58 | 59 |
    60 |

    自动化运维平台.  Developed by OP.

    61 |
    62 | 63 | 64 | 65 | 66 | 67 | 68 | 69 | 75 | 76 | -------------------------------------------------------------------------------- /sql/static/rollback.html: -------------------------------------------------------------------------------- 1 | {% extends "base.html" %} 2 | 3 | {% block content %} 4 |
    5 |
    6 | {% csrf_token %} 7 | 9 | 10 | 11 | 12 | 13 | 14 | 提交回滚请求 15 |
    16 |
    17 |
    18 |
    19 |
    20 | {% endblock content %} 21 | 22 | {% block js %} 23 | {% load staticfiles %} 24 | 25 | 94 | {% endblock %} 95 | 96 | -------------------------------------------------------------------------------- /sql/static/user/css/user1.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 | .user-center-div-col-md-10 { 13 | margin-bottom:50px; 14 | } 15 | 16 | .login-div-center { 17 | position: absolute; 18 | top: 45%; 19 | left: 50%; 20 | width: 500px; 21 | border-radius: 4px; 22 | box-shadow: 0 0 0 1px rgba(0,0,0,0.07); 23 | transform: translate(-50%, -50%); 24 | background-color: #f6f7f8; 25 | } 26 | 27 | .lsb-login { 28 | display: flex; 29 | justify-content: center; 30 | align-items: center; 31 | height: 90vh; 32 | } 33 | 34 | .mypanalbox { 35 | background-color: #FFF; 36 | border-radius: 8px; 37 | box-shadow: 3px 3px 3px; 38 | } 39 | 40 | .login-form { 41 | margin: 20px; 42 | } 43 | 44 | .sidebar { 45 | position: fixed; 46 | top: 51px; 47 | bottom: 0; 48 | left: 0; 49 | #z-index: 1000; 50 | display: block; 51 | padding: 20px; 52 | overflow-x: hidden; 53 | overflow-y: auto; 54 | background-color: #f5f5f5; 55 | border-right: 1px solid #eee; 56 | } 57 | -------------------------------------------------------------------------------- /sql/static/user/js/charts.js: -------------------------------------------------------------------------------- 1 | var personDataSummary = null; 2 | var personCorlor = [ 3 | 'rgba(255, 99, 132, 0.9)', 4 | 'rgba(54, 162, 235, 0.8)', 5 | 'rgba(255, 206, 86, 0.7)', 6 | 'rgba(75, 192, 192, 0.8)', 7 | 'rgba(153, 102, 255, 0.9)'] 8 | 9 | $(document).ready(function() { 10 | var isCharts = window.location.pathname.indexOf("charts"); 11 | if (isCharts != -1) { 12 | getPersonWork() 13 | getMonthWork() 14 | } 15 | }); 16 | 17 | function getPersonWork(){ 18 | $.ajax({ 19 | type: 'get', 20 | url: '/getPersonCharts/', 21 | success: function(data) { 22 | var data_person = new Array(); 23 | var lb_person = new Array(); 24 | var bg_corlor = new Array() 25 | for (var i=0; i" + sqlworkflowStatus.finish + "" 34 | } 35 | else if (value === sqlworkflowStatus.abort) { 36 | return "" + sqlworkflowStatus.abort + "" 37 | } 38 | else if (value === sqlworkflowStatus.autoreviewing) { 39 | return "" + sqlworkflowStatus.autoreviewing + "" 40 | } 41 | else if (value === sqlworkflowStatus.manreviewing) { 42 | return "" + sqlworkflowStatus.manreviewing + "" 43 | } 44 | else if (value === sqlworkflowStatus.pass) { 45 | return "" + sqlworkflowStatus.pass + "" 46 | } 47 | else if (value === sqlworkflowStatus.timingtask) { 48 | return "" + sqlworkflowStatus.timingtask + "" 49 | } 50 | else if (value === sqlworkflowStatus.executing) { 51 | return "" + sqlworkflowStatus.executing + "" 52 | } 53 | else if (value === sqlworkflowStatus.autoreviewwrong) { 54 | return "" + sqlworkflowStatus.autoreviewwrong + "" 55 | } 56 | else if (value === sqlworkflowStatus.exception) { 57 | return "" + sqlworkflowStatus.exception + "" 58 | } 59 | else { 60 | return "" + '未知状态' + "" 61 | } 62 | } 63 | 64 | function workflow_type_formatter(value) { 65 | if (value === workflow_type.query) { 66 | return workflow_type.query_display 67 | } 68 | else { 69 | return '未知状态' 70 | } 71 | } 72 | 73 | function workflow_status_formatter(value) { 74 | if (value === workflow_status.audit_wait) { 75 | return "" + workflow_status.audit_wait_display + " " 76 | } 77 | else if (value === workflow_status.audit_success) { 78 | return " " + workflow_status.audit_success_display + " " 79 | } 80 | else if (value === workflow_status.audit_reject) { 81 | return "" + workflow_status.audit_reject_display + " " 82 | } 83 | else if (value === workflow_status.audit_abort) { 84 | return "" + workflow_status.audit_abort_display + " " 85 | } 86 | else { 87 | return "" + '未知状态' + "" 88 | } 89 | } 90 | -------------------------------------------------------------------------------- /sql/static/user/js/login.js: -------------------------------------------------------------------------------- 1 | //回车键提交表单登录 2 | $(document).ready(function() { 3 | $(document).keydown(function(event) { 4 | //keycode==13为回车键 5 | if (event.keyCode == 13) { 6 | $('#btnLogin').addClass('disabled'); 7 | $('#btnLogin').prop('disabled', true); 8 | authenticateUser(); 9 | } 10 | }); 11 | }); 12 | 13 | $('#btnLogin').click(function(){ 14 | $('#btnLogin').addClass('disabled'); 15 | $('#btnLogin').prop('disabled', true); 16 | authenticateUser(); 17 | }); 18 | 19 | function authenticateUser() { 20 | var inputUsername = $('#inputUsername'); 21 | var inputPassword = $('#inputPassword'); 22 | $.ajax({ 23 | type: "post", 24 | url: "/authenticate/", 25 | dataType: "json", 26 | data: { 27 | username: inputUsername.val(), 28 | password: inputPassword.val() 29 | }, 30 | complete: function () { 31 | $('#btnLogin').removeClass('disabled'); 32 | $('#btnLogin').prop('disabled', false); 33 | }, 34 | success: function (data) { 35 | if (data.status == 0) { 36 | $(location).attr('href','/allworkflow/'); 37 | } else { 38 | $('#wrongpwd-modal-body').html(data.msg); 39 | $('#wrongpwd-modal').modal({ 40 | keyboard: true 41 | }); 42 | } 43 | }, 44 | error: function (XMLHttpRequest, textStatus, errorThrown) { 45 | alert(errorThrown); 46 | } 47 | }); 48 | }; 49 | -------------------------------------------------------------------------------- /sql/static/user/js/rollback.js: -------------------------------------------------------------------------------- 1 | $(document).ready(function () { 2 | var isRollback = window.location.pathname.indexOf("rollback"); 3 | if (isRollback != -1) { 4 | $("#btnSubmitRollback").click(function () { 5 | $(this).button('loading').delay(2500).queue(function () { 6 | $(this).button('reset'); 7 | $(this).dequeue(); 8 | }); 9 | var editWorkflowNname = $("#editWorkflowNname").val(); 10 | var editSqlContent = $("#editSqlContent").val(); 11 | var editClustername = $("#editClustername").val(); 12 | var editIsbackup = $("#editIsbackup").val(); 13 | var editReviewman = $("#editReviewman").val(); 14 | var editSubReviewman = $("#editSubReviewman").val(); 15 | sessionStorage.removeItem('editWorkflowDetailId'); 16 | sessionStorage.setItem('editWorkflowNname', editWorkflowNname); 17 | sessionStorage.setItem('editSqlContent', editSqlContent); 18 | sessionStorage.setItem('editClustername', editClustername); 19 | sessionStorage.setItem('editIsbackup', editIsbackup); 20 | sessionStorage.setItem('editReviewman', editReviewman); 21 | sessionStorage.setItem('editSubReviewman', editSubReviewman); 22 | }); 23 | } 24 | ; 25 | }); 26 | -------------------------------------------------------------------------------- /sql/templatetags/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/sql/templatetags/__init__.py -------------------------------------------------------------------------------- /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(',', '
    ')) 13 | -------------------------------------------------------------------------------- /sql/tests.py: -------------------------------------------------------------------------------- 1 | from django.test import TestCase 2 | 3 | # Create your tests here. 4 | -------------------------------------------------------------------------------- /sql/urls.py: -------------------------------------------------------------------------------- 1 | # -*- coding: UTF-8 -*- 2 | 3 | from django.conf.urls import url, include 4 | from . import views, views_ajax, query, jobs 5 | from django.conf import settings 6 | 7 | urlpatterns = [ 8 | url(r'^$', views.sqlworkflow, name='sqlworkflow'), 9 | url(r'^index/$', views.sqlworkflow, name='sqlworkflow'), 10 | url(r'^login/$', views.login, name='login'), 11 | url(r'^logout/$', views.logout, name='logout'), 12 | url(r'^submitsql/$', views.submitSql, name='submitSql'), 13 | url(r'^editsql/$', views.submitSql, name='editsql'), 14 | url(r'^allworkflow/$', views.sqlworkflow, name='sqlworkflow'), 15 | 16 | url(r'^autoreview/$', views.autoreview, name='autoreview'), 17 | url(r'^detail/(?P[0-9]+)/$', views.detail, name='detail'), 18 | url(r'^passed/$', views.passed, name='passed'), 19 | url(r'^execute/$', views.execute, name='execute'), 20 | url(r'^timingtask/$', views.timingtask, name='timingtask'), 21 | url(r'^cancel/$', views.cancel, name='cancel'), 22 | url(r'^rollback/$', views.rollback, name='rollback'), 23 | url(r'^sqlquery/$', views.sqlquery, name='sqlquery'), 24 | url(r'^slowquery/$', views.slowquery, name='slowquery'), 25 | url(r'^sqladvisor/$', views.sqladvisor, name='sqladvisor'), 26 | url(r'^slowquery_advisor/$', views.sqladvisor, name='slowquery_advisor'), 27 | url(r'^queryapplylist/$', views.queryapplylist, name='queryapplylist'), 28 | url(r'^queryapplydetail/(?P[0-9]+)/$', views.queryapplydetail, name='queryapplydetail'), 29 | url(r'^queryuserprivileges/$', views.queryuserprivileges, name='queryuserprivileges'), 30 | url(r'^diagnosis_process/$', views.diagnosis_process, name='diagnosis_process'), 31 | url(r'^diagnosis_sapce/$', views.diagnosis_sapce, name='diagnosis_sapce'), 32 | url(r'^workflow/$', views.workflows, name='workflows'), 33 | url(r'^workflowdetail/(?P[0-9]+)/$', views.workflowsdetail, name='workflowsdetail'), 34 | url(r'^dbaprinciples/$', views.dbaprinciples, name='dbaprinciples'), 35 | url(r'^charts/$', views.charts, name='charts'), 36 | 37 | url(r'^authenticate/$', views_ajax.authenticateEntry, name='authenticate'), 38 | url(r'^sqlworkflow/$', views_ajax.sqlworkflow, name='sqlworkflow'), 39 | url(r'^simplecheck/$', views_ajax.simplecheck, name='simplecheck'), 40 | url(r'^getMonthCharts/$', views_ajax.getMonthCharts, name='getMonthCharts'), 41 | url(r'^getPersonCharts/$', views_ajax.getPersonCharts, name='getPersonCharts'), 42 | url(r'^getOscPercent/$', views_ajax.getOscPercent, name='getOscPercent'), 43 | url(r'^getWorkflowStatus/$', views_ajax.getWorkflowStatus, name='getWorkflowStatus'), 44 | url(r'^stopOscProgress/$', views_ajax.stopOscProgress, name='stopOscProgress'), 45 | url(r'^sqladvisorcheck/$', views_ajax.sqladvisorcheck, name='sqladvisorcheck'), 46 | url(r'^workflowlist/$', views_ajax.workflowlist, name='workflowlist'), 47 | 48 | url(r'^getClusterList/$', query.getClusterList, name='getClusterList'), 49 | url(r'^getdbNameList/$', query.getdbNameList, name='getdbNameList'), 50 | url(r'^getTableNameList/$', query.getTableNameList, name='getTableNameList'), 51 | url(r'^getColumnNameList/$', query.getColumnNameList, name='getColumnNameList'), 52 | url(r'^getqueryapplylist/$', query.getqueryapplylist, name='getqueryapplylist'), 53 | url(r'^getuserprivileges/$', query.getuserprivileges, name='getuserprivileges'), 54 | url(r'^applyforprivileges/$', query.applyforprivileges, name='applyforprivileges'), 55 | url(r'^modifyqueryprivileges/$', query.modifyqueryprivileges, name='modifyqueryprivileges'), 56 | url(r'^queryprivaudit/$', query.queryprivaudit, name='queryprivaudit'), 57 | url(r'^query/$', query.query, name='query'), 58 | url(r'^querylog/$', query.querylog, name='querylog'), 59 | url(r'^explain/$', query.explain, name='explain'), 60 | url(r'^slowquery_review/$', query.slowquery_review, name='slowquery_review'), 61 | url(r'^slowquery_review_history/$', query.slowquery_review_history, name='slowquery_review_history'), 62 | url(r'^process_status/$', views_ajax.process_status, name='process_status'), 63 | url(r'^create_kill_session/$', views_ajax.create_kill_session,name='create_kill_session'), 64 | url(r'^kill_session/$', views_ajax.kill_session, name='kill_session'), 65 | url(r'^sapce_status/$', views_ajax.tablesapce, name='tablesapce'), 66 | url(r'^del_sqlcronjob/$', jobs.del_sqlcronjob, name='del_sqlcronjob'), 67 | ] 68 | 69 | -------------------------------------------------------------------------------- /src/docker/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM docker.io/centos 2 | 3 | #archer 4 | RUN yum -y install unzip git gcc gcc-c++ make cmake bison openssl-devel mysql-devel openldap-devel \ 5 | && yum -y install epel-release \ 6 | && yum -y install python34 python34-pip python34-devel.x86_64 \ 7 | && cd /opt \ 8 | && pip3 install virtualenv -i https://mirrors.ustc.edu.cn/pypi/web/simple/ \ 9 | && virtualenv venv4archer --python=python3.4 \ 10 | && source venv4archer/bin/activate \ 11 | && git clone https://github.com/hhyo/archer.git \ 12 | && pip3 install -r /opt/archer/src/docker/requirements.txt -i https://mirrors.ustc.edu.cn/pypi/web/simple/ \ 13 | && cp /opt/archer/src/docker/pymysql/connections.py /opt/venv4archer/lib/python3.4/site-packages/pymysql/ \ 14 | && cp /opt/archer/src/docker/pymysql/cursors.py /opt/venv4archer/lib/python3.4/site-packages/pymysql/ \ 15 | && yum -y install nginx \ 16 | && cp /opt/archer/src/docker/nginx.conf /etc/nginx/ \ 17 | && nginx \ 18 | #sqladvisor 19 | && yum -y install libaio-devel libffi-devel glib2 glib2-devel \ 20 | && yum -y install http://www.percona.com/downloads/percona-release/redhat/0.1-3/percona-release-0.1-3.noarch.rpm \ 21 | && yum -y install Percona-Server-devel-56.x86_64 Percona-Server-shared-56.x86_64 Percona-Server-client-56.x86_64 \ 22 | && git clone https://github.com/Meituan-Dianping/SQLAdvisor.git \ 23 | && cd /opt/SQLAdvisor/ \ 24 | && yum -y install make \ 25 | && cmake -DBUILD_CONFIG=mysql_release -DCMAKE_BUILD_TYPE=debug -DCMAKE_INSTALL_PREFIX=/usr/local/sqlparser ./ \ 26 | && make && make install \ 27 | && cd sqladvisor/ \ 28 | && cmake -DCMAKE_BUILD_TYPE=debug ./ \ 29 | && make \ 30 | #修改中文支持 31 | && rm -rf /etc/localtime && ln -s /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \ 32 | && yum -y install kde-l10n-Chinese && yum -y reinstall glibc-common \ 33 | && localedef -c -f UTF-8 -i zh_CN zh_CN.utf8 34 | 35 | ENV LANG en_US.UTF-8 36 | ENV LC_ALL zh_CN.utf8 37 | 38 | #port 39 | EXPOSE 9123 40 | 41 | #start service 42 | ENTRYPOINT bash /opt/archer/src/docker/startup.sh && bash 43 | -------------------------------------------------------------------------------- /src/docker/nginx.conf: -------------------------------------------------------------------------------- 1 | # For more information on configuration, see: 2 | # * Official English Documentation: http://nginx.org/en/docs/ 3 | # * Official Russian Documentation: http://nginx.org/ru/docs/ 4 | 5 | user nginx; 6 | worker_processes auto; 7 | error_log /var/log/nginx/error.log; 8 | pid /run/nginx.pid; 9 | 10 | # Load dynamic modules. See /usr/share/nginx/README.dynamic. 11 | include /usr/share/nginx/modules/*.conf; 12 | 13 | events { 14 | worker_connections 1024; 15 | } 16 | 17 | http { 18 | log_format main '$remote_addr - $remote_user [$time_local] "$request" ' 19 | '$status $body_bytes_sent "$http_referer" ' 20 | '"$http_user_agent" "$http_x_forwarded_for"'; 21 | 22 | access_log /var/log/nginx/access.log main; 23 | 24 | access_log on; 25 | sendfile on; 26 | tcp_nopush on; 27 | tcp_nodelay on; 28 | keepalive_timeout 65; 29 | types_hash_max_size 2048; 30 | 31 | include /etc/nginx/mime.types; 32 | default_type application/octet-stream; 33 | 34 | # Load modular configuration files from the /etc/nginx/conf.d directory. 35 | # See http://nginx.org/en/docs/ngx_core_module.html#include 36 | # for more information. 37 | include /etc/nginx/conf.d/*.conf; 38 | 39 | server{ 40 | listen 9123; #监听的端口 41 | server_name archer; 42 | proxy_read_timeout 600s; #超时时间与gunicorn超时时间设置一致 43 | 44 | location / { 45 | proxy_pass http://127.0.0.1:8888; 46 | proxy_set_header Host $host; 47 | proxy_set_header X-Forwarded-Host $host:nginx_port; 48 | proxy_set_header X-Real-IP $remote_addr; 49 | proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; 50 | proxy_set_header X-Forwarded-Proto $scheme; 51 | } 52 | 53 | location /static { 54 | alias /opt/archer/static; 55 | } 56 | 57 | error_page 404 /404.html; 58 | location = /40x.html { 59 | } 60 | 61 | error_page 500 502 503 504 /50x.html; 62 | location = /50x.html { 63 | } 64 | } 65 | 66 | # Settings for a TLS enabled server. 67 | # 68 | # server { 69 | # listen 443 ssl http2 default_server; 70 | # listen [::]:443 ssl http2 default_server; 71 | # server_name _; 72 | # root /usr/share/nginx/html; 73 | # 74 | # ssl_certificate "/etc/pki/nginx/server.crt"; 75 | # ssl_certificate_key "/etc/pki/nginx/private/server.key"; 76 | # ssl_session_cache shared:SSL:1m; 77 | # ssl_session_timeout 10m; 78 | # ssl_ciphers HIGH:!aNULL:!MD5; 79 | # ssl_prefer_server_ciphers on; 80 | # 81 | # # Load configuration files for the default server block. 82 | # include /etc/nginx/default.d/*.conf; 83 | # 84 | # location / { 85 | # } 86 | # 87 | # error_page 404 /404.html; 88 | # location = /40x.html { 89 | # } 90 | # 91 | # error_page 500 502 503 504 /50x.html; 92 | # location = /50x.html { 93 | # } 94 | # } 95 | 96 | } 97 | -------------------------------------------------------------------------------- /src/docker/requirements.txt: -------------------------------------------------------------------------------- 1 | certifi==2022.12.7 2 | chardet==3.0.4 3 | Django==2.2.28 4 | idna==2.6 5 | Naked==0.1.31 6 | pycrypto==2.6.1 7 | PyMySQL==0.7.11 8 | PyYAML==5.4 9 | requests==2.18.4 10 | shellescape==3.4.1 11 | simplejson==3.14.0 12 | urllib3==1.26.5 13 | django-admin-bootstrapped==2.5.7 14 | django-apscheduler==0.2.8 15 | gunicorn==19.7.1 16 | django-auth-ldap==1.3.0 17 | aliyun-python-sdk-core==2.3.5 18 | aliyun-python-sdk-core-v3==2.5.3 19 | aliyun-python-sdk-rds==2.1.1 20 | -------------------------------------------------------------------------------- /src/docker/startup.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | 3 | cd /opt/archer 4 | #切换python运行环境 5 | source /opt/venv4archer/bin/activate 6 | #修改重定向端口,启动ngnix 7 | sed -i "s/nginx_port/$NGINX_PORT/g" /etc/nginx/nginx.conf 8 | /usr/sbin/nginx 9 | 10 | #收集所有的静态文件到STATIC_ROOT 11 | python3 manage.py collectstatic -v0 --noinput 12 | 13 | settings=${1:-"archer.settings"} 14 | ip=${2:-"127.0.0.1"} 15 | port=${3:-8888} 16 | 17 | gunicorn -w 2 --env DJANGO_SETTINGS_MODULE=${settings} --error-logfile=/tmp/archer.err -b ${ip}:${port} --timeout 600 archer.wsgi:application 18 | -------------------------------------------------------------------------------- /src/docs/mysql_db_design_guide.docx: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/docs/mysql_db_design_guide.docx -------------------------------------------------------------------------------- /src/init_sql/master->v2.0.sql: -------------------------------------------------------------------------------- 1 | CREATE TABLE `aliyun_access_key` ( 2 | `id` INT(11) NOT NULL AUTO_INCREMENT, 3 | `ak` VARCHAR(50) NOT NULL, 4 | `secret` VARCHAR(100) NOT NULL, 5 | `is_enable` INT(11) NOT NULL, 6 | `remark` VARCHAR(50) NOT NULL, 7 | PRIMARY KEY (`id`) 8 | ) 9 | ENGINE = InnoDB 10 | DEFAULT CHARSET = utf8; 11 | 12 | CREATE TABLE `aliyun_rds_config` ( 13 | `id` INT(11) NOT NULL AUTO_INCREMENT, 14 | `rds_dbinstanceid` VARCHAR(100) NOT NULL, 15 | `cluster_name` VARCHAR(50) NOT NULL, 16 | PRIMARY KEY (`id`), 17 | UNIQUE KEY `uniq_cluster_name` (`cluster_name`) 18 | ) 19 | ENGINE = InnoDB 20 | DEFAULT CHARSET = utf8; 21 | 22 | ALTER TABLE `data_masking_columns` 23 | DROP COLUMN `cluster_id`, 24 | ADD INDEX `idx_cluster_name` (`cluster_name`); 25 | 26 | ALTER TABLE `data_masking_rules` 27 | ADD COLUMN `hide_group` INT(11) NOT NULL 28 | AFTER `rule_regex`, 29 | ADD UNIQUE INDEX `uniq_rule_type` (`rule_type`); 30 | 31 | ALTER TABLE `query_log` 32 | DROP COLUMN `cluster_id`; 33 | 34 | ALTER TABLE `query_privileges` 35 | ADD COLUMN `priv_type` INT(11) NOT NULL 36 | AFTER `limit_num`, 37 | DROP COLUMN `cluster_id`, 38 | MODIFY COLUMN `valid_date` DATE NOT NULL 39 | AFTER `table_name`; 40 | 41 | ALTER TABLE `query_privileges_apply` 42 | ADD COLUMN `db_list` LONGTEXT NOT NULL 43 | AFTER `user_name`, 44 | ADD COLUMN `priv_type` INT(11) NOT NULL 45 | AFTER `limit_num`, 46 | DROP COLUMN `cluster_id`, 47 | DROP COLUMN `db_name`, 48 | MODIFY COLUMN `valid_date` DATE NOT NULL 49 | AFTER `table_list`; 50 | 51 | ALTER TABLE `sql_master_config` 52 | ADD UNIQUE INDEX `uniq_cluster_name` (`cluster_name`); 53 | 54 | ALTER TABLE `sql_slave_config` 55 | DROP COLUMN `cluster_id`, 56 | ADD UNIQUE INDEX `uniq_cluster_name` (`cluster_name`); 57 | 58 | ALTER TABLE `sql_workflow` 59 | ADD COLUMN `is_manual` INT(11) NOT NULL 60 | AFTER `execute_result`, 61 | ADD COLUMN `audit_remark` LONGTEXT NULL 62 | AFTER `is_manual`; 63 | 64 | ALTER TABLE `workflow_audit` 65 | ADD UNIQUE INDEX `idx_workflow_id_type` (`workflow_id`, `workflow_type`); 66 | 67 | ALTER TABLE `workflow_audit_setting` 68 | ADD UNIQUE INDEX `uniq_workflow_type` (`workflow_type`); 69 | -------------------------------------------------------------------------------- /src/screenshots/admin.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/admin.png -------------------------------------------------------------------------------- /src/screenshots/allworkflow.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/allworkflow.png -------------------------------------------------------------------------------- /src/screenshots/applyforprivileges.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/applyforprivileges.png -------------------------------------------------------------------------------- /src/screenshots/autoreview.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/autoreview.png -------------------------------------------------------------------------------- /src/screenshots/bugs/bug1.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/bugs/bug1.png -------------------------------------------------------------------------------- /src/screenshots/bugs/bug2.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/bugs/bug2.png -------------------------------------------------------------------------------- /src/screenshots/bugs/bug3.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/bugs/bug3.png -------------------------------------------------------------------------------- /src/screenshots/charts.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/charts.png -------------------------------------------------------------------------------- /src/screenshots/datamasking.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/datamasking.png -------------------------------------------------------------------------------- /src/screenshots/datamaskingcolumns.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/datamaskingcolumns.png -------------------------------------------------------------------------------- /src/screenshots/datamaskingrules.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/datamaskingrules.png -------------------------------------------------------------------------------- /src/screenshots/finish.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/finish.png -------------------------------------------------------------------------------- /src/screenshots/login.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/login.png -------------------------------------------------------------------------------- /src/screenshots/manageprivileges.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/manageprivileges.png -------------------------------------------------------------------------------- /src/screenshots/osc_progress.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/osc_progress.png -------------------------------------------------------------------------------- /src/screenshots/process.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/process.png -------------------------------------------------------------------------------- /src/screenshots/pymysql.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/pymysql.png -------------------------------------------------------------------------------- /src/screenshots/query.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/query.png -------------------------------------------------------------------------------- /src/screenshots/querylog.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/querylog.png -------------------------------------------------------------------------------- /src/screenshots/rollback.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/rollback.png -------------------------------------------------------------------------------- /src/screenshots/slowquery.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/slowquery.png -------------------------------------------------------------------------------- /src/screenshots/slowquerylog.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/slowquerylog.png -------------------------------------------------------------------------------- /src/screenshots/sqladvisor.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/sqladvisor.png -------------------------------------------------------------------------------- /src/screenshots/submitsql.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/submitsql.png -------------------------------------------------------------------------------- /src/screenshots/waitingforme.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/waitingforme.png -------------------------------------------------------------------------------- /src/screenshots/workflowconfig.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jly8866/archer/e208c195f969637e3e2c6d110db7153092f04c8f/src/screenshots/workflowconfig.png -------------------------------------------------------------------------------- /src/script/analysis_slow_query.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | DIR="$( cd "$( dirname "$0" )" && pwd )" 3 | cd $DIR 4 | 5 | #配置archer数据库的连接地址 6 | monitor_db_host="127.0.0.1" 7 | monitor_db_port=3306 8 | monitor_db_user="root" 9 | monitor_db_password="123456" 10 | monitor_db_database="archer" 11 | 12 | #被监控机慢日志位置 13 | slowquery_file="/home/mysql/log_slow.log" 14 | pt_query_digest="/usr/bin/pt-query-digest" 15 | 16 | #被监控机连接信息 17 | hostname=mysql_host:mysql_port # 被监控机连接信息,和archer主库配置内容保持一致,用于archer做筛选 18 | 19 | #获取上次分析时间,初始化时请删除last_analysis_time_$hostname文件,可分析全部日志数据 20 | if [ -s last_analysis_time_$hostname ]; then 21 | last_analysis_time=`cat last_analysis_time_$hostname` 22 | else 23 | last_analysis_time='1000-01-01 00:00:00' 24 | fi 25 | 26 | #收集日志 27 | #RDS需要增加--no-version-check选项 28 | $pt_query_digest \ 29 | --user=$monitor_db_user --password=$monitor_db_password --port=$monitor_db_port \ 30 | --review h=$monitor_db_host,D=$monitor_db_database,t=mysql_slow_query_review \ 31 | --history h=$monitor_db_host,D=$monitor_db_database,t=mysql_slow_query_review_history \ 32 | --no-report --limit=100% --charset=utf8 \ 33 | --since "$last_analysis_time" \ 34 | --filter="\$event->{Bytes} = length(\$event->{arg}) and \$event->{hostname}=\"$hostname\" and \$event->{client}=\$event->{ip} " \ 35 | $slowquery_file > /tmp/analysis_slow_query.log 36 | 37 | echo `date +"%Y-%m-%d %H:%M:%S"`>last_analysis_time_$hostname -------------------------------------------------------------------------------- /src/script/centos7_install.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | DIR="$( cd "$( dirname "$0" )" && pwd )" 3 | cd $DIR 4 | git clone -b archer-2.0 https://github.com/jly8866/archer.git \ 5 | && yum -y install gcc gcc-c++ make cmake bison openssl-devel mysql-devel \ 6 | && yum -y install epel-release \ 7 | && yum -y install python34 python34-pip python34-devel.x86_64 \ 8 | && pip3 install virtualenv -i https://mirrors.ustc.edu.cn/pypi/web/simple/ \ 9 | && virtualenv venv4archer --python=python3.4 \ 10 | && source venv4archer/bin/activate \ 11 | && pip3 install -r archer/requirements.txt -i https://mirrors.ustc.edu.cn/pypi/web/simple/ \ 12 | && cp archer/src/docker/pymysql/connections.py venv4archer/lib/python3.4/site-packages/pymysql/ \ 13 | && cp archer/src/docker/pymysql/cursors.py venv4archer/lib/python3.4/site-packages/pymysql/ \ 14 | -------------------------------------------------------------------------------- /src/script/mysql_slow_query_review.sql: -------------------------------------------------------------------------------- 1 | CREATE TABLE `mysql_slow_query_review` ( 2 | `checksum` CHAR(32) NOT NULL, 3 | `fingerprint` longtext NOT NULL, 4 | `sample` longtext NOT NULL, 5 | `first_seen` datetime(6) DEFAULT NULL, 6 | `last_seen` datetime(6) DEFAULT NULL, 7 | `reviewed_by` varchar(20) DEFAULT NULL, 8 | `reviewed_on` datetime(6) DEFAULT NULL, 9 | `comments` longtext, 10 | `reviewed_status` varchar(24) DEFAULT NULL, 11 | PRIMARY KEY (`checksum`), 12 | KEY `idx_last_seen` (`last_seen`) 13 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8; 14 | 15 | CREATE TABLE `mysql_slow_query_review_history` ( 16 | `id` int(11) NOT NULL AUTO_INCREMENT, 17 | `hostname_max` varchar(64) NOT NULL, 18 | `client_max` varchar(64) DEFAULT NULL, 19 | `user_max` varchar(64) NOT NULL, 20 | `db_max` varchar(64) DEFAULT NULL, 21 | `checksum` CHAR(32) NOT NULL, 22 | `sample` longtext NOT NULL, 23 | `ts_min` datetime(6) NOT NULL, 24 | `ts_max` datetime(6) NOT NULL, 25 | `ts_cnt` float DEFAULT NULL, 26 | `Query_time_sum` float DEFAULT NULL, 27 | `Query_time_min` float DEFAULT NULL, 28 | `Query_time_max` float DEFAULT NULL, 29 | `Query_time_pct_95` float DEFAULT NULL, 30 | `Query_time_stddev` float DEFAULT NULL, 31 | `Query_time_median` float DEFAULT NULL, 32 | `Lock_time_sum` float DEFAULT NULL, 33 | `Lock_time_min` float DEFAULT NULL, 34 | `Lock_time_max` float DEFAULT NULL, 35 | `Lock_time_pct_95` float DEFAULT NULL, 36 | `Lock_time_stddev` float DEFAULT NULL, 37 | `Lock_time_median` float DEFAULT NULL, 38 | `Rows_sent_sum` float DEFAULT NULL, 39 | `Rows_sent_min` float DEFAULT NULL, 40 | `Rows_sent_max` float DEFAULT NULL, 41 | `Rows_sent_pct_95` float DEFAULT NULL, 42 | `Rows_sent_stddev` float DEFAULT NULL, 43 | `Rows_sent_median` float DEFAULT NULL, 44 | `Rows_examined_sum` float DEFAULT NULL, 45 | `Rows_examined_min` float DEFAULT NULL, 46 | `Rows_examined_max` float DEFAULT NULL, 47 | `Rows_examined_pct_95` float DEFAULT NULL, 48 | `Rows_examined_stddev` float DEFAULT NULL, 49 | `Rows_examined_median` float DEFAULT NULL, 50 | `Rows_affected_sum` float DEFAULT NULL, 51 | `Rows_affected_min` float DEFAULT NULL, 52 | `Rows_affected_max` float DEFAULT NULL, 53 | `Rows_affected_pct_95` float DEFAULT NULL, 54 | `Rows_affected_stddev` float DEFAULT NULL, 55 | `Rows_affected_median` float DEFAULT NULL, 56 | `Rows_read_sum` float DEFAULT NULL, 57 | `Rows_read_min` float DEFAULT NULL, 58 | `Rows_read_max` float DEFAULT NULL, 59 | `Rows_read_pct_95` float DEFAULT NULL, 60 | `Rows_read_stddev` float DEFAULT NULL, 61 | `Rows_read_median` float DEFAULT NULL, 62 | `Merge_passes_sum` float DEFAULT NULL, 63 | `Merge_passes_min` float DEFAULT NULL, 64 | `Merge_passes_max` float DEFAULT NULL, 65 | `Merge_passes_pct_95` float DEFAULT NULL, 66 | `Merge_passes_stddev` float DEFAULT NULL, 67 | `Merge_passes_median` float DEFAULT NULL, 68 | `InnoDB_IO_r_ops_min` float DEFAULT NULL, 69 | `InnoDB_IO_r_ops_max` float DEFAULT NULL, 70 | `InnoDB_IO_r_ops_pct_95` float DEFAULT NULL, 71 | `InnoDB_IO_r_ops_stddev` float DEFAULT NULL, 72 | `InnoDB_IO_r_ops_median` float DEFAULT NULL, 73 | `InnoDB_IO_r_bytes_min` float DEFAULT NULL, 74 | `InnoDB_IO_r_bytes_max` float DEFAULT NULL, 75 | `InnoDB_IO_r_bytes_pct_95` float DEFAULT NULL, 76 | `InnoDB_IO_r_bytes_stddev` float DEFAULT NULL, 77 | `InnoDB_IO_r_bytes_median` float DEFAULT NULL, 78 | `InnoDB_IO_r_wait_min` float DEFAULT NULL, 79 | `InnoDB_IO_r_wait_max` float DEFAULT NULL, 80 | `InnoDB_IO_r_wait_pct_95` float DEFAULT NULL, 81 | `InnoDB_IO_r_wait_stddev` float DEFAULT NULL, 82 | `InnoDB_IO_r_wait_median` float DEFAULT NULL, 83 | `InnoDB_rec_lock_wait_min` float DEFAULT NULL, 84 | `InnoDB_rec_lock_wait_max` float DEFAULT NULL, 85 | `InnoDB_rec_lock_wait_pct_95` float DEFAULT NULL, 86 | `InnoDB_rec_lock_wait_stddev` float DEFAULT NULL, 87 | `InnoDB_rec_lock_wait_median` float DEFAULT NULL, 88 | `InnoDB_queue_wait_min` float DEFAULT NULL, 89 | `InnoDB_queue_wait_max` float DEFAULT NULL, 90 | `InnoDB_queue_wait_pct_95` float DEFAULT NULL, 91 | `InnoDB_queue_wait_stddev` float DEFAULT NULL, 92 | `InnoDB_queue_wait_median` float DEFAULT NULL, 93 | `InnoDB_pages_distinct_min` float DEFAULT NULL, 94 | `InnoDB_pages_distinct_max` float DEFAULT NULL, 95 | `InnoDB_pages_distinct_pct_95` float DEFAULT NULL, 96 | `InnoDB_pages_distinct_stddev` float DEFAULT NULL, 97 | `InnoDB_pages_distinct_median` float DEFAULT NULL, 98 | `QC_Hit_cnt` float DEFAULT NULL, 99 | `QC_Hit_sum` float DEFAULT NULL, 100 | `Full_scan_cnt` float DEFAULT NULL, 101 | `Full_scan_sum` float DEFAULT NULL, 102 | `Full_join_cnt` float DEFAULT NULL, 103 | `Full_join_sum` float DEFAULT NULL, 104 | `Tmp_table_cnt` float DEFAULT NULL, 105 | `Tmp_table_sum` float DEFAULT NULL, 106 | `Tmp_table_on_disk_cnt` float DEFAULT NULL, 107 | `Tmp_table_on_disk_sum` float DEFAULT NULL, 108 | `Filesort_cnt` float DEFAULT NULL, 109 | `Filesort_sum` float DEFAULT NULL, 110 | `Filesort_on_disk_cnt` float DEFAULT NULL, 111 | `Filesort_on_disk_sum` float DEFAULT NULL, 112 | `Bytes_sum` float DEFAULT NULL, 113 | `Bytes_min` float DEFAULT NULL, 114 | `Bytes_max` float DEFAULT NULL, 115 | `Bytes_pct_95` float DEFAULT NULL, 116 | `Bytes_stddev` float DEFAULT NULL, 117 | `Bytes_median` float DEFAULT NULL, 118 | PRIMARY KEY (`id`), 119 | KEY `idx_hostname_max_ts_min` (`hostname_max`,`ts_min`), 120 | KEY `idx_checksum` (`checksum`) 121 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8; -------------------------------------------------------------------------------- /startup.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | 3 | #收集所有的静态文件到STATIC_ROOT 4 | python3 manage.py collectstatic -v0 --noinput 5 | 6 | settings=${1:-"archer.settings"} 7 | ip=${2:-"0.0.0.0"} 8 | port=${3:-8888} 9 | 10 | gunicorn -w 2 --env DJANGO_SETTINGS_MODULE=${settings} --error-logfile=/tmp/archer.err -b ${ip}:${port} --timeout 600 --daemon archer.wsgi:application 11 | -------------------------------------------------------------------------------- /stop.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | 3 | port=${1:-8888} 4 | ps -ef|grep gunicorn|grep ${port}|awk '{print $2}'|xargs kill -15 5 | --------------------------------------------------------------------------------