""" Django settings for server project. Generated by 'django-admin startproject' using Django 3.0.3. For more information on this file, see https://docs.djangoproject.com/en/3.0/topics/settings/ For the full list of settings and their values, see https://docs.djangoproject.com/en/3.0/ref/settings/ """ from datetime import datetime, timedelta import os import json import sys from config.conf import * from django.core.cache import cache import logging # Build paths inside the project like this: os.path.join(BASE_DIR, ...) BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) sys.path.insert(0, os.path.join(BASE_DIR, 'apps')) # Quick-start development settings - unsuitable for production # See https://docs.djangoproject.com/en/3.0/howto/deployment/checklist/ # SECURITY WARNING: keep the secret key used in production secret! # SECURITY WARNING: don't run with debug turned on in production! ALLOWED_HOSTS = ['*'] SYS_NAME = '星途工厂综合管理系统' SYS_VERSION = '2.8.2025092816' X_FRAME_OPTIONS = 'SAMEORIGIN' # Application definition INSTALLED_APPS = [ 'daphne', 'django.contrib.admin', 'django.contrib.auth', 'django.contrib.contenttypes', 'django.contrib.sessions', 'django.contrib.messages', 'django.contrib.staticfiles', 'corsheaders', 'django_celery_beat', 'django_celery_results', 'drf_yasg', 'rest_framework', 'django_filters', 'apps.develop', 'apps.utils', 'apps.third', 'apps.system', 'apps.auth1', 'apps.monitor', 'apps.wf', 'apps.ecm', 'apps.hrm', 'apps.ichat', 'apps.am', 'apps.vm', 'apps.rpm', 'apps.opm', 'apps.bi', 'apps.em', 'apps.mtm', 'apps.wpm', 'apps.qm', 'apps.enm', 'apps.fim', 'apps.inm', 'apps.sam', 'apps.pum', 'apps.pm', 'apps.enp', 'apps.edu', 'apps.dpm', 'apps.cm', 'apps.cms', 'apps.wpmw', 'apps.ofm' ] MIDDLEWARE = [ 'django.middleware.security.SecurityMiddleware', 'django.contrib.sessions.middleware.SessionMiddleware', 'corsheaders.middleware.CorsMiddleware', 'django.middleware.common.CommonMiddleware', 'django.middleware.csrf.CsrfViewMiddleware', 'django.contrib.auth.middleware.AuthenticationMiddleware', 'django.contrib.messages.middleware.MessageMiddleware', 'django.middleware.clickjacking.XFrameOptionsMiddleware', ] ROOT_URLCONF = 'server.urls' TEMPLATES = [ { 'BACKEND': 'django.template.backends.django.DjangoTemplates', 'DIRS': ['dist'], 'APP_DIRS': True, 'OPTIONS': { 'context_processors': [ 'django.template.context_processors.debug', 'django.template.context_processors.request', 'django.contrib.auth.context_processors.auth', 'django.contrib.messages.context_processors.messages', ], }, }, ] # WSGI WSGI_APPLICATION = 'server.wsgi.application' # ASGI ASGI_APPLICATION = 'server.asgi.application' CHANNEL_LAYERS = { 'default': { 'BACKEND': 'channels_redis.core.RedisChannelLayer', 'CONFIG': { "hosts": [('127.0.0.1', 6379)], "capacity": 1500, "expiry": 10 }, }, } # Database # https://docs.djangoproject.com/en/3.0/ref/settings/#databases # Password validation # https://docs.djangoproject.com/en/3.0/ref/settings/#auth-password-validators AUTH_PASSWORD_VALIDATORS = [ { 'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator', }, { 'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator', }, { 'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator', }, { 'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator', }, ] # Internationalization # https://docs.djangoproject.com/en/3.0/topics/i18n/ LANGUAGE_CODE = 'zh-hans' TIME_ZONE = 'Asia/Shanghai' USE_I18N = True USE_L10N = True USE_TZ = True # Static files (CSS, JavaScript, Images) # https://docs.djangoproject.com/en/3.0/howto/static-files/ STATIC_URL = '/static/' STATIC_ROOT = os.path.join(BASE_DIR, 'dist/static') # STATICFILES_DIRS = ( # os.path.join(BASE_DIR, 'dist/static'), # ) MEDIA_URL = '/media/' MEDIA_ROOT = os.path.join(BASE_DIR, 'media') # 人脸库配置 # 如果地址不存在,则自动创建/现在直接存库可不用 FACE_PATH = os.path.join(BASE_DIR, 'media/face') if not os.path.exists(FACE_PATH): os.makedirs(FACE_PATH) # 默认主键 DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField' # restframework配置 REST_FRAMEWORK = { 'DEFAULT_AUTHENTICATION_CLASSES': [ 'rest_framework_simplejwt.authentication.JWTAuthentication', 'rest_framework.authentication.BasicAuthentication', 'rest_framework.authentication.SessionAuthentication', ], 'DEFAULT_PERMISSION_CLASSES': [ 'rest_framework.permissions.IsAuthenticated', 'apps.utils.permission.RbacPermission' ], # 'DEFAULT_RENDERER_CLASSES': [ # 'apps.utils.response.FitJSONRenderer', # 'rest_framework.renderers.BrowsableAPIRenderer' # ], 'DEFAULT_FILTER_BACKENDS': [ 'django_filters.rest_framework.DjangoFilterBackend', 'rest_framework.filters.SearchFilter', 'rest_framework.filters.OrderingFilter' ], 'DEFAULT_VERSIONING_CLASS': 'rest_framework.versioning.NamespaceVersioning', 'DEFAULT_PAGINATION_CLASS': 'apps.utils.pagination.MyPagination', 'DATETIME_FORMAT': '%Y-%m-%d %H:%M:%S', 'DATETIME_INPUT_FORMATS': ['iso-8601', '%Y-%m-%d %H:%M:%S'], 'DATE_FORMAT': '%Y-%m-%d', 'DEFAULT_SCHEMA_CLASS': 'rest_framework.schemas.coreapi.AutoSchema', # 'UNAUTHENTICATED_USER': None, # 'UNAUTHENTICATED_TOKEN': None, 'EXCEPTION_HANDLER': 'apps.utils.exceptions.custom_exception_hander', 'DEFAULT_THROTTLE_RATES': { 'anon': '100/second', 'user': '200/second' } } # session配置 SESSION_ENGINE = 'django.contrib.sessions.backends.cache' CSRF_COOKIE_NAME = 'csrftoken' # simplejwt配置 SIMPLE_JWT = { 'ACCESS_TOKEN_LIFETIME': timedelta(days=7), 'REFRESH_TOKEN_LIFETIME': timedelta(days=60), } # 跨域配置/可用nginx处理,无需引入corsheaders CORS_ORIGIN_ALLOW_ALL = True CORS_ALLOW_CREDENTIALS = True # Auth配置 AUTH_USER_MODEL = 'system.User' AUTHENTICATION_BACKENDS = ( 'apps.auth1.authentication.CustomBackend', ) # celery配置,celery正常运行必须安装redis CELERYD_MAX_TASKS_PER_CHILD = 100 # 每个worker最多执行100个任务就会被销毁,可防止内存泄露 CELERY_TIMEZONE = 'Asia/Shanghai' # 设置时区 CELERY_ENABLE_UTC = True # 启动时区设置 CELERY_RESULT_BACKEND = 'django-db' CELERY_BEAT_SCHEDULER = 'django_celery_beat.schedulers:DatabaseScheduler' CELERY_ACCEPT_CONTENT = ['application/json'] CELERY_TASK_SERIALIZER = 'json' CELERY_RESULT_SERIALIZER = 'json' CELERY_RESULT_EXTENDED = True CELERY_TASK_TRACK_STARTED = True CELERYD_SOFT_TIME_LIMIT = 60*10 # swagger配置 SWAGGER_SETTINGS = { 'LOGIN_URL': '/django/admin/login/', 'LOGOUT_URL': '/django/admin/logout/', } # 日志配置 # 创建日志的路径 LOG_PATH = os.path.join(BASE_DIR, 'log') # 如果地址不存在,则自动创建log文件夹 if not os.path.exists(LOG_PATH): os.makedirs(LOG_PATH) class TimedSizeRotatingHandler(logging.handlers.TimedRotatingFileHandler): def __init__(self, filename, when='midnight', interval=1, backupCount=0, maxBytes=0, encoding=None, delay=False, utc=False, atTime=None): super().__init__(filename, when, interval, backupCount, encoding, delay, utc, atTime) self.maxBytes = maxBytes def shouldRollover(self, record): if self.maxBytes > 0 and os.path.exists(self.baseFilename): if os.stat(self.baseFilename).st_size >= self.maxBytes: return True return super().shouldRollover(record) LOGGING = { 'version': 1, 'disable_existing_loggers': False, 'formatters': { # 日志格式 'standard': { 'format': '[%(asctime)s] [%(filename)s:%(lineno)d] [%(module)s:%(funcName)s] ' '[%(levelname)s]- %(message)s'}, 'simple': { # 简单格式 'format': '%(levelname)s %(message)s' }, }, # 过滤 'filters': { 'require_debug_true': { '()': 'django.utils.log.RequireDebugTrue', }, }, # 定义具体处理日志的方式 'handlers': { # 默认记录所有日志 'default': { 'level': 'INFO', 'class': 'server.settings.TimedSizeRotatingHandler', 'filename': os.path.join(LOG_PATH, 'all.log'), 'when': 'midnight', # 每天午夜滚动 'interval': 1, 'maxBytes': 1024 * 1024 * 2, # 文件大小 'backupCount': 30, # 备份数 'formatter': 'standard', # 输出格式 'encoding': 'utf-8', # 设置默认编码,否则打印出来汉字乱码 'delay': True, # 延迟打开文件,减少锁定冲突 }, # 输出错误日志 'error': { 'level': 'ERROR', 'class': 'server.settings.TimedSizeRotatingHandler', 'filename': os.path.join(LOG_PATH, 'error.log'), 'when': 'midnight', 'interval': 1, 'maxBytes': 1024 * 1024 * 2, # 文件大小 'backupCount': 30, # 备份数 'formatter': 'standard', # 输出格式 'encoding': 'utf-8', # 设置默认编码 'delay': True, # 延迟打开文件,减少锁定冲突 }, # 控制台输出 'console': { 'level': 'DEBUG', 'class': 'logging.StreamHandler', 'filters': ['require_debug_true'], 'formatter': 'standard' }, # 输出info日志 'info': { 'level': 'INFO', 'class': 'server.settings.TimedSizeRotatingHandler', 'filename': os.path.join(LOG_PATH, 'info.log'), 'when': 'midnight', 'interval': 1, 'maxBytes': 1024 * 1024 * 2, 'backupCount': 30, 'formatter': 'standard', 'encoding': 'utf-8', # 设置默认编码 'delay': True, # 延迟打开文件,减少锁定冲突 }, }, # 配置用哪几种 handlers 来处理日志 'loggers': { # 类型 为 django 处理所有类型的日志, 默认调用 'django': { 'handlers': ['default', 'console'], 'level': 'INFO', 'propagate': False }, # log 调用时需要当作参数传入 'log': { 'handlers': ['error', 'info', 'console', 'default'], 'level': 'INFO', 'propagate': True }, } } ##### 加载客户可自定义配置并提供操作方法 ##### SYS_JSON_PATH = os.path.join(BASE_DIR, 'config/conf.json') def get_sysconfig(key='', default='raise_error', reload=False): """获取系统配置可指定key字符串 """ config = cache.get('system_config', None) if config is None or reload: # 读取配置文件 if not os.path.exists(SYS_JSON_PATH): raise SystemError('未找到配置文件') with open(SYS_JSON_PATH, 'r', encoding='utf-8') as f: config = json.loads(f.read()) cache.set('system_config', config) if key: k_l = key.split('.') for k in k_l: try: config = config[k] except KeyError: if default == 'raise_error': raise else: return default return config def update_dict(dict1, dict2): for key, value in dict2.items(): if key == 'apk_file': # apk_file拷贝到固定位置 from shutil import copyfile copyfile(BASE_DIR + value, BASE_DIR + '/media/zc_ehs.apk') if key in dict1 and isinstance(dict1[key], dict) and isinstance(value, dict): update_dict(dict1[key], value) else: dict1[key] = value def update_sysconfig(new_dict): config = get_sysconfig() update_dict(config, new_dict) with open(SYS_JSON_PATH, 'wb') as f: f.write(json.dumps(config, indent=4, ensure_ascii=False).encode('utf-8')) cache.set('system_config', config)