├── .gitignore ├── Readme.md ├── python_yaodian.sql ├── server ├── manage.py ├── myapp │ ├── __init__.py │ ├── admin.py │ ├── apps.py │ ├── auth │ │ ├── __init__.py │ │ └── authentication.py │ ├── handler.py │ ├── middlewares │ │ ├── LogMiddleware.py │ │ └── __init__.py │ ├── models.py │ ├── permission │ │ ├── __init__.py │ │ └── permission.py │ ├── serializers.py │ ├── tests.py │ ├── urls.py │ ├── utils.py │ └── views │ │ ├── __init__.py │ │ └── admin │ │ ├── __init__.py │ │ ├── ad.py │ │ ├── banner.py │ │ ├── classification.py │ │ ├── comment.py │ │ ├── customer.py │ │ ├── errorLog.py │ │ ├── loginLog.py │ │ ├── notice.py │ │ ├── opLog.py │ │ ├── overview.py │ │ ├── record.py │ │ ├── tag.py │ │ └── user.py ├── readme.md ├── requirements.txt ├── server │ ├── __init__.py │ ├── asgi.py │ ├── settings.py │ ├── urls.py │ └── wsgi.py └── upload │ ├── ad │ ├── 1674045266113.jpeg │ ├── 1674045282581.jpeg │ ├── 1674045308177.png │ ├── 1674045324510.jpeg │ ├── 1684565423182.jpeg │ ├── 1684565863904.jpeg │ └── 1684565876995.png │ ├── avatar │ ├── 1676553050529.png │ ├── 1676553366217.png │ ├── 1676553498600.jpeg │ ├── 1676553815688.jpeg │ ├── 1677240189427.png │ ├── 1677982820781.jpeg │ ├── 1679146350134.jpeg │ ├── 1684593239449.png │ └── 1684593453676.jpeg │ ├── banner │ ├── 1673963977440.jpeg │ ├── 1673964384835.png │ ├── 1673964652167.jpeg │ ├── 1673965110189.jpeg │ ├── 1673965198155.jpeg │ ├── 1673965389141.jpeg │ ├── 1673965574311.png │ ├── 1673965709533.jpeg │ ├── 1673965718720.png │ └── 1673965728690.jpeg │ └── cover │ ├── 1.jpeg │ ├── 1.jpg │ ├── 10.jpg │ ├── 11.jpg │ ├── 12.jpg │ ├── 13.jpg │ ├── 14.jpg │ ├── 15.jpg │ ├── 16.jpg │ ├── 1672749055571.jpeg │ ├── 1674044230851.jpeg │ ├── 1676186518276.jpeg │ ├── 1676186872895.png │ ├── 1676186935002.png │ ├── 1676188277099.png │ ├── 1676188355688.png │ ├── 1676381084256.png │ ├── 1676381091144.png │ ├── 1676381097051.png │ ├── 1676381103032.png │ ├── 1676381110015.png │ ├── 1677500674281.jpeg │ ├── 1677501266461.jpeg │ ├── 1677501470234.jpeg │ ├── 1677501544737.jpeg │ ├── 1677505180730.jpeg │ ├── 1677505298772.jpeg │ ├── 1677505357042.jpeg │ ├── 1677505364969.jpeg │ ├── 1677505380921.jpeg │ ├── 1677505393025.jpeg │ ├── 1677505410960.jpeg │ ├── 1677505421920.jpeg │ ├── 1677505436478.jpeg │ ├── 1677505452753.jpeg │ ├── 1677505579480.jpeg │ ├── 1677505616285.jpeg │ ├── 1677505626565.jpeg │ ├── 1677505637425.jpeg │ ├── 1677505648826.jpeg │ ├── 1677505659291.jpeg │ ├── 1677505667178.jpeg │ ├── 1677505685641.jpeg │ ├── 1677505695894.jpeg │ ├── 1677505706333.jpeg │ ├── 1677505876732.jpeg │ ├── 1677505884200.jpeg │ ├── 1677505890616.jpeg │ ├── 1677505897079.jpeg │ ├── 1677505910282.jpeg │ ├── 1677505919134.jpeg │ ├── 1677505928898.jpeg │ ├── 1677505937048.jpeg │ ├── 1677505945207.jpeg │ ├── 1677505953782.jpeg │ ├── 1677588447622.jpeg │ ├── 1677677497401.jpeg │ ├── 1678529114670.jpeg │ ├── 1679056479437.jpeg │ ├── 1679056528013.png │ ├── 1679056561868.jpeg │ ├── 1679056587496.jpeg │ ├── 1679058036014.jpeg │ ├── 1679058043365.jpeg │ ├── 1679229610791.jpeg │ ├── 1679229780971.jpeg │ ├── 1679229853940.jpeg │ ├── 1679229963230.jpeg │ ├── 1679230045590.jpeg │ ├── 1679230092657.jpeg │ ├── 1679230147996.jpeg │ ├── 1679230364390.jpeg │ ├── 1679230543572.jpeg │ ├── 1679230586243.jpeg │ ├── 1679230641879.jpeg │ ├── 1679230894621.jpeg │ ├── 1679230935716.jpeg │ ├── 1679230989243.jpeg │ ├── 1679231022326.jpeg │ ├── 1679231052822.jpeg │ ├── 1679231084420.jpeg │ ├── 1679231113572.jpeg │ ├── 1679231151929.jpeg │ ├── 1679231184020.jpeg │ ├── 1679231221940.jpeg │ ├── 1679231465845.jpeg │ ├── 1679232418133.jpeg │ ├── 1679232448857.jpeg │ ├── 1679232480948.jpeg │ ├── 1679232514230.jpeg │ ├── 1679232543082.jpeg │ ├── 1679232587234.jpeg │ ├── 1679232615578.jpeg │ ├── 1679232650073.jpeg │ ├── 1679232676818.jpeg │ ├── 1679232717023.jpeg │ ├── 1679232756075.jpeg │ ├── 1679315151478.jpeg │ ├── 1679315191893.jpeg │ ├── 1679315240787.jpeg │ ├── 1679315276060.jpeg │ ├── 1679315310720.jpeg │ ├── 1679315343720.jpeg │ ├── 1679315365210.jpeg │ ├── 1679315396953.jpeg │ ├── 1679315437571.jpeg │ ├── 1679315478327.jpeg │ ├── 1679315749022.jpeg │ ├── 1679315761797.jpeg │ ├── 1679315803245.jpeg │ ├── 1679315842977.jpeg │ ├── 1679315883620.jpeg │ ├── 1679315915467.jpeg │ ├── 1679315954931.jpeg │ ├── 1679315985852.jpeg │ ├── 1679316037517.jpeg │ ├── 1679316072493.jpeg │ ├── 1679316422812.jpeg │ ├── 1679316453474.jpeg │ ├── 1679316484842.jpeg │ ├── 1679316513693.jpeg │ ├── 1679316533369.jpeg │ ├── 1679316566135.jpeg │ ├── 1679316605104.jpeg │ ├── 1679316637185.jpeg │ ├── 1679316666891.jpeg │ ├── 1679316698087.jpeg │ ├── 1679403034316.jpeg │ ├── 1679403102225.jpeg │ ├── 1679403138846.jpeg │ ├── 1679403158073.jpeg │ ├── 1679403193320.jpeg │ ├── 1679403239138.jpeg │ ├── 1679403285129.jpeg │ ├── 1679403331202.jpeg │ ├── 1679403370669.jpeg │ ├── 1679403403756.jpeg │ ├── 1679403503106.jpeg │ ├── 1679403631942.jpeg │ ├── 1679403697740.jpeg │ ├── 1679403736341.jpeg │ ├── 1679488092121.jpeg │ ├── 1684567758685.jpeg │ ├── 1684567804066.jpeg │ ├── 1684567891591.png │ ├── 1685170107820.jpeg │ ├── 1685170987684.jpeg │ ├── 1685171442599.jpeg │ ├── 1685193225537.jpeg │ ├── 1685240184357.jpeg │ ├── 1685240191000.jpeg │ ├── 1685263611250.jpeg │ ├── 1685263639285.jpeg │ ├── 1685263674742.jpeg │ ├── 1685263699773.jpeg │ ├── 1685263937295.jpeg │ ├── 1685263997721.jpeg │ ├── 1685264026586.jpeg │ ├── 1685264114074.jpeg │ ├── 1685264135851.jpeg │ ├── 1685264163148.jpeg │ ├── 1685264250423.jpeg │ ├── 1685264270509.jpeg │ ├── 1685264295265.jpeg │ ├── 1685264355503.jpeg │ ├── 1685264420316.jpeg │ ├── 1685264443807.jpeg │ ├── 1685264535734.jpeg │ ├── 1685264562903.jpeg │ ├── 1685264586283.jpeg │ ├── 1685264771439.jpeg │ ├── 1685264800237.jpeg │ ├── 1685264830789.jpeg │ ├── 1685264878356.jpeg │ ├── 1685265162834.jpeg │ ├── 1691491016343.jpeg │ ├── 17.jpg │ ├── 18.jpg │ ├── 19.jpg │ ├── 2.jpg │ ├── 20.jpg │ ├── 21.jpg │ ├── 22.jpg │ ├── 23.jpg │ ├── 24.jpg │ ├── 25.jpg │ ├── 26.jpg │ ├── 27.jpg │ ├── 28.jpg │ ├── 29.jpg │ ├── 3.jpg │ ├── 30.jpg │ ├── 4.jpg │ ├── 5.jpg │ ├── 6.jpg │ ├── 7.jpg │ ├── 8.jpg │ └── 9.jpg └── web ├── .eslintignore ├── .eslintrc.js ├── .gitignore ├── .stylelintignore ├── README.md ├── build ├── constant.ts └── vite │ └── plugins │ ├── autoImport.ts │ ├── component.ts │ ├── compress.ts │ ├── imagemin.ts │ ├── index.ts │ ├── progress.ts │ ├── restart.ts │ ├── unocss.ts │ └── visualizer.ts ├── index.html ├── package-lock.json ├── package.json ├── postcss.config.js ├── prettier.config.js ├── public ├── favicon.ico └── images │ ├── admin-login-bg.jpg │ ├── bg2.jpg │ └── demo.jpg ├── src ├── App.vue ├── api │ ├── ad.ts │ ├── classification.ts │ ├── comment.ts │ ├── customer.ts │ ├── log.ts │ ├── notice.ts │ ├── order.ts │ ├── overview.ts │ ├── tag.ts │ ├── thing.ts │ └── user.ts ├── assets │ ├── fonts │ │ ├── Blimone-ExtraBold.woff │ │ ├── Blimone-ExtraLight.woff │ │ ├── Blimone-Light.woff │ │ └── Blimone-Regular.woff │ ├── icons │ │ ├── logo.png │ │ └── svg │ │ │ ├── github.svg │ │ │ ├── logo.svg │ │ │ ├── marks.svg │ │ │ ├── test.svg │ │ │ ├── ts.svg │ │ │ └── twitter.svg │ ├── images │ │ ├── Play.png │ │ ├── add.svg │ │ ├── address-right-icon.svg │ │ ├── ali-pay-icon.svg │ │ ├── avatar.jpg │ │ ├── banner-02.webp │ │ ├── banner2.svg │ │ ├── cart-icon.svg │ │ ├── clear-search.svg │ │ ├── code-icon.svg │ │ ├── delete-icon.svg │ │ ├── ebook-download-icon.svg │ │ ├── k-logo.png │ │ ├── login-banner.png │ │ ├── login.png │ │ ├── logo.png │ │ ├── mail-icon.svg │ │ ├── message-icon.svg │ │ ├── order-address-icon.svg │ │ ├── order-icon.svg │ │ ├── order-point-icon.svg │ │ ├── order-thing-icon.svg │ │ ├── pwd-hidden.svg │ │ ├── pwd-icon.svg │ │ ├── qunerweima.jpg │ │ ├── read-online-icon.svg │ │ ├── recommend-hover.svg │ │ ├── register-name.svg │ │ ├── search-icon.svg │ │ ├── searchIcon.svg │ │ ├── setting-card-icon.svg │ │ ├── setting-icon.svg │ │ ├── setting-msg-icon.svg │ │ ├── setting-push-icon.svg │ │ ├── setting-safe-icon.svg │ │ ├── share-icon.svg │ │ ├── tel-icon.svg │ │ ├── want-read-hover.svg │ │ ├── wb-share.svg │ │ └── wx-pay-icon.svg │ └── styles │ │ └── base.less ├── core │ └── bootstrap.js ├── main.js ├── router │ ├── index.js │ └── root.js ├── store │ ├── index.js │ └── modules │ │ ├── app │ │ ├── index.ts │ │ └── types.ts │ │ └── user │ │ ├── index.ts │ │ └── types.ts ├── styles │ ├── index.less │ └── reset.less ├── utils │ ├── auth.ts │ ├── http │ │ └── axios │ │ │ ├── index.ts │ │ │ ├── status.ts │ │ │ └── type.ts │ ├── index.ts │ └── result.ts └── views │ ├── ad.vue │ ├── admin-login.vue │ ├── classification.vue │ ├── comment.vue │ ├── customer.vue │ ├── error-log.vue │ ├── login-log.vue │ ├── main.vue │ ├── notice.vue │ ├── op-log.vue │ ├── overview.vue │ ├── sys-info.vue │ ├── tag.vue │ ├── thing.vue │ └── user.vue ├── stylelint.config.js ├── tsconfig.json ├── types ├── auto-imports.d.ts ├── components.d.ts └── env.d.ts ├── vite.config.ts └── yarn.lock /.gitignore: -------------------------------------------------------------------------------- 1 | /server/myapp/migrations 2 | /server/myapp/views/__pycache__/ 3 | /server/.idea 4 | /web/.idea 5 | /web/dist 6 | /web/node_modules 7 | /server/.idea/ 8 | /.idea/ 9 | __pycache__ 10 | .idea 11 | server/.idea 12 | .DS_Store 13 | server/.DS_Store 14 | web/.DS_Store -------------------------------------------------------------------------------- /Readme.md: -------------------------------------------------------------------------------- 1 | > 基于python+django+vue.js开发的药店信息管理系统,学习过程中,遇到问题可以付费咨询:lengqin1024(微信) 2 | 3 | ### 功能介绍 4 | 5 | 平台采用B/S结构,后端采用主流的Python语言进行开发,前端采用主流的Vue.js进行开发。 6 | 7 | 功能包括:药品管理、分类管理、顾客管理、用户管理、日志管理、系统信息模块。 8 | 9 | 10 | ### 演示地址 11 | 12 | http://yaodian.gitapp.cn 13 | 14 | 演示帐号: 15 | 16 | 用户名:admin123 17 | 密码:admin123 18 | 19 | ### 代码结构 20 | 21 | - server目录是后端代码 22 | - web目录是前端代码 23 | 24 | ### 部署运行 25 | 26 | #### 后端运行步骤 27 | 28 | (1) 安装python 3.8 29 | 30 | (2) 安装依赖。进入server目录下,执行 pip install -r requirements.txt 31 | 32 | (3) 安装mysql 5.7数据库,并创建数据库,命名为xxx,创建SQL如下: 33 | ``` 34 | CREATE DATABASE IF NOT EXISTS xxx DEFAULT CHARSET utf8 COLLATE utf8_general_ci 35 | ``` 36 | (4) 恢复xxx.sql数据。在mysql下依次执行如下命令: 37 | 38 | ``` 39 | mysql> use xxx; 40 | mysql> source D:/xxx/xxx/xxx.sql; 41 | ``` 42 | 43 | (5) 启动django服务。在server目录下执行: 44 | ``` 45 | python manage.py runserver 46 | ``` 47 | 48 | #### 前端运行步骤 49 | 50 | (1) 安装node 16.14 51 | 52 | (2) 进入web目录下,安装依赖,执行: 53 | ``` 54 | npm install 55 | ``` 56 | (3) 运行项目 57 | ``` 58 | npm run dev 59 | ``` 60 | 61 | #### 常见问题 62 | 63 | - 连接后端失败怎么办? 64 | 65 | 编辑前端的constants.js文件,将base_url设置为你自己电脑的ip和端口 66 | 67 | - 需要什么数据库版本? 68 | 69 | 本系统采用的是mysql5.7开发的,理论上5.7以上都支持 70 | 71 | - pip安装依赖失败怎么样? 72 | 73 | 建议使用国内镜像源安装 74 | 75 | 76 | ### 技术咨询 77 | 78 | 微信:lengqin1024 79 | 80 | -------------------------------------------------------------------------------- /server/manage.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | """Django's command-line utility for administrative tasks.""" 3 | import os 4 | import sys 5 | 6 | 7 | def main(): 8 | """Run administrative tasks.""" 9 | os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'server.settings') 10 | try: 11 | from django.core.management import execute_from_command_line 12 | except ImportError as exc: 13 | raise ImportError( 14 | "Couldn't import Django. Are you sure it's installed and " 15 | "available on your PYTHONPATH environment variable? Did you " 16 | "forget to activate a virtual environment?" 17 | ) from exc 18 | execute_from_command_line(sys.argv) 19 | 20 | 21 | if __name__ == '__main__': 22 | main() 23 | -------------------------------------------------------------------------------- /server/myapp/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/myapp/__init__.py -------------------------------------------------------------------------------- /server/myapp/admin.py: -------------------------------------------------------------------------------- 1 | from django.contrib import admin 2 | 3 | # Register your models here. 4 | from myapp.models import Classification, Thing, Tag, User, Comment 5 | 6 | admin.site.register(Classification) 7 | admin.site.register(Tag) 8 | admin.site.register(Thing) 9 | admin.site.register(User) 10 | admin.site.register(Comment) 11 | -------------------------------------------------------------------------------- /server/myapp/apps.py: -------------------------------------------------------------------------------- 1 | from django.apps import AppConfig 2 | 3 | 4 | class MyappConfig(AppConfig): 5 | default_auto_field = 'django.db.models.BigAutoField' 6 | name = 'myapp' 7 | -------------------------------------------------------------------------------- /server/myapp/auth/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/myapp/auth/__init__.py -------------------------------------------------------------------------------- /server/myapp/auth/authentication.py: -------------------------------------------------------------------------------- 1 | from rest_framework import exceptions 2 | from rest_framework.authentication import BaseAuthentication 3 | 4 | from myapp.models import User 5 | 6 | 7 | # 接口认证 8 | class AdminTokenAuthtication(BaseAuthentication): 9 | def authenticate(self, request): 10 | adminToken = request.META.get("HTTP_ADMINTOKEN") 11 | # print("检查adminToken==>" + adminToken) 12 | 13 | users = User.objects.filter(admin_token=adminToken) 14 | """ 15 | 判定条件: 16 | 1. 传了adminToken 17 | 2. 查到了该帐号 18 | 3. 该帐号是管理员或演示帐号 19 | """ 20 | if not adminToken or len(users) == 0 or users[0].role == '2': 21 | raise exceptions.AuthenticationFailed("AUTH_FAIL_END") 22 | else: 23 | print('adminToken验证通过') 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /server/myapp/handler.py: -------------------------------------------------------------------------------- 1 | from rest_framework.response import Response 2 | 3 | 4 | class APIResponse(Response): 5 | def __init__(self, code=0, msg='', data=None, status=200, headers=None, content_type=None, **kwargs): 6 | dic = {'code': code, 'msg': msg} 7 | if data is not None: 8 | dic['data'] = data 9 | 10 | dic.update(kwargs) # 这里使用update 11 | super().__init__(data=dic, status=status, 12 | template_name=None, headers=headers, 13 | exception=False, content_type=content_type) 14 | -------------------------------------------------------------------------------- /server/myapp/middlewares/LogMiddleware.py: -------------------------------------------------------------------------------- 1 | # -*- coding:utf-8 -*- 2 | import time 3 | import json 4 | 5 | from django.utils.deprecation import MiddlewareMixin 6 | 7 | from myapp import utils 8 | from myapp.serializers import OpLogSerializer 9 | 10 | 11 | class OpLogs(MiddlewareMixin): 12 | 13 | def __init__(self, *args): 14 | super(OpLogs, self).__init__(*args) 15 | 16 | self.start_time = None # 开始时间 17 | self.end_time = None # 响应时间 18 | self.data = {} # dict数据 19 | 20 | def process_request(self, request): 21 | 22 | self.start_time = time.time() # 开始时间 23 | 24 | re_ip = utils.get_ip(request) 25 | re_method = request.method 26 | re_content = request.GET if re_method == 'GET' else request.POST 27 | if re_content: 28 | re_content = json.dumps(re_content) 29 | else: 30 | re_content = None 31 | 32 | self.data.update( 33 | { 34 | 're_url': request.path, 35 | 're_method': re_method, 36 | 're_ip': re_ip, 37 | # 're_content': re_content, 38 | } 39 | ) 40 | # print(self.data) 41 | 42 | def process_response(self, request, response): 43 | 44 | # 耗时毫秒/ms 45 | self.end_time = time.time() # 响应时间 46 | access_time = self.end_time - self.start_time 47 | self.data['access_time'] = round(access_time * 1000) 48 | 49 | # 入库 50 | serializer = OpLogSerializer(data=self.data) 51 | if serializer.is_valid(): 52 | serializer.save() 53 | 54 | return response 55 | -------------------------------------------------------------------------------- /server/myapp/middlewares/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/myapp/middlewares/__init__.py -------------------------------------------------------------------------------- /server/myapp/permission/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/myapp/permission/__init__.py -------------------------------------------------------------------------------- /server/myapp/permission/permission.py: -------------------------------------------------------------------------------- 1 | from myapp.models import User 2 | 3 | 4 | def isDemoAdminUser(request): 5 | adminToken = request.META.get("HTTP_ADMINTOKEN") 6 | users = User.objects.filter(admin_token=adminToken) 7 | if len(users) > 0: 8 | user = users[0] 9 | if user.role == '3': # (角色3)表示演示帐号 10 | print('演示帐号===>') 11 | return True 12 | return False 13 | -------------------------------------------------------------------------------- /server/myapp/tests.py: -------------------------------------------------------------------------------- 1 | from django.test import TestCase 2 | 3 | # Create your tests here. 4 | -------------------------------------------------------------------------------- /server/myapp/urls.py: -------------------------------------------------------------------------------- 1 | from django.urls import path 2 | 3 | from myapp import views 4 | 5 | app_name = 'myapp' 6 | urlpatterns = [ 7 | # api 8 | path('admin/overview/count', views.admin.overview.count), 9 | path('admin/overview/sysInfo', views.admin.overview.sysInfo), 10 | path('admin/thing/list', views.admin.thing.list_api), 11 | path('admin/thing/detail', views.admin.thing.detail), 12 | path('admin/thing/create', views.admin.thing.create), 13 | path('admin/thing/update', views.admin.thing.update), 14 | path('admin/thing/delete', views.admin.thing.delete), 15 | path('admin/customer/list', views.admin.customer.list_api), 16 | path('admin/customer/create', views.admin.customer.create), 17 | path('admin/customer/update', views.admin.customer.update), 18 | path('admin/customer/delete', views.admin.customer.delete), 19 | path('admin/comment/list', views.admin.comment.list_api), 20 | path('admin/comment/create', views.admin.comment.create), 21 | path('admin/comment/update', views.admin.comment.update), 22 | path('admin/comment/delete', views.admin.comment.delete), 23 | path('admin/classification/list', views.admin.classification.list_api), 24 | path('admin/classification/create', views.admin.classification.create), 25 | path('admin/classification/update', views.admin.classification.update), 26 | path('admin/classification/delete', views.admin.classification.delete), 27 | path('admin/tag/list', views.admin.tag.list_api), 28 | path('admin/tag/create', views.admin.tag.create), 29 | path('admin/tag/update', views.admin.tag.update), 30 | path('admin/tag/delete', views.admin.tag.delete), 31 | path('admin/record/list', views.admin.record.list_api), 32 | path('admin/record/create', views.admin.record.create), 33 | path('admin/record/update', views.admin.record.update), 34 | path('admin/record/delete', views.admin.record.delete), 35 | path('admin/banner/list', views.admin.banner.list_api), 36 | path('admin/banner/create', views.admin.banner.create), 37 | path('admin/banner/update', views.admin.banner.update), 38 | path('admin/banner/delete', views.admin.banner.delete), 39 | path('admin/ad/list', views.admin.ad.list_api), 40 | path('admin/ad/create', views.admin.ad.create), 41 | path('admin/ad/update', views.admin.ad.update), 42 | path('admin/ad/delete', views.admin.ad.delete), 43 | path('admin/notice/list', views.admin.notice.list_api), 44 | path('admin/notice/create', views.admin.notice.create), 45 | path('admin/notice/update', views.admin.notice.update), 46 | path('admin/notice/delete', views.admin.notice.delete), 47 | path('admin/loginLog/list', views.admin.loginLog.list_api), 48 | path('admin/loginLog/create', views.admin.loginLog.create), 49 | path('admin/loginLog/update', views.admin.loginLog.update), 50 | path('admin/loginLog/delete', views.admin.loginLog.delete), 51 | path('admin/opLog/list', views.admin.opLog.list_api), 52 | path('admin/errorLog/list', views.admin.errorLog.list_api), 53 | path('admin/user/list', views.admin.user.list_api), 54 | path('admin/user/create', views.admin.user.create), 55 | path('admin/user/update', views.admin.user.update), 56 | path('admin/user/updatePwd', views.admin.user.updatePwd), 57 | path('admin/user/delete', views.admin.user.delete), 58 | path('admin/user/info', views.admin.user.info), 59 | path('admin/adminLogin', views.admin.user.admin_login), 60 | 61 | 62 | ] 63 | -------------------------------------------------------------------------------- /server/myapp/utils.py: -------------------------------------------------------------------------------- 1 | import datetime 2 | import hashlib 3 | import time 4 | 5 | from rest_framework.views import exception_handler 6 | 7 | from myapp.serializers import ErrorLogSerializer 8 | 9 | def get_timestamp(): 10 | return int(round(time.time() * 1000)) 11 | 12 | def md5value(key): 13 | input_name = hashlib.md5() 14 | input_name.update(key.encode("utf-8")) 15 | md5str = (input_name.hexdigest()).lower() 16 | print('计算md5:', md5str) 17 | return md5str 18 | 19 | 20 | def dict_fetchall(cursor): # cursor是执行sql_str后的记录,作入参 21 | columns = [col[0] for col in cursor.description] # 得到域的名字col[0],组成List 22 | return [ 23 | dict(zip(columns, row)) for row in cursor.fetchall() 24 | ] 25 | 26 | 27 | def get_ip(request): 28 | """ 29 | 获取请求者的IP信息 30 | """ 31 | x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR') 32 | if x_forwarded_for: 33 | ip = x_forwarded_for.split(',')[0] 34 | else: 35 | ip = request.META.get('REMOTE_ADDR') 36 | return ip 37 | 38 | 39 | def get_ua(request): 40 | """ 41 | 获取请求者的IP信息 42 | """ 43 | ua = request.META.get('HTTP_USER_AGENT') 44 | return ua[0:200] 45 | 46 | 47 | def getWeekDays(): 48 | """ 49 | 获取近一周的日期 50 | """ 51 | week_days = [] 52 | now = datetime.datetime.now() 53 | for i in range(7): 54 | day = now - datetime.timedelta(days=i) 55 | week_days.append(day.strftime('%Y-%m-%d %H:%M:%S.%f')[:10]) 56 | week_days.reverse() # 逆序 57 | return week_days 58 | 59 | 60 | def get_monday(): 61 | """ 62 | 获取本周周一日期 63 | """ 64 | now = datetime.datetime.now() 65 | monday = now - datetime.timedelta(now.weekday()) 66 | return monday.strftime('%Y-%m-%d %H:%M:%S.%f')[:10] 67 | 68 | 69 | def log_error(request, content): 70 | """ 71 | 记录错误日志 72 | """ 73 | ip = get_ip(request) 74 | method = request.method 75 | url = request.path 76 | 77 | data = { 78 | 'ip': ip, 79 | 'method': method, 80 | 'url': url, 81 | 'content': content 82 | } 83 | 84 | # 入库 85 | serializer = ErrorLogSerializer(data=data) 86 | if serializer.is_valid(): 87 | serializer.save() 88 | -------------------------------------------------------------------------------- /server/myapp/views/__init__.py: -------------------------------------------------------------------------------- 1 | from myapp.views.admin import * 2 | -------------------------------------------------------------------------------- /server/myapp/views/admin/__init__.py: -------------------------------------------------------------------------------- 1 | from myapp.views.admin.thing import * 2 | from myapp.views.admin.classification import * 3 | from myapp.views.admin.tag import * 4 | from myapp.views.admin.user import * 5 | from myapp.views.admin.comment import * 6 | from myapp.views.admin.record import * 7 | from myapp.views.admin.overview import * 8 | from myapp.views.admin.loginLog import * 9 | from myapp.views.admin.opLog import * 10 | from myapp.views.admin.errorLog import * 11 | from myapp.views.admin.banner import * 12 | from myapp.views.admin.ad import * 13 | from myapp.views.admin.notice import * 14 | from myapp.views.admin.customer import * 15 | -------------------------------------------------------------------------------- /server/myapp/views/admin/ad.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view, authentication_classes 3 | 4 | from myapp.auth.authentication import AdminTokenAuthtication 5 | from myapp.handler import APIResponse 6 | from myapp.models import Ad 7 | from myapp.permission.permission import isDemoAdminUser 8 | from myapp.serializers import AdSerializer 9 | 10 | 11 | @api_view(['GET']) 12 | def list_api(request): 13 | if request.method == 'GET': 14 | ads = Ad.objects.all().order_by('-create_time') 15 | serializer = AdSerializer(ads, many=True) 16 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 17 | 18 | 19 | @api_view(['POST']) 20 | @authentication_classes([AdminTokenAuthtication]) 21 | def create(request): 22 | if isDemoAdminUser(request): 23 | return APIResponse(code=1, msg='演示帐号无法操作') 24 | 25 | serializer = AdSerializer(data=request.data) 26 | if serializer.is_valid(): 27 | serializer.save() 28 | return APIResponse(code=0, msg='创建成功', data=serializer.data) 29 | 30 | return APIResponse(code=1, msg='创建失败') 31 | 32 | 33 | @api_view(['POST']) 34 | @authentication_classes([AdminTokenAuthtication]) 35 | def update(request): 36 | if isDemoAdminUser(request): 37 | return APIResponse(code=1, msg='演示帐号无法操作') 38 | 39 | try: 40 | pk = request.GET.get('id', -1) 41 | ad = Ad.objects.get(pk=pk) 42 | except Ad.DoesNotExist: 43 | return APIResponse(code=1, msg='对象不存在') 44 | 45 | serializer = AdSerializer(ad, data=request.data) 46 | if serializer.is_valid(): 47 | serializer.save() 48 | return APIResponse(code=0, msg='更新成功', data=serializer.data) 49 | else: 50 | print(serializer.errors) 51 | 52 | return APIResponse(code=1, msg='更新失败') 53 | 54 | 55 | @api_view(['POST']) 56 | @authentication_classes([AdminTokenAuthtication]) 57 | def delete(request): 58 | if isDemoAdminUser(request): 59 | return APIResponse(code=1, msg='演示帐号无法操作') 60 | 61 | try: 62 | ids = request.GET.get('ids') 63 | ids_arr = ids.split(',') 64 | Ad.objects.filter(id__in=ids_arr).delete() 65 | except Ad.DoesNotExist: 66 | return APIResponse(code=1, msg='对象不存在') 67 | 68 | return APIResponse(code=0, msg='删除成功') 69 | -------------------------------------------------------------------------------- /server/myapp/views/admin/banner.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view, authentication_classes 3 | 4 | from myapp.auth.authentication import AdminTokenAuthtication 5 | from myapp.handler import APIResponse 6 | from myapp.models import Banner 7 | from myapp.permission.permission import isDemoAdminUser 8 | from myapp.serializers import BannerSerializer 9 | 10 | 11 | @api_view(['GET']) 12 | def list_api(request): 13 | if request.method == 'GET': 14 | banners = Banner.objects.all().order_by('-create_time') 15 | serializer = BannerSerializer(banners, many=True) 16 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 17 | 18 | 19 | @api_view(['POST']) 20 | @authentication_classes([AdminTokenAuthtication]) 21 | def create(request): 22 | if isDemoAdminUser(request): 23 | return APIResponse(code=1, msg='演示帐号无法操作') 24 | 25 | serializer = BannerSerializer(data=request.data) 26 | if serializer.is_valid(): 27 | serializer.save() 28 | return APIResponse(code=0, msg='创建成功', data=serializer.data) 29 | 30 | return APIResponse(code=1, msg='创建失败') 31 | 32 | 33 | @api_view(['POST']) 34 | @authentication_classes([AdminTokenAuthtication]) 35 | def update(request): 36 | if isDemoAdminUser(request): 37 | return APIResponse(code=1, msg='演示帐号无法操作') 38 | 39 | try: 40 | pk = request.GET.get('id', -1) 41 | banner = Banner.objects.get(pk=pk) 42 | except Banner.DoesNotExist: 43 | return APIResponse(code=1, msg='对象不存在') 44 | 45 | serializer = BannerSerializer(banner, data=request.data) 46 | if serializer.is_valid(): 47 | serializer.save() 48 | return APIResponse(code=0, msg='更新成功', data=serializer.data) 49 | else: 50 | print(serializer.errors) 51 | 52 | return APIResponse(code=1, msg='更新失败') 53 | 54 | 55 | @api_view(['POST']) 56 | @authentication_classes([AdminTokenAuthtication]) 57 | def delete(request): 58 | if isDemoAdminUser(request): 59 | return APIResponse(code=1, msg='演示帐号无法操作') 60 | 61 | try: 62 | ids = request.GET.get('ids') 63 | ids_arr = ids.split(',') 64 | Banner.objects.filter(id__in=ids_arr).delete() 65 | except Banner.DoesNotExist: 66 | return APIResponse(code=1, msg='对象不存在') 67 | 68 | return APIResponse(code=0, msg='删除成功') 69 | -------------------------------------------------------------------------------- /server/myapp/views/admin/classification.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from django.db import connection 3 | from django.db.models import Q 4 | from rest_framework.decorators import api_view, authentication_classes 5 | 6 | from myapp.auth.authentication import AdminTokenAuthtication 7 | from myapp.handler import APIResponse 8 | from myapp.models import Classification 9 | from myapp.permission.permission import isDemoAdminUser 10 | from myapp.serializers import ClassificationSerializer 11 | from myapp.utils import dict_fetchall 12 | 13 | 14 | @api_view(['GET']) 15 | def list_api(request): 16 | if request.method == 'GET': 17 | classifications = Classification.objects.all().order_by('-create_time') 18 | serializer = ClassificationSerializer(classifications, many=True) 19 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 20 | 21 | 22 | @api_view(['POST']) 23 | @authentication_classes([AdminTokenAuthtication]) 24 | def create(request): 25 | if isDemoAdminUser(request): 26 | return APIResponse(code=1, msg='演示帐号无法操作') 27 | 28 | classification = Classification.objects.filter(title=request.data['title']) 29 | if len(classification) > 0: 30 | return APIResponse(code=1, msg='该名称已存在') 31 | 32 | serializer = ClassificationSerializer(data=request.data) 33 | if serializer.is_valid(): 34 | serializer.save() 35 | return APIResponse(code=0, msg='创建成功', data=serializer.data) 36 | 37 | return APIResponse(code=1, msg='创建失败') 38 | 39 | 40 | @api_view(['POST']) 41 | @authentication_classes([AdminTokenAuthtication]) 42 | def update(request): 43 | if isDemoAdminUser(request): 44 | return APIResponse(code=1, msg='演示帐号无法操作') 45 | 46 | try: 47 | pk = request.GET.get('id', -1) 48 | print(pk) 49 | classification = Classification.objects.get(pk=pk) 50 | except Classification.DoesNotExist: 51 | return APIResponse(code=1, msg='对象不存在') 52 | 53 | serializer = ClassificationSerializer(classification, data=request.data) 54 | if serializer.is_valid(): 55 | serializer.save() 56 | return APIResponse(code=0, msg='更新成功', data=serializer.data) 57 | 58 | return APIResponse(code=1, msg='更新失败') 59 | 60 | 61 | @api_view(['POST']) 62 | @authentication_classes([AdminTokenAuthtication]) 63 | def delete(request): 64 | if isDemoAdminUser(request): 65 | return APIResponse(code=1, msg='演示帐号无法操作') 66 | 67 | try: 68 | ids = request.GET.get('ids') 69 | ids_arr = ids.split(',') 70 | # 删除自身和自身的子孩子 71 | Classification.objects.filter(Q(id__in=ids_arr)).delete() 72 | except Classification.DoesNotExist: 73 | return APIResponse(code=1, msg='对象不存在') 74 | return APIResponse(code=0, msg='删除成功') 75 | -------------------------------------------------------------------------------- /server/myapp/views/admin/comment.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view, authentication_classes 3 | 4 | from myapp.auth.authentication import AdminTokenAuthtication 5 | from myapp.handler import APIResponse 6 | from myapp.models import Comment 7 | from myapp.permission.permission import isDemoAdminUser 8 | from myapp.serializers import CommentSerializer 9 | 10 | 11 | @api_view(['GET']) 12 | def list_api(request): 13 | if request.method == 'GET': 14 | comments = Comment.objects.select_related("thing").all().order_by('-comment_time') 15 | # print(comments) 16 | serializer = CommentSerializer(comments, many=True) 17 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 18 | 19 | 20 | @api_view(['POST']) 21 | @authentication_classes([AdminTokenAuthtication]) 22 | def create(request): 23 | if isDemoAdminUser(request): 24 | return APIResponse(code=1, msg='演示帐号无法操作') 25 | 26 | serializer = CommentSerializer(data=request.data) 27 | if serializer.is_valid(): 28 | serializer.save() 29 | return APIResponse(code=0, msg='创建成功', data=serializer.data) 30 | else: 31 | print(serializer.errors) 32 | 33 | return APIResponse(code=1, msg='创建失败') 34 | 35 | 36 | @api_view(['POST']) 37 | @authentication_classes([AdminTokenAuthtication]) 38 | def update(request): 39 | if isDemoAdminUser(request): 40 | return APIResponse(code=1, msg='演示帐号无法操作') 41 | 42 | try: 43 | pk = request.GET.get('id', -1) 44 | comments = Comment.objects.get(pk=pk) 45 | except Comment.DoesNotExist: 46 | return APIResponse(code=1, msg='对象不存在') 47 | 48 | serializer = CommentSerializer(comments, data=request.data) 49 | if serializer.is_valid(): 50 | serializer.save() 51 | return APIResponse(code=0, msg='更新成功', data=serializer.data) 52 | 53 | return APIResponse(code=1, msg='更新失败') 54 | 55 | 56 | @api_view(['POST']) 57 | @authentication_classes([AdminTokenAuthtication]) 58 | def delete(request): 59 | if isDemoAdminUser(request): 60 | return APIResponse(code=1, msg='演示帐号无法操作') 61 | 62 | try: 63 | ids = request.GET.get('ids') 64 | ids_arr = ids.split(',') 65 | Comment.objects.filter(id__in=ids_arr).delete() 66 | except Comment.DoesNotExist: 67 | return APIResponse(code=1, msg='对象不存在') 68 | 69 | return APIResponse(code=0, msg='删除成功') 70 | -------------------------------------------------------------------------------- /server/myapp/views/admin/customer.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view, authentication_classes 3 | 4 | from myapp import utils 5 | from myapp.auth.authentication import AdminTokenAuthtication 6 | from myapp.handler import APIResponse 7 | from myapp.models import Customer 8 | from myapp.permission.permission import isDemoAdminUser 9 | from myapp.serializers import CustomerSerializer 10 | 11 | 12 | @api_view(['GET']) 13 | def list_api(request): 14 | if request.method == 'GET': 15 | tags = Customer.objects.all().order_by('-create_time') 16 | serializer = CustomerSerializer(tags, many=True) 17 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 18 | 19 | 20 | @api_view(['POST']) 21 | @authentication_classes([AdminTokenAuthtication]) 22 | def create(request): 23 | if isDemoAdminUser(request): 24 | return APIResponse(code=1, msg='演示帐号无法操作') 25 | 26 | serializer = CustomerSerializer(data=request.data) 27 | if serializer.is_valid(): 28 | serializer.save() 29 | return APIResponse(code=0, msg='创建成功', data=serializer.data) 30 | else: 31 | utils.log_error(request, '参数错误') 32 | 33 | return APIResponse(code=1, msg='创建失败') 34 | 35 | 36 | @api_view(['POST']) 37 | @authentication_classes([AdminTokenAuthtication]) 38 | def update(request): 39 | if isDemoAdminUser(request): 40 | return APIResponse(code=1, msg='演示帐号无法操作') 41 | 42 | try: 43 | pk = request.GET.get('id', -1) 44 | tags = Customer.objects.get(pk=pk) 45 | except Customer.DoesNotExist: 46 | return APIResponse(code=1, msg='对象不存在') 47 | 48 | serializer = CustomerSerializer(tags, data=request.data) 49 | if serializer.is_valid(): 50 | serializer.save() 51 | return APIResponse(code=0, msg='更新成功', data=serializer.data) 52 | else: 53 | utils.log_error(request, '参数错误') 54 | 55 | return APIResponse(code=1, msg='更新失败') 56 | 57 | 58 | @api_view(['POST']) 59 | @authentication_classes([AdminTokenAuthtication]) 60 | def delete(request): 61 | if isDemoAdminUser(request): 62 | return APIResponse(code=1, msg='演示帐号无法操作') 63 | 64 | try: 65 | ids = request.GET.get('ids') 66 | ids_arr = ids.split(',') 67 | Customer.objects.filter(id__in=ids_arr).delete() 68 | except Customer.DoesNotExist: 69 | return APIResponse(code=1, msg='对象不存在') 70 | 71 | return APIResponse(code=0, msg='删除成功') 72 | -------------------------------------------------------------------------------- /server/myapp/views/admin/errorLog.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view 3 | 4 | from myapp.handler import APIResponse 5 | from myapp.models import ErrorLog 6 | from myapp.serializers import ErrorLogSerializer 7 | 8 | 9 | @api_view(['GET']) 10 | def list_api(request): 11 | if request.method == 'GET': 12 | errorLogs = ErrorLog.objects.all().order_by('-log_time') 13 | serializer = ErrorLogSerializer(errorLogs, many=True) 14 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 15 | -------------------------------------------------------------------------------- /server/myapp/views/admin/loginLog.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view, authentication_classes 3 | 4 | from myapp.auth.authentication import AdminTokenAuthtication 5 | from myapp.handler import APIResponse 6 | from myapp.models import LoginLog 7 | from myapp.permission.permission import isDemoAdminUser 8 | from myapp.serializers import LoginLogSerializer 9 | 10 | 11 | @api_view(['GET']) 12 | def list_api(request): 13 | if request.method == 'GET': 14 | loginLogs = LoginLog.objects.all().order_by('-log_time') 15 | serializer = LoginLogSerializer(loginLogs, many=True) 16 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 17 | 18 | 19 | @api_view(['POST']) 20 | def create(request): 21 | 22 | serializer = LoginLogSerializer(data=request.data) 23 | if serializer.is_valid(): 24 | serializer.save() 25 | return APIResponse(code=0, msg='创建成功', data=serializer.data) 26 | 27 | return APIResponse(code=1, msg='创建失败') 28 | 29 | 30 | @api_view(['POST']) 31 | @authentication_classes([AdminTokenAuthtication]) 32 | def update(request): 33 | try: 34 | pk = request.GET.get('id', -1) 35 | loginLogs = LoginLog.objects.get(pk=pk) 36 | except LoginLog.DoesNotExist: 37 | return APIResponse(code=1, msg='对象不存在') 38 | 39 | serializer = LoginLogSerializer(loginLogs, data=request.data) 40 | if serializer.is_valid(): 41 | serializer.save() 42 | return APIResponse(code=0, msg='更新成功', data=serializer.data) 43 | 44 | return APIResponse(code=1, msg='更新失败') 45 | 46 | 47 | @api_view(['POST']) 48 | @authentication_classes([AdminTokenAuthtication]) 49 | def delete(request): 50 | if isDemoAdminUser(request): 51 | return APIResponse(code=1, msg='演示帐号无法操作') 52 | 53 | try: 54 | ids = request.GET.get('ids') 55 | ids_arr = ids.split(',') 56 | LoginLog.objects.filter(id__in=ids_arr).delete() 57 | except LoginLog.DoesNotExist: 58 | return APIResponse(code=1, msg='对象不存在') 59 | 60 | return APIResponse(code=0, msg='删除成功') 61 | -------------------------------------------------------------------------------- /server/myapp/views/admin/notice.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view, authentication_classes 3 | 4 | from myapp.auth.authentication import AdminTokenAuthtication 5 | from myapp.handler import APIResponse 6 | from myapp.models import Notice 7 | from myapp.permission.permission import isDemoAdminUser 8 | from myapp.serializers import NoticeSerializer 9 | 10 | 11 | @api_view(['GET']) 12 | def list_api(request): 13 | if request.method == 'GET': 14 | notices = Notice.objects.all().order_by('-create_time') 15 | serializer = NoticeSerializer(notices, many=True) 16 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 17 | 18 | 19 | @api_view(['POST']) 20 | @authentication_classes([AdminTokenAuthtication]) 21 | def create(request): 22 | if isDemoAdminUser(request): 23 | return APIResponse(code=1, msg='演示帐号无法操作') 24 | 25 | serializer = NoticeSerializer(data=request.data) 26 | if serializer.is_valid(): 27 | serializer.save() 28 | return APIResponse(code=0, msg='创建成功', data=serializer.data) 29 | 30 | return APIResponse(code=1, msg='创建失败') 31 | 32 | 33 | @api_view(['POST']) 34 | @authentication_classes([AdminTokenAuthtication]) 35 | def update(request): 36 | if isDemoAdminUser(request): 37 | return APIResponse(code=1, msg='演示帐号无法操作') 38 | 39 | try: 40 | pk = request.GET.get('id', -1) 41 | notice = Notice.objects.get(pk=pk) 42 | except Notice.DoesNotExist: 43 | return APIResponse(code=1, msg='对象不存在') 44 | 45 | serializer = NoticeSerializer(notice, data=request.data) 46 | if serializer.is_valid(): 47 | serializer.save() 48 | return APIResponse(code=0, msg='更新成功', data=serializer.data) 49 | else: 50 | print(serializer.errors) 51 | 52 | return APIResponse(code=1, msg='更新失败') 53 | 54 | 55 | @api_view(['POST']) 56 | @authentication_classes([AdminTokenAuthtication]) 57 | def delete(request): 58 | if isDemoAdminUser(request): 59 | return APIResponse(code=1, msg='演示帐号无法操作') 60 | 61 | try: 62 | ids = request.GET.get('ids') 63 | ids_arr = ids.split(',') 64 | Notice.objects.filter(id__in=ids_arr).delete() 65 | except Notice.DoesNotExist: 66 | return APIResponse(code=1, msg='对象不存在') 67 | 68 | return APIResponse(code=0, msg='删除成功') 69 | -------------------------------------------------------------------------------- /server/myapp/views/admin/opLog.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view 3 | 4 | from myapp.handler import APIResponse 5 | from myapp.models import OpLog 6 | from myapp.serializers import OpLogSerializer 7 | 8 | 9 | @api_view(['GET']) 10 | def list_api(request): 11 | if request.method == 'GET': 12 | opLogs = OpLog.objects.all().order_by('-re_time')[:100] 13 | serializer = OpLogSerializer(opLogs, many=True) 14 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 15 | -------------------------------------------------------------------------------- /server/myapp/views/admin/record.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view 3 | 4 | from myapp.handler import APIResponse 5 | from myapp.models import Record 6 | from myapp.serializers import RecordSerializer 7 | 8 | 9 | @api_view(['GET']) 10 | def list_api(request): 11 | if request.method == 'GET': 12 | records = Record.objects.all() 13 | serializer = RecordSerializer(records, many=True) 14 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 15 | 16 | 17 | @api_view(['POST']) 18 | def create(request): 19 | 20 | serializer = RecordSerializer(data=request.data) 21 | if serializer.is_valid(): 22 | serializer.save() 23 | return APIResponse(code=0, msg='创建成功', data=serializer.data) 24 | 25 | return APIResponse(code=1, msg='创建失败') 26 | 27 | 28 | @api_view(['POST']) 29 | def update(request): 30 | try: 31 | pk = request.GET.get('id', -1) 32 | records = Record.objects.get(pk=pk) 33 | except Record.DoesNotExist: 34 | return APIResponse(code=1, msg='对象不存在') 35 | 36 | serializer = RecordSerializer(records, data=request.data) 37 | if serializer.is_valid(): 38 | serializer.save() 39 | return APIResponse(code=0, msg='更新成功', data=serializer.data) 40 | 41 | return APIResponse(code=1, msg='更新失败') 42 | 43 | 44 | @api_view(['POST']) 45 | def delete(request): 46 | try: 47 | ids = request.GET.get('ids') 48 | ids_arr = ids.split(',') 49 | Record.objects.filter(id__in=ids_arr).delete() 50 | except Record.DoesNotExist: 51 | return APIResponse(code=1, msg='对象不存在') 52 | 53 | return APIResponse(code=0, msg='删除成功') 54 | -------------------------------------------------------------------------------- /server/myapp/views/admin/tag.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from rest_framework.decorators import api_view, authentication_classes 3 | 4 | from myapp import utils 5 | from myapp.auth.authentication import AdminTokenAuthtication 6 | from myapp.handler import APIResponse 7 | from myapp.models import Tag 8 | from myapp.permission.permission import isDemoAdminUser 9 | from myapp.serializers import TagSerializer 10 | 11 | 12 | @api_view(['GET']) 13 | def list_api(request): 14 | if request.method == 'GET': 15 | tags = Tag.objects.all().order_by('-create_time') 16 | serializer = TagSerializer(tags, many=True) 17 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 18 | 19 | 20 | @api_view(['POST']) 21 | @authentication_classes([AdminTokenAuthtication]) 22 | def create(request): 23 | if isDemoAdminUser(request): 24 | return APIResponse(code=1, msg='演示帐号无法操作') 25 | 26 | tags = Tag.objects.filter(title=request.data['title']) 27 | if len(tags) > 0: 28 | return APIResponse(code=1, msg='该名称已存在') 29 | 30 | serializer = TagSerializer(data=request.data) 31 | if serializer.is_valid(): 32 | serializer.save() 33 | return APIResponse(code=0, msg='创建成功', data=serializer.data) 34 | else: 35 | utils.log_error(request, '参数错误') 36 | 37 | return APIResponse(code=1, msg='创建失败') 38 | 39 | 40 | @api_view(['POST']) 41 | @authentication_classes([AdminTokenAuthtication]) 42 | def update(request): 43 | if isDemoAdminUser(request): 44 | return APIResponse(code=1, msg='演示帐号无法操作') 45 | 46 | try: 47 | pk = request.GET.get('id', -1) 48 | tags = Tag.objects.get(pk=pk) 49 | except Tag.DoesNotExist: 50 | return APIResponse(code=1, msg='对象不存在') 51 | 52 | serializer = TagSerializer(tags, data=request.data) 53 | if serializer.is_valid(): 54 | serializer.save() 55 | return APIResponse(code=0, msg='更新成功', data=serializer.data) 56 | else: 57 | utils.log_error(request, '参数错误') 58 | 59 | return APIResponse(code=1, msg='更新失败') 60 | 61 | 62 | @api_view(['POST']) 63 | @authentication_classes([AdminTokenAuthtication]) 64 | def delete(request): 65 | if isDemoAdminUser(request): 66 | return APIResponse(code=1, msg='演示帐号无法操作') 67 | 68 | try: 69 | ids = request.GET.get('ids') 70 | ids_arr = ids.split(',') 71 | Tag.objects.filter(id__in=ids_arr).delete() 72 | except Tag.DoesNotExist: 73 | return APIResponse(code=1, msg='对象不存在') 74 | 75 | return APIResponse(code=0, msg='删除成功') 76 | -------------------------------------------------------------------------------- /server/readme.md: -------------------------------------------------------------------------------- 1 | ### 后端部署步骤 2 | 3 | 4 | 5 | 1. 安装mysql数据库,启动服务 6 | 2. 打开cmd命令行,进入mysql,并新建数据库 7 | ``` 8 | mysql -u root -p 9 | CREATE DATABASE IF NOT EXISTS python_yaodian DEFAULT CHARSET utf8 COLLATE utf8_general_ci; 10 | ``` 11 | 3. 恢复sql数据 12 | ``` 13 | use python_yaodian 14 | source python_yaodian.sql 15 | ``` 16 | 4. 修改settings.py中的配置信息 17 | 5. 复制资源,将upload文件夹复制到server目录下 18 | 6. 安装python 3.8 19 | 7. 安装依赖包 20 | ``` 21 | pip install -r requirements.txt -i https://mirrors.aliyun.com/pypi/simple 22 | ``` 23 | 运行项目 24 | ``` 25 | python manage.py runserver 26 | ``` 27 | 7. 后期维护改动 28 | 29 | 将修改的py文件覆盖服务器的py文件即可,重启django 30 | 31 | ### 删除数据库 32 | 33 | drop database if exists xxxx; 34 | 35 | ### 创建数据库 36 | 37 | CREATE DATABASE IF NOT EXISTS xxxx DEFAULT CHARSET utf8 COLLATE utf8_general_ci; 38 | 39 | 40 | ### 迁移数据库表 41 | 42 | ``` 43 | python manage.py makemigrations; 44 | 45 | python manage.py migrate; 46 | 47 | python manage.py makemigrations myapp; 48 | 49 | python manage.py migrate myapp; 50 | ``` 51 | 52 | ### 跨域配置 53 | 54 | django-cors-headers 55 | 56 | ### 多对多技术参考 57 | 58 | https://www.cnblogs.com/SunshineKimi/p/14140900.html 59 | 60 | ### 二级分类设计 61 | https://blog.csdn.net/weixin_47971206/article/details/124199978 62 | 63 | ### 常见问题 64 | 65 | 多对多的查询可通过related_name别名查询 66 | join查询 67 | ForeignKey的时候字段会自动加_id后缀 68 | 学习SerializerMethodField 69 | 跨域配置 django-cors-headers 70 | 数据库备份命令: 71 | mysqldump -u root -p --databases 数据库名称 > xxx.sql 72 | 数据库还原命令: 73 | source D:/xxx/xxx/xxx.sql; 74 | 创建管理员命令: 75 | insert into b_user(username,password,role,status) values('admin111',md5('admin111'),1,'0'); 76 | 77 | 接口请求频次限制 78 | 79 | 80 | ### 登录接口 81 | 82 | 调login -> 生成token 83 | 84 | 85 | ### 注意 86 | 87 | update接口的时候,如果model里面存在多对多字段,则需要设置explode 88 | 89 | 90 | 91 | 92 | 93 | 94 | -------------------------------------------------------------------------------- /server/requirements.txt: -------------------------------------------------------------------------------- 1 | Django==3.2.11 2 | PyMySQL==1.0.2 3 | djangorestframework==3.13.0 4 | django-cors-headers==3.13.0 5 | Pillow==9.1.1 6 | psutil==5.9.4 -------------------------------------------------------------------------------- /server/server/__init__.py: -------------------------------------------------------------------------------- 1 | import pymysql 2 | pymysql.install_as_MySQLdb() 3 | 4 | print("===============install pymysql==============") -------------------------------------------------------------------------------- /server/server/asgi.py: -------------------------------------------------------------------------------- 1 | """ 2 | ASGI config for server project. 3 | 4 | It exposes the ASGI callable as a module-level variable named ``application``. 5 | 6 | For more information on this file, see 7 | https://docs.djangoproject.com/en/4.1/howto/deployment/asgi/ 8 | """ 9 | 10 | import os 11 | 12 | from django.core.asgi import get_asgi_application 13 | 14 | os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'server.settings') 15 | 16 | application = get_asgi_application() 17 | -------------------------------------------------------------------------------- /server/server/urls.py: -------------------------------------------------------------------------------- 1 | """server URL Configuration 2 | 3 | The `urlpatterns` list routes URLs to views. For more information please see: 4 | https://docs.djangoproject.com/en/4.1/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: path('', 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: path('', Home.as_view(), name='home') 12 | Including another URLconf 13 | 1. Import the include() function: from django.urls import include, path 14 | 2. Add a URL to urlpatterns: path('blog/', include('blog.urls')) 15 | """ 16 | from django.conf.urls.static import static 17 | from django.contrib import admin 18 | from django.urls import path, include 19 | 20 | from server import settings 21 | 22 | urlpatterns = [ 23 | path('admin/', admin.site.urls), 24 | path('myapp/', include('myapp.urls')), 25 | ] + static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT) 26 | -------------------------------------------------------------------------------- /server/server/wsgi.py: -------------------------------------------------------------------------------- 1 | """ 2 | WSGI config for server 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/4.1/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', 'server.settings') 15 | 16 | application = get_wsgi_application() 17 | -------------------------------------------------------------------------------- /server/upload/ad/1674045266113.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/ad/1674045266113.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1674045282581.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/ad/1674045282581.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1674045308177.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/ad/1674045308177.png -------------------------------------------------------------------------------- /server/upload/ad/1674045324510.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/ad/1674045324510.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1684565423182.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/ad/1684565423182.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1684565863904.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/ad/1684565863904.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1684565876995.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/ad/1684565876995.png -------------------------------------------------------------------------------- /server/upload/avatar/1676553050529.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/avatar/1676553050529.png -------------------------------------------------------------------------------- /server/upload/avatar/1676553366217.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/avatar/1676553366217.png -------------------------------------------------------------------------------- /server/upload/avatar/1676553498600.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/avatar/1676553498600.jpeg -------------------------------------------------------------------------------- /server/upload/avatar/1676553815688.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/avatar/1676553815688.jpeg -------------------------------------------------------------------------------- /server/upload/avatar/1677240189427.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/avatar/1677240189427.png -------------------------------------------------------------------------------- /server/upload/avatar/1677982820781.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/avatar/1677982820781.jpeg -------------------------------------------------------------------------------- /server/upload/avatar/1679146350134.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/avatar/1679146350134.jpeg -------------------------------------------------------------------------------- /server/upload/avatar/1684593239449.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/avatar/1684593239449.png -------------------------------------------------------------------------------- /server/upload/avatar/1684593453676.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/avatar/1684593453676.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673963977440.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673963977440.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673964384835.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673964384835.png -------------------------------------------------------------------------------- /server/upload/banner/1673964652167.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673964652167.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965110189.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673965110189.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965198155.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673965198155.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965389141.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673965389141.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965574311.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673965574311.png -------------------------------------------------------------------------------- /server/upload/banner/1673965709533.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673965709533.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965718720.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673965718720.png -------------------------------------------------------------------------------- /server/upload/banner/1673965728690.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/banner/1673965728690.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1.jpg -------------------------------------------------------------------------------- /server/upload/cover/10.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/10.jpg -------------------------------------------------------------------------------- /server/upload/cover/11.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/11.jpg -------------------------------------------------------------------------------- /server/upload/cover/12.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/12.jpg -------------------------------------------------------------------------------- /server/upload/cover/13.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/13.jpg -------------------------------------------------------------------------------- /server/upload/cover/14.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/14.jpg -------------------------------------------------------------------------------- /server/upload/cover/15.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/15.jpg -------------------------------------------------------------------------------- /server/upload/cover/16.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/16.jpg -------------------------------------------------------------------------------- /server/upload/cover/1672749055571.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1672749055571.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1674044230851.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1674044230851.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1676186518276.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676186518276.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1676186872895.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676186872895.png -------------------------------------------------------------------------------- /server/upload/cover/1676186935002.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676186935002.png -------------------------------------------------------------------------------- /server/upload/cover/1676188277099.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676188277099.png -------------------------------------------------------------------------------- /server/upload/cover/1676188355688.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676188355688.png -------------------------------------------------------------------------------- /server/upload/cover/1676381084256.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676381084256.png -------------------------------------------------------------------------------- /server/upload/cover/1676381091144.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676381091144.png -------------------------------------------------------------------------------- /server/upload/cover/1676381097051.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676381097051.png -------------------------------------------------------------------------------- /server/upload/cover/1676381103032.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676381103032.png -------------------------------------------------------------------------------- /server/upload/cover/1676381110015.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1676381110015.png -------------------------------------------------------------------------------- /server/upload/cover/1677500674281.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677500674281.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677501266461.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677501266461.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677501470234.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677501470234.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677501544737.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677501544737.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505180730.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505180730.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505298772.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505298772.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505357042.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505357042.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505364969.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505364969.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505380921.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505380921.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505393025.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505393025.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505410960.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505410960.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505421920.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505421920.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505436478.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505436478.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505452753.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505452753.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505579480.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505579480.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505616285.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505616285.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505626565.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505626565.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505637425.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505637425.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505648826.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505648826.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505659291.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505659291.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505667178.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505667178.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505685641.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505685641.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505695894.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505695894.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505706333.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505706333.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505876732.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505876732.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505884200.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505884200.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505890616.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505890616.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505897079.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505897079.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505910282.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505910282.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505919134.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505919134.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505928898.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505928898.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505937048.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505937048.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505945207.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505945207.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505953782.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677505953782.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677588447622.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677588447622.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677677497401.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1677677497401.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1678529114670.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1678529114670.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679056479437.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679056479437.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679056528013.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679056528013.png -------------------------------------------------------------------------------- /server/upload/cover/1679056561868.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679056561868.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679056587496.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679056587496.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679058036014.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679058036014.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679058043365.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679058043365.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679229610791.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679229610791.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679229780971.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679229780971.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679229853940.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679229853940.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679229963230.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679229963230.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230045590.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230045590.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230092657.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230092657.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230147996.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230147996.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230364390.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230364390.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230543572.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230543572.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230586243.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230586243.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230641879.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230641879.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230894621.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230894621.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230935716.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230935716.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230989243.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679230989243.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231022326.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679231022326.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231052822.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679231052822.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231084420.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679231084420.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231113572.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679231113572.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231151929.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679231151929.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231184020.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679231184020.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231221940.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679231221940.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231465845.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679231465845.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232418133.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232418133.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232448857.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232448857.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232480948.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232480948.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232514230.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232514230.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232543082.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232543082.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232587234.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232587234.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232615578.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232615578.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232650073.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232650073.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232676818.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232676818.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232717023.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232717023.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232756075.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679232756075.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315151478.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315151478.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315191893.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315191893.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315240787.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315240787.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315276060.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315276060.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315310720.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315310720.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315343720.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315343720.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315365210.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315365210.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315396953.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315396953.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315437571.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315437571.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315478327.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315478327.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315749022.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315749022.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315761797.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315761797.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315803245.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315803245.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315842977.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315842977.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315883620.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315883620.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315915467.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315915467.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315954931.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315954931.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315985852.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679315985852.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316037517.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316037517.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316072493.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316072493.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316422812.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316422812.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316453474.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316453474.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316484842.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316484842.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316513693.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316513693.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316533369.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316533369.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316566135.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316566135.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316605104.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316605104.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316637185.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316637185.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316666891.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316666891.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316698087.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679316698087.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403034316.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403034316.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403102225.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403102225.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403138846.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403138846.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403158073.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403158073.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403193320.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403193320.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403239138.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403239138.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403285129.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403285129.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403331202.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403331202.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403370669.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403370669.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403403756.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403403756.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403503106.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403503106.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403631942.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403631942.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403697740.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403697740.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403736341.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679403736341.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679488092121.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1679488092121.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1684567758685.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1684567758685.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1684567804066.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1684567804066.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1684567891591.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1684567891591.png -------------------------------------------------------------------------------- /server/upload/cover/1685170107820.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685170107820.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685170987684.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685170987684.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685171442599.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685171442599.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685193225537.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685193225537.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685240184357.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685240184357.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685240191000.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685240191000.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685263611250.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685263611250.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685263639285.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685263639285.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685263674742.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685263674742.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685263699773.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685263699773.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685263937295.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685263937295.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685263997721.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685263997721.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264026586.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264026586.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264114074.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264114074.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264135851.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264135851.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264163148.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264163148.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264250423.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264250423.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264270509.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264270509.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264295265.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264295265.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264355503.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264355503.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264420316.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264420316.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264443807.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264443807.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264535734.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264535734.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264562903.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264562903.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264586283.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264586283.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264771439.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264771439.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264800237.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264800237.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264830789.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264830789.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685264878356.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685264878356.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1685265162834.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1685265162834.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1691491016343.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/1691491016343.jpeg -------------------------------------------------------------------------------- /server/upload/cover/17.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/17.jpg -------------------------------------------------------------------------------- /server/upload/cover/18.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/18.jpg -------------------------------------------------------------------------------- /server/upload/cover/19.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/19.jpg -------------------------------------------------------------------------------- /server/upload/cover/2.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/2.jpg -------------------------------------------------------------------------------- /server/upload/cover/20.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/20.jpg -------------------------------------------------------------------------------- /server/upload/cover/21.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/21.jpg -------------------------------------------------------------------------------- /server/upload/cover/22.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/22.jpg -------------------------------------------------------------------------------- /server/upload/cover/23.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/23.jpg -------------------------------------------------------------------------------- /server/upload/cover/24.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/24.jpg -------------------------------------------------------------------------------- /server/upload/cover/25.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/25.jpg -------------------------------------------------------------------------------- /server/upload/cover/26.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/26.jpg -------------------------------------------------------------------------------- /server/upload/cover/27.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/27.jpg -------------------------------------------------------------------------------- /server/upload/cover/28.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/28.jpg -------------------------------------------------------------------------------- /server/upload/cover/29.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/29.jpg -------------------------------------------------------------------------------- /server/upload/cover/3.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/3.jpg -------------------------------------------------------------------------------- /server/upload/cover/30.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/30.jpg -------------------------------------------------------------------------------- /server/upload/cover/4.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/4.jpg -------------------------------------------------------------------------------- /server/upload/cover/5.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/5.jpg -------------------------------------------------------------------------------- /server/upload/cover/6.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/6.jpg -------------------------------------------------------------------------------- /server/upload/cover/7.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/7.jpg -------------------------------------------------------------------------------- /server/upload/cover/8.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/8.jpg -------------------------------------------------------------------------------- /server/upload/cover/9.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/server/upload/cover/9.jpg -------------------------------------------------------------------------------- /web/.eslintignore: -------------------------------------------------------------------------------- 1 | 2 | *.sh 3 | node_modules 4 | *.md 5 | *.woff 6 | *.ttf 7 | .vscode 8 | .idea 9 | dist 10 | /public 11 | /docs 12 | .husky 13 | .local 14 | /bin 15 | Dockerfile 16 | -------------------------------------------------------------------------------- /web/.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | root: true, 3 | env: { 4 | browser: true, 5 | node: true, 6 | es6: true, 7 | }, 8 | parser: 'vue-eslint-parser', 9 | parserOptions: { 10 | parser: '@typescript-eslint/parser', 11 | ecmaVersion: 2020, 12 | sourceType: 'module', 13 | jsxPragma: 'React', 14 | ecmaFeatures: { 15 | jsx: true, 16 | }, 17 | }, 18 | extends: ['plugin:vue/vue3-recommended', 'plugin:@typescript-eslint/recommended', 'plugin:prettier/recommended'], 19 | rules: { 20 | 'vue/script-setup-uses-vars': 'error', 21 | '@typescript-eslint/ban-ts-ignore': 'off', 22 | '@typescript-eslint/explicit-function-return-type': 'off', 23 | '@typescript-eslint/no-explicit-any': 'off', 24 | '@typescript-eslint/no-var-requires': 'off', 25 | '@typescript-eslint/no-empty-function': 'off', 26 | 'vue/custom-event-name-casing': 'off', 27 | 'no-use-before-define': 'off', 28 | '@typescript-eslint/no-use-before-define': 'off', 29 | '@typescript-eslint/ban-ts-comment': 'off', 30 | '@typescript-eslint/ban-types': 'off', 31 | '@typescript-eslint/no-non-null-assertion': 'off', 32 | '@typescript-eslint/explicit-module-boundary-types': 'off', 33 | '@typescript-eslint/no-unused-vars': 'off', 34 | 'no-unused-vars': 'off', 35 | 'space-before-function-paren': 'off', 36 | 37 | 'vue/attributes-order': 'off', 38 | 'vue/one-component-per-file': 'off', 39 | 'vue/html-closing-bracket-newline': 'off', 40 | 'vue/max-attributes-per-line': 'off', 41 | 'vue/multiline-html-element-content-newline': 'off', 42 | 'vue/singleline-html-element-content-newline': 'off', 43 | 'vue/attribute-hyphenation': 'off', 44 | 'vue/require-default-prop': 'off', 45 | 'vue/require-explicit-emits': 'off', 46 | 'vue/html-self-closing': [ 47 | 'error', 48 | { 49 | html: { 50 | void: 'always', 51 | normal: 'never', 52 | component: 'always', 53 | }, 54 | svg: 'always', 55 | math: 'always', 56 | }, 57 | ], 58 | 'vue/multi-word-component-names': 'off', 59 | }, 60 | }; 61 | -------------------------------------------------------------------------------- /web/.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | .DS_Store 3 | dist 4 | dist-ssr 5 | .local 6 | .history 7 | # local env files 8 | .env.local 9 | .env.*.local 10 | .eslintcache 11 | .github 12 | .husky 13 | .vscode 14 | 15 | # Log files 16 | npm-debug.log* 17 | yarn-debug.log* 18 | yarn-error.log* 19 | pnpm-debug.log* 20 | pnpm-lock.yaml* 21 | 22 | # Editor directories and files 23 | .idea 24 | # .vscode 25 | *.suo 26 | *.ntvs* 27 | *.njsproj 28 | *.sln 29 | *.sw? 30 | ./packages 31 | ./history 32 | -------------------------------------------------------------------------------- /web/.stylelintignore: -------------------------------------------------------------------------------- 1 | /dist/* 2 | /public/* 3 | public/* 4 | -------------------------------------------------------------------------------- /web/README.md: -------------------------------------------------------------------------------- 1 | # 学习文档 2 | 3 | #### 部署步骤 4 | 5 | 1. 修改constants.ts中的BASE_URL 6 | 2. vite build 7 | 3. 将dist部署到nginx 8 | 9 | 10 | #### 配置解释 11 | 12 | 1. env.development 开发环境配置 13 | 2. eslintrc.js 代码规范化提示 14 | 3. vite.config.js vite 开发服务器配置 15 | 16 | #### 常见问题 17 | 18 | ##### 变量 19 | https://blog.csdn.net/qq_41636947/article/details/117907448 20 | 21 | ##### antd的css引入方式 22 | 在index.html里面引入的cdn 23 | 24 | ##### cdn 25 | https://cdn.jsdelivr.net/npm/ant-design-vue@3.2.20/dist/ 26 | https://cdn.staticfile.org/ant-design-vue/3.2.20/antd.min.css 27 | 28 | #### public文件夹内容在build后会自动打到dist中 29 | -------------------------------------------------------------------------------- /web/build/constant.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * @name Config 3 | * @description 项目配置 4 | */ 5 | 6 | 7 | // 本地服务端口 8 | export const VITE_PORT = 3000; 9 | 10 | // 包依赖分析 11 | export const ANALYSIS = true; 12 | 13 | // 代码压缩 14 | export const COMPRESSION = true; 15 | 16 | -------------------------------------------------------------------------------- /web/build/vite/plugins/autoImport.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * @name AutoImportDeps 3 | * @description 按需加载,自动引入 4 | */ 5 | import AutoImport from 'unplugin-auto-import/vite'; 6 | // import { AntDesignVueResolver} from 'unplugin-vue-components/resolvers'; 7 | 8 | export const AutoImportDeps = () => { 9 | return AutoImport({ 10 | dts: 'types/auto-imports.d.ts', 11 | imports: [ 12 | 'vue', 13 | 'pinia', 14 | 'vue-router', 15 | { 16 | '@vueuse/core': [], 17 | }, 18 | { 19 | 'naive-ui': ['useDialog', 'useMessage', 'useNotification', 'useLoadingBar'], 20 | }, 21 | ], 22 | resolvers: [ 23 | // AntDesignVueResolver(), 24 | ], 25 | }); 26 | }; 27 | -------------------------------------------------------------------------------- /web/build/vite/plugins/component.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * @name AutoRegistryComponents 3 | * @description 按需加载,自动引入组件 4 | */ 5 | import Components from 'unplugin-vue-components/vite'; 6 | import { 7 | ElementPlusResolver, 8 | VueUseComponentsResolver, 9 | AntDesignVueResolver, 10 | TDesignResolver, 11 | NaiveUiResolver, 12 | } from 'unplugin-vue-components/resolvers'; 13 | export const AutoRegistryComponents = () => { 14 | return Components({ 15 | dirs: ['src/components'], 16 | extensions: ['vue'], 17 | deep: true, 18 | dts: 'types/components.d.ts', 19 | directoryAsNamespace: false, 20 | globalNamespaces: [], 21 | directives: true, 22 | importPathTransform: (v) => v, 23 | allowOverrides: false, 24 | include: [/\.vue$/, /\.vue\?vue/], 25 | exclude: [/[\\/]node_modules[\\/]/, /[\\/]\.git[\\/]/, /[\\/]\.nuxt[\\/]/], 26 | resolvers: [ 27 | ElementPlusResolver(), 28 | VueUseComponentsResolver(), 29 | AntDesignVueResolver(), 30 | TDesignResolver({ 31 | library: 'vue-next', 32 | }), 33 | NaiveUiResolver(), 34 | ], 35 | }); 36 | }; 37 | -------------------------------------------------------------------------------- /web/build/vite/plugins/compress.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * @name ConfigCompressPlugin 3 | * @description 开启.gz压缩 4 | */ 5 | import viteCompression from 'vite-plugin-compression'; 6 | import { COMPRESSION } from '../../constant'; 7 | 8 | export const ConfigCompressPlugin = () => { 9 | if (COMPRESSION) { 10 | return viteCompression({ 11 | verbose: true, // 默认即可 12 | disable: false, //开启压缩(不禁用),默认即可 13 | deleteOriginFile: false, //删除源文件 14 | threshold: 10240, //压缩前最小文件大小 15 | algorithm: 'gzip', //压缩算法 16 | ext: '.gz', //文件类型 17 | }); 18 | } 19 | return []; 20 | }; 21 | -------------------------------------------------------------------------------- /web/build/vite/plugins/imagemin.ts: -------------------------------------------------------------------------------- 1 | import viteImagemin from 'vite-plugin-imagemin'; 2 | 3 | export function ConfigImageminPlugin() { 4 | const plugin = viteImagemin({ 5 | gifsicle: { 6 | optimizationLevel: 7, 7 | interlaced: false, 8 | }, 9 | mozjpeg: { 10 | quality: 20, 11 | }, 12 | optipng: { 13 | optimizationLevel: 7, 14 | }, 15 | pngquant: { 16 | quality: [0.8, 0.9], 17 | speed: 4, 18 | }, 19 | svgo: { 20 | plugins: [ 21 | { 22 | name: 'removeViewBox', 23 | }, 24 | { 25 | name: 'removeEmptyAttrs', 26 | active: false, 27 | }, 28 | ], 29 | }, 30 | }); 31 | return plugin; 32 | } 33 | -------------------------------------------------------------------------------- /web/build/vite/plugins/index.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * @name createVitePlugins 3 | * @description 封装plugins数组统一调用 4 | */ 5 | import {PluginOption} from 'vite'; 6 | import vue from '@vitejs/plugin-vue'; 7 | import vueJsx from '@vitejs/plugin-vue-jsx'; 8 | import {AutoImportDeps} from './autoImport'; 9 | import {ConfigCompressPlugin} from './compress'; 10 | import {ConfigRestartPlugin} from './restart'; 11 | import {ConfigProgressPlugin} from './progress'; 12 | import {ConfigVisualizerConfig} from "./visualizer"; 13 | 14 | export function createVitePlugins(isBuild: boolean) { 15 | const vitePlugins = [ 16 | // vue支持 17 | vue(), 18 | // JSX支持 19 | vueJsx(), 20 | // setup语法糖组件名支持 21 | // vueSetupExtend(), 22 | // 提供https证书 23 | // VitePluginCertificate({ 24 | // source: 'coding', 25 | // }) as PluginOption, 26 | ]; 27 | 28 | // 自动按需引入组件 29 | // vitePlugins.push(AutoRegistryComponents()); 30 | 31 | // 自动按需引入依赖 32 | vitePlugins.push(AutoImportDeps()); 33 | 34 | // 自动生成路由 35 | // vitePlugins.push(ConfigPagesPlugin()); 36 | 37 | // 开启.gz压缩 rollup-plugin-gzip 38 | vitePlugins.push(ConfigCompressPlugin()); 39 | 40 | // 监听配置文件改动重启 41 | vitePlugins.push(ConfigRestartPlugin()); 42 | 43 | // 构建时显示进度条 44 | vitePlugins.push(ConfigProgressPlugin()); 45 | 46 | // 构建时显示进度条 47 | vitePlugins.push(ConfigVisualizerConfig()); 48 | 49 | 50 | return vitePlugins; 51 | } 52 | -------------------------------------------------------------------------------- /web/build/vite/plugins/progress.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * @name ConfigProgressPlugin 3 | * @description 构建显示进度条 4 | */ 5 | 6 | import progress from 'vite-plugin-progress'; 7 | export const ConfigProgressPlugin = () => { 8 | return progress(); 9 | }; 10 | -------------------------------------------------------------------------------- /web/build/vite/plugins/restart.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * @name ConfigRestartPlugin 3 | * @description 监听配置文件修改自动重启Vite 4 | */ 5 | import ViteRestart from 'vite-plugin-restart'; 6 | export const ConfigRestartPlugin = () => { 7 | return ViteRestart({ 8 | restart: ['*.config.[jt]s', '**/config/*.[jt]s'], 9 | }); 10 | }; 11 | -------------------------------------------------------------------------------- /web/build/vite/plugins/unocss.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * @name ConfigUnocssPlugin 3 | * @description 监听配置文件修改自动重启Vite 4 | */ 5 | 6 | // Unocss 7 | import Unocss from 'unocss/vite'; 8 | 9 | export const ConfigUnocssPlugin = () => { 10 | return Unocss(); 11 | }; 12 | -------------------------------------------------------------------------------- /web/build/vite/plugins/visualizer.ts: -------------------------------------------------------------------------------- 1 | import visualizer from 'rollup-plugin-visualizer'; 2 | import { ANALYSIS } from '../../constant'; 3 | 4 | export function ConfigVisualizerConfig() { 5 | if (ANALYSIS) { 6 | return visualizer({ 7 | filename: 'dist/report.html', 8 | open: true, 9 | gzipSize: true, 10 | emitFile: false 11 | }); 12 | } 13 | return []; 14 | } 15 | -------------------------------------------------------------------------------- /web/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 药店信息管理 9 | 10 | 11 | 13 | 14 | 15 | 16 | 17 |
18 | 19 | 20 | 21 | -------------------------------------------------------------------------------- /web/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "my-web-app", 3 | "version": "0.1.2", 4 | "author": "lengqin1024", 5 | "scripts": { 6 | "dev": "vite --mode development", 7 | "build": "vite build", 8 | "preview": "vite preview" 9 | }, 10 | "dependencies": { 11 | "@ant-design/icons-vue": "^6.1.0", 12 | "@vueuse/components": "^9.10.0", 13 | "@vueuse/core": "^9.10.0", 14 | "ant-design-vue": "^3.2.20", 15 | "axios": "^1.2.2", 16 | "pinia": "^2.0.28", 17 | "pinia-plugin-persistedstate": "^3.0.2", 18 | "qs": "^6.11.0", 19 | "vue": "^3.2.45", 20 | "vue-router": "^4.1.6" 21 | }, 22 | "devDependencies": { 23 | "@types/qs": "^6.9.7", 24 | "@typescript-eslint/eslint-plugin": "^5.48.0", 25 | "@typescript-eslint/parser": "^5.48.0", 26 | "@vitejs/plugin-vue": "^4.0.0", 27 | "@vitejs/plugin-vue-jsx": "^3.0.0", 28 | "autoprefixer": "^10.4.13", 29 | "eslint": "8.22.0", 30 | "eslint-config-prettier": "^8.6.0", 31 | "eslint-define-config": "^1.13.0", 32 | "eslint-plugin-prettier": "^4.2.1", 33 | "eslint-plugin-vue": "^9.8.0", 34 | "less": "^4.1.3", 35 | "less-loader": "^11.1.0", 36 | "postcss": "^8.4.21", 37 | "postcss-html": "^1.5.0", 38 | "postcss-less": "^6.0.0", 39 | "prettier": "^2.8.3", 40 | "rollup-plugin-visualizer": "^5.9.0", 41 | "stylelint": "^14.16.1", 42 | "stylelint-config-standard": "^29.0.0", 43 | "stylelint-order": "^6.0.1", 44 | "typescript": "4.9.4", 45 | "unplugin-auto-import": "^0.12.2", 46 | "vite": "^4.0.3", 47 | "vite-plugin-compression": "^0.5.1", 48 | "vite-plugin-progress": "^0.0.6", 49 | "vite-plugin-restart": "^0.3.1" 50 | } 51 | } 52 | -------------------------------------------------------------------------------- /web/postcss.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | plugins: { 3 | autoprefixer: {}, 4 | }, 5 | }; 6 | -------------------------------------------------------------------------------- /web/prettier.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | printWidth: 140, 3 | semi: true, 4 | vueIndentScriptAndStyle: true, 5 | singleQuote: true, 6 | trailingComma: 'all', 7 | proseWrap: 'never', 8 | htmlWhitespaceSensitivity: 'strict', 9 | endOfLine: 'auto', 10 | }; 11 | -------------------------------------------------------------------------------- /web/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/public/favicon.ico -------------------------------------------------------------------------------- /web/public/images/admin-login-bg.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/public/images/admin-login-bg.jpg -------------------------------------------------------------------------------- /web/public/images/bg2.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/public/images/bg2.jpg -------------------------------------------------------------------------------- /web/public/images/demo.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/public/images/demo.jpg -------------------------------------------------------------------------------- /web/src/App.vue: -------------------------------------------------------------------------------- 1 | 6 | 11 | 12 | 19 | -------------------------------------------------------------------------------- /web/src/api/ad.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/admin/ad/list', 5 | create = '/myapp/admin/ad/create', 6 | update = '/myapp/admin/ad/update', 7 | delete = '/myapp/admin/ad/delete', 8 | } 9 | 10 | const listApi = async (params: any) => 11 | get({url: URL.list, params: params, data: {}, headers: {}}); 12 | const createApi = async (data: any) => 13 | post({ 14 | url: URL.create, 15 | params: {}, 16 | data: data, 17 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 18 | }); 19 | const updateApi = async (params: any, data: any) => 20 | post({ 21 | url: URL.update, 22 | params: params, 23 | data: data, 24 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 25 | }); 26 | const deleteApi = async (params: any) => 27 | post({url: URL.delete, params: params, headers: {}}); 28 | 29 | export {listApi, createApi, updateApi, deleteApi}; 30 | -------------------------------------------------------------------------------- /web/src/api/classification.ts: -------------------------------------------------------------------------------- 1 | import { get, post } from '/@/utils/http/axios'; 2 | enum URL { 3 | list = '/myapp/admin/classification/list', 4 | create = '/myapp/admin/classification/create', 5 | update = '/myapp/admin/classification/update', 6 | delete = '/myapp/admin/classification/delete', 7 | } 8 | 9 | const listApi = async (params: any) => get({ url: URL.list, params: params, data: {}, headers: {} }); 10 | const createApi = async (data: any) => 11 | post({ url: URL.create, params: {}, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 12 | const updateApi = async (params: any, data: any) => 13 | post({ url: URL.update, params: params, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 14 | const deleteApi = async (params: any) => post({ url: URL.delete, params: params, headers: {} }); 15 | 16 | export { listApi, createApi, updateApi, deleteApi }; 17 | -------------------------------------------------------------------------------- /web/src/api/comment.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/admin/comment/list', 5 | create = '/myapp/admin/comment/create', 6 | delete = '/myapp/admin/comment/delete', 7 | listThingComments = '/api/comment/listThingComments', 8 | listUserComments = '/api/comment/listUserComments', 9 | like = '/api/comment/like' 10 | } 11 | 12 | const listApi = async (params: any) => get({url: URL.list, params: params, data: {}, headers: {}}); 13 | const createApi = async (data: any) => post({ 14 | url: URL.create, 15 | params: {}, 16 | data: data, 17 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 18 | }); 19 | const deleteApi = async (params: any) => post({url: URL.delete, params: params, headers: {}}); 20 | const listThingCommentsApi = async (params: any) => get({url: URL.listThingComments, params: params, data: {}, headers: {}}); 21 | const listUserCommentsApi = async (params: any) => get({url: URL.listUserComments, params: params, data: {}, headers: {}}); 22 | const likeApi = async (params: any) => post({url: URL.like, params: params, headers: {}}); 23 | 24 | export {listApi, createApi, deleteApi, listThingCommentsApi, listUserCommentsApi, likeApi}; 25 | -------------------------------------------------------------------------------- /web/src/api/customer.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/admin/customer/list', 5 | create = '/myapp/admin/customer/create', 6 | update = '/myapp/admin/customer/update', 7 | delete = '/myapp/admin/customer/delete', 8 | } 9 | 10 | const listApi = async (params: any) => 11 | get({url: URL.list, params: params, data: {}, headers: {}}); 12 | const createApi = async (data: any) => 13 | post({ 14 | url: URL.create, 15 | params: {}, 16 | data: data, 17 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 18 | }); 19 | const updateApi = async (params: any, data: any) => 20 | post({ 21 | url: URL.update, 22 | params: params, 23 | data: data, 24 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 25 | }); 26 | const deleteApi = async (params: any) => 27 | post({url: URL.delete, params: params, headers: {}}); 28 | 29 | export {listApi, createApi, updateApi, deleteApi}; 30 | -------------------------------------------------------------------------------- /web/src/api/log.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | loginLogList = '/myapp/admin/loginLog/list', 5 | opLogList = '/myapp/admin/opLog/list', 6 | errorLogList = '/myapp/admin/errorLog/list', 7 | } 8 | 9 | const listLoginLogApi = async (params: any) => 10 | get({url: URL.loginLogList, params: params, data: {}, headers: {}}); 11 | const listOpLogListApi = async (params: any) => 12 | get({url: URL.opLogList, params: params, data: {}, headers: {}}); 13 | const listErrorLogListApi = async (params: any) => 14 | get({url: URL.errorLogList, params: params, data: {}, headers: {}}); 15 | 16 | export {listLoginLogApi, listOpLogListApi, listErrorLogListApi}; 17 | -------------------------------------------------------------------------------- /web/src/api/notice.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/admin/notice/list', 5 | create = '/myapp/admin/notice/create', 6 | update = '/myapp/admin/notice/update', 7 | delete = '/myapp/admin/notice/delete', 8 | } 9 | 10 | const listApi = async (params: any) => 11 | get({url: URL.list, params: params, data: {}, headers: {}}); 12 | const createApi = async (data: any) => 13 | post({ 14 | url: URL.create, 15 | params: {}, 16 | data: data, 17 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 18 | }); 19 | const updateApi = async (params: any, data: any) => 20 | post({ 21 | url: URL.update, 22 | params: params, 23 | data: data, 24 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 25 | }); 26 | const deleteApi = async (params: any) => 27 | post({url: URL.delete, params: params, headers: {}}); 28 | 29 | export {listApi, createApi, updateApi, deleteApi}; 30 | -------------------------------------------------------------------------------- /web/src/api/order.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/admin/order/list', 5 | create = '/myapp/admin/order/create', 6 | update = '/myapp/admin/order/update', 7 | delete = '/myapp/admin/order/delete', 8 | cancel = '/myapp/admin/order/cancel_order', 9 | cancelUserOrder = '/api/order/cancelUserOrder', 10 | userOrderList = '/api/order/userOrderList', 11 | } 12 | 13 | const listApi = async (params: any) => 14 | get({url: URL.list, params: params, data: {}, headers: {}}); 15 | const userOrderListApi = async (params: any) => 16 | get({url: URL.userOrderList, params: params, data: {}, headers: {}}); 17 | 18 | const createApi = async (data: any) => 19 | post({ 20 | url: URL.create, 21 | params: {}, 22 | data: data, 23 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 24 | }); 25 | const updateApi = async (params: any, data: any) => 26 | post({ 27 | url: URL.update, 28 | params: params, 29 | data: data, 30 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 31 | }); 32 | const deleteApi = async (params: any) => 33 | post({url: URL.delete, params: params, headers: {}}); 34 | 35 | const cancelApi = async (params: any) => 36 | post({url: URL.cancel, params: params, headers: {}}); 37 | 38 | const cancelUserOrderApi = async (params: any) => 39 | post({url: URL.cancelUserOrder, params: params, headers: {}}); 40 | 41 | export {listApi, userOrderListApi, createApi, updateApi, deleteApi, cancelApi, cancelUserOrderApi}; 42 | -------------------------------------------------------------------------------- /web/src/api/overview.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/admin/overview/count', 5 | sysInfo= '/myapp/admin/overview/sysInfo', 6 | } 7 | 8 | const listApi = async (params: any) => 9 | get({url: URL.list, params: params, data: {}, headers: {}}); 10 | 11 | 12 | const sysInfoApi = async (params: any) => 13 | get({url: URL.sysInfo, params: params, data: {}, headers: {}}); 14 | 15 | export {listApi, sysInfoApi}; 16 | -------------------------------------------------------------------------------- /web/src/api/tag.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/admin/tag/list', 5 | create = '/myapp/admin/tag/create', 6 | update = '/myapp/admin/tag/update', 7 | delete = '/myapp/admin/tag/delete', 8 | } 9 | 10 | const listApi = async (params: any) => 11 | get({url: URL.list, params: params, data: {}, headers: {}}); 12 | const createApi = async (data: any) => 13 | post({ 14 | url: URL.create, 15 | params: {}, 16 | data: data, 17 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 18 | }); 19 | const updateApi = async (params: any, data: any) => 20 | post({ 21 | url: URL.update, 22 | params: params, 23 | data: data, 24 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 25 | }); 26 | const deleteApi = async (params: any) => 27 | post({url: URL.delete, params: params, headers: {}}); 28 | 29 | export {listApi, createApi, updateApi, deleteApi}; 30 | -------------------------------------------------------------------------------- /web/src/api/thing.ts: -------------------------------------------------------------------------------- 1 | // 权限问题后期增加 2 | import { get, post } from '/@/utils/http/axios'; 3 | import { UserState } from '/@/store/modules/user/types'; 4 | // import axios from 'axios'; 5 | enum URL { 6 | list = '/myapp/admin/thing/list', 7 | create = '/myapp/admin/thing/create', 8 | update = '/myapp/admin/thing/update', 9 | delete = '/myapp/admin/thing/delete', 10 | detail = '/api/thing/detail', 11 | } 12 | 13 | const listApi = async (params: any) => get({ url: URL.list, params: params, data: {}, headers: {} }); 14 | const createApi = async (data: any) => 15 | post({ url: URL.create, params: {}, data: data, timeout:20000, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 16 | const updateApi = async (params:any, data: any) => 17 | post({ url: URL.update,params: params, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 18 | const deleteApi = async (params: any) => post({ url: URL.delete, params: params, headers: {} }); 19 | const detailApi = async (params: any) => get({ url: URL.detail, params: params, headers: {} }); 20 | 21 | export { listApi, createApi, updateApi, deleteApi, detailApi }; 22 | -------------------------------------------------------------------------------- /web/src/api/user.ts: -------------------------------------------------------------------------------- 1 | // 权限问题后期增加 2 | import { get, post } from '/@/utils/http/axios'; 3 | import { UserState } from '/@/store/modules/user/types'; 4 | // import axios from 'axios'; 5 | enum URL { 6 | login = '/myapp/admin/adminLogin', 7 | userList = '/myapp/admin/user/list', 8 | detail = '/api/user/detail', 9 | create = '/myapp/admin/user/create', 10 | update = '/myapp/admin/user/update', 11 | delete = '/myapp/admin/user/delete', 12 | userLogin = '/api/user/userLogin', 13 | userRegister = '/api/user/userRegister', 14 | updateUserPwd = '/api/user/updatePwd', 15 | updateUserInfo = '/api/user/updateUserInfo' 16 | } 17 | interface LoginRes { 18 | token: string; 19 | } 20 | 21 | export interface LoginData { 22 | username: string; 23 | password: string; 24 | } 25 | 26 | const loginApi = async (data: LoginData) => post({ url: URL.login, data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 27 | const listApi = async (params: any) => get({ url: URL.userList, params: params, data: {}, headers: {} }); 28 | const detailApi = async (params: any) => get({ url: URL.detail, params: params, data: {}, headers: {} }); 29 | const createApi = async (data: any) => post({ url: URL.create, params: {}, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 30 | const updateApi = async (params: any, data: any) => post({ url: URL.update,params: params, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 31 | const deleteApi = async (params: any) => post({ url: URL.delete, params: params, headers: {} }); 32 | const userLoginApi = async (data: LoginData) => post({ url: URL.userLogin, data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 33 | const userRegisterApi = async (data: any) => post({ url: URL.userRegister, params: {}, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 34 | const updateUserPwdApi = async (params: any) => post({ url: URL.updateUserPwd, params: params }); 35 | const updateUserInfoApi = async (data: any) => post({ url: URL.updateUserInfo, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 36 | 37 | export { loginApi, listApi, detailApi, createApi, updateApi, deleteApi, userLoginApi, userRegisterApi, updateUserPwdApi, updateUserInfoApi}; 38 | -------------------------------------------------------------------------------- /web/src/assets/fonts/Blimone-ExtraBold.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/fonts/Blimone-ExtraBold.woff -------------------------------------------------------------------------------- /web/src/assets/fonts/Blimone-ExtraLight.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/fonts/Blimone-ExtraLight.woff -------------------------------------------------------------------------------- /web/src/assets/fonts/Blimone-Light.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/fonts/Blimone-Light.woff -------------------------------------------------------------------------------- /web/src/assets/fonts/Blimone-Regular.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/fonts/Blimone-Regular.woff -------------------------------------------------------------------------------- /web/src/assets/icons/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/icons/logo.png -------------------------------------------------------------------------------- /web/src/assets/icons/svg/github.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /web/src/assets/icons/svg/marks.svg: -------------------------------------------------------------------------------- 1 | 2 | 4 | -------------------------------------------------------------------------------- /web/src/assets/icons/svg/test.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /web/src/assets/icons/svg/ts.svg: -------------------------------------------------------------------------------- 1 | 2 | 5 | -------------------------------------------------------------------------------- /web/src/assets/icons/svg/twitter.svg: -------------------------------------------------------------------------------- 1 | 2 | 4 | -------------------------------------------------------------------------------- /web/src/assets/images/Play.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/images/Play.png -------------------------------------------------------------------------------- /web/src/assets/images/add.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/address-right-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /web/src/assets/images/ali-pay-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /web/src/assets/images/avatar.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/images/avatar.jpg -------------------------------------------------------------------------------- /web/src/assets/images/banner-02.webp: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/images/banner-02.webp -------------------------------------------------------------------------------- /web/src/assets/images/cart-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /web/src/assets/images/clear-search.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/code-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/delete-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | -------------------------------------------------------------------------------- /web/src/assets/images/ebook-download-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/k-logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/images/k-logo.png -------------------------------------------------------------------------------- /web/src/assets/images/login-banner.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/images/login-banner.png -------------------------------------------------------------------------------- /web/src/assets/images/login.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/images/login.png -------------------------------------------------------------------------------- /web/src/assets/images/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/images/logo.png -------------------------------------------------------------------------------- /web/src/assets/images/mail-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/message-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /web/src/assets/images/order-address-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/order-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /web/src/assets/images/order-point-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/order-thing-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /web/src/assets/images/pwd-hidden.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | -------------------------------------------------------------------------------- /web/src/assets/images/pwd-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/qunerweima.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/images/qunerweima.jpg -------------------------------------------------------------------------------- /web/src/assets/images/read-online-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | -------------------------------------------------------------------------------- /web/src/assets/images/recommend-hover.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /web/src/assets/images/register-name.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /web/src/assets/images/search-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /web/src/assets/images/searchIcon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /web/src/assets/images/setting-card-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | -------------------------------------------------------------------------------- /web/src/assets/images/setting-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /web/src/assets/images/setting-msg-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /web/src/assets/images/setting-safe-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/share-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /web/src/assets/images/tel-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /web/src/assets/images/want-read-hover.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /web/src/assets/images/wb-share.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /web/src/assets/images/wx-pay-icon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /web/src/assets/styles/base.less: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_yaodian/697c83d6791da509b459c5487fbb59c3bf150876/web/src/assets/styles/base.less -------------------------------------------------------------------------------- /web/src/core/bootstrap.js: -------------------------------------------------------------------------------- 1 | // localStorage恢复到内存 2 | 3 | import {useUserStore} from "/@/store"; 4 | import {USER_ID, USER_NAME, USER_TOKEN, ADMIN_USER_ID, ADMIN_USER_NAME, ADMIN_USER_TOKEN} from "/@/store/constants"; 5 | 6 | export default function Initializer () { 7 | const userStore = useUserStore() 8 | userStore.$patch((state)=>{ 9 | state.user_id = localStorage.getItem(USER_ID) 10 | state.user_name = localStorage.getItem(USER_NAME) 11 | state.user_token = localStorage.getItem(USER_TOKEN) 12 | 13 | state.admin_user_id = localStorage.getItem(ADMIN_USER_ID) 14 | state.admin_user_name = localStorage.getItem(ADMIN_USER_NAME) 15 | state.admin_user_token = localStorage.getItem(ADMIN_USER_TOKEN) 16 | console.log('恢复store完毕==>', state) 17 | }) 18 | 19 | } 20 | -------------------------------------------------------------------------------- /web/src/main.js: -------------------------------------------------------------------------------- 1 | import { createApp } from 'vue'; 2 | import App from './App.vue'; 3 | import router from './router'; 4 | import piniaStore from './store'; 5 | 6 | import bootstrap from './core/bootstrap'; 7 | import '/@/styles/reset.less'; 8 | import '/@/styles/index.less'; 9 | import Antd from 'ant-design-vue'; 10 | 11 | const app = createApp(App); 12 | 13 | 14 | app.use(Antd); 15 | app.use(router); 16 | app.use(piniaStore); 17 | app.use(bootstrap) 18 | app.mount('#app'); 19 | -------------------------------------------------------------------------------- /web/src/router/index.js: -------------------------------------------------------------------------------- 1 | import {createRouter, createWebHistory} from 'vue-router'; 2 | import root from './root'; 3 | 4 | import { ADMIN_USER_TOKEN, USER_TOKEN } from '/@/store/constants' 5 | 6 | // 路由权限白名单 7 | const allowList = ['adminLogin', 'login', 'register', 'portal', 'search', 'detail', '403', '404'] 8 | // 前台登录地址 9 | const loginRoutePath = '/index/login' 10 | // 后台登录地址 11 | const adminLoginRoutePath = '/adminLogin' 12 | 13 | 14 | const router = createRouter({ 15 | history: createWebHistory(), 16 | routes: root, 17 | }); 18 | 19 | router.beforeEach(async (to, from, next) => { 20 | console.log(to, from) 21 | 22 | /** 路由 **/ 23 | if (to.path.startsWith('/admin')) { 24 | if (localStorage.getItem(ADMIN_USER_TOKEN)) { 25 | if (to.path === adminLoginRoutePath) { 26 | next({ path: '/' }) 27 | } else { 28 | next() 29 | } 30 | } else { 31 | if (allowList.includes(to.name)) { 32 | // 在免登录名单,直接进入 33 | next() 34 | } else { 35 | next({ path: adminLoginRoutePath, query: { redirect: to.fullPath } }) 36 | } 37 | } 38 | 39 | // next() 40 | } 41 | 42 | 43 | }); 44 | 45 | router.afterEach((_to) => { 46 | // 回到顶部 47 | let html = document.getElementById("html"); 48 | if(html){ 49 | html.scrollTo(0, 0) 50 | } 51 | }); 52 | 53 | export default router; 54 | -------------------------------------------------------------------------------- /web/src/router/root.js: -------------------------------------------------------------------------------- 1 | // 路由表 2 | const constantRouterMap = [ 3 | 4 | { 5 | path: '/', 6 | redirect: '/admin', 7 | }, 8 | { 9 | path: '/adminLogin', 10 | name: 'adminLogin', 11 | component: () => import('/@/views/admin-login.vue'), 12 | }, 13 | { 14 | path: '/admin', 15 | name: 'admin', 16 | redirect: '/admin/thing', 17 | component: () => import('/@/views/main.vue'), 18 | children: [ 19 | { path: 'overview', name: 'overview', component: () => import('/@/views/overview.vue') }, 20 | { path: 'thing', name: 'thing', component: () => import('/@/views/thing.vue') }, 21 | { path: 'comment', name: 'comment', component: () => import('/@/views/comment.vue') }, 22 | { path: 'user', name: 'user', component: () => import('/@/views/user.vue') }, 23 | { path: 'customer', name: 'customer', component: () => import('/@/views/customer.vue') }, 24 | { path: 'classification', name: 'classification', component: () => import('/@/views/classification.vue') }, 25 | { path: 'tag', name: 'tag', component: () => import('/@/views/tag.vue') }, 26 | { path: 'ad', name: 'ad', component: () => import('/@/views/ad.vue') }, 27 | { path: 'notice', name: 'notice', component: () => import('/@/views/notice.vue') }, 28 | { path: 'loginLog', name: 'loginLog', component: () => import('/@/views/login-log.vue') }, 29 | { path: 'opLog', name: 'opLog', component: () => import('/@/views/op-log.vue') }, 30 | { path: 'errorLog', name: 'errorLog', component: () => import('/@/views/error-log.vue') }, 31 | { path: 'sysInfo', name: 'sysInfo', component: () => import('/@/views/sys-info.vue') }, 32 | ] 33 | }, 34 | ]; 35 | 36 | export default constantRouterMap; 37 | -------------------------------------------------------------------------------- /web/src/store/index.js: -------------------------------------------------------------------------------- 1 | import { createPinia } from 'pinia'; 2 | import { useAppStore } from './modules/app'; 3 | import { useUserStore } from './modules/user'; 4 | 5 | import piniaPluginPersistedstate from 'pinia-plugin-persistedstate'; 6 | 7 | const pinia = createPinia(); 8 | pinia.use(piniaPluginPersistedstate); 9 | 10 | export { useAppStore, useUserStore }; 11 | export default pinia; 12 | -------------------------------------------------------------------------------- /web/src/store/modules/app/index.ts: -------------------------------------------------------------------------------- 1 | import { defineStore } from 'pinia'; 2 | import piniaStore from '/@/store/index'; 3 | import { AppState } from './types'; 4 | 5 | export const useAppStore = defineStore( 6 | // 唯一ID 7 | 'app', 8 | { 9 | state: () => ({ 10 | title: 'FastVue3, 一个快速开箱即用的Vue3+Vite模板', 11 | h1: 'Vue3 + Vite3.x + TypeScript + Pinia大厂开发必备', 12 | theme: '', 13 | }), 14 | getters: {}, 15 | actions: { 16 | updateSettings(partial: Partial) { 17 | this.$patch(partial); 18 | }, 19 | 20 | // Change theme color 21 | toggleTheme(dark: boolean) { 22 | if (dark) { 23 | this.theme = 'dark'; 24 | document.documentElement.classList.add('dark'); 25 | } else { 26 | this.theme = 'light'; 27 | document.documentElement.classList.remove('dark'); 28 | } 29 | }, 30 | }, 31 | persist: { 32 | key: 'theme', 33 | storage: localStorage, 34 | paths: ['theme'], 35 | }, 36 | }, 37 | ); 38 | 39 | export function useAppOutsideStore() { 40 | return useAppStore(piniaStore); 41 | } 42 | -------------------------------------------------------------------------------- /web/src/store/modules/app/types.ts: -------------------------------------------------------------------------------- 1 | export interface AppState { 2 | theme: string; 3 | colorWeek: boolean; 4 | navbar: boolean; 5 | menu: boolean; 6 | menuCollapse: boolean; 7 | footer: boolean; 8 | themeColor: string; 9 | menuWidth: number; 10 | globalSettings: boolean; 11 | [key: string]: unknown; 12 | } 13 | -------------------------------------------------------------------------------- /web/src/store/modules/user/index.ts: -------------------------------------------------------------------------------- 1 | import { defineStore } from 'pinia'; 2 | import {loginApi as adminLogin} from '/@/api/user'; 3 | import { setToken, clearToken } from '/@/utils/auth'; 4 | import { UserState } from './types'; 5 | import {USER_ID, USER_NAME, USER_TOKEN, ADMIN_USER_ID,ADMIN_USER_NAME,ADMIN_USER_TOKEN} from "/@/store/constants"; 6 | 7 | export const useUserStore = defineStore('user', { 8 | state: (): UserState => ({ 9 | user_id: undefined, 10 | user_name: undefined, 11 | user_token: undefined, 12 | 13 | admin_user_id: undefined, 14 | admin_user_name: undefined, 15 | admin_user_token: undefined, 16 | }), 17 | getters: {}, 18 | actions: { 19 | 20 | // 管理员登录 21 | async adminLogin(loginForm) { 22 | const result = await adminLogin(loginForm); 23 | console.log('result==>', result) 24 | 25 | if(result.code === 0) { 26 | this.$patch((state)=>{ 27 | state.admin_user_id = result.data.id 28 | state.admin_user_name = result.data.username 29 | state.admin_user_token = result.data.admin_token 30 | console.log('state==>', state) 31 | }) 32 | 33 | localStorage.setItem(ADMIN_USER_TOKEN, result.data.admin_token) 34 | localStorage.setItem(ADMIN_USER_NAME, result.data.username) 35 | localStorage.setItem(ADMIN_USER_ID, result.data.id) 36 | } 37 | 38 | return result; 39 | }, 40 | // 管理员登出 41 | async adminLogout() { 42 | // await userLogout(); 43 | this.$patch((state)=>{ 44 | localStorage.removeItem(ADMIN_USER_ID) 45 | localStorage.removeItem(ADMIN_USER_NAME) 46 | localStorage.removeItem(ADMIN_USER_TOKEN) 47 | 48 | state.admin_user_id = undefined 49 | state.admin_user_name = undefined 50 | state.admin_user_token = undefined 51 | }) 52 | }, 53 | }, 54 | }); 55 | -------------------------------------------------------------------------------- /web/src/store/modules/user/types.ts: -------------------------------------------------------------------------------- 1 | export type RoleType = '' | '*' | 'admin' | 'user'; 2 | export interface UserState { 3 | user_id: any; 4 | user_name: any; 5 | user_token: any; 6 | 7 | admin_user_id: any; 8 | admin_user_name: any; 9 | admin_user_token: any; 10 | } 11 | -------------------------------------------------------------------------------- /web/src/styles/index.less: -------------------------------------------------------------------------------- 1 | //自定义css 2 | a { 3 | color: #1890ff; 4 | } 5 | -------------------------------------------------------------------------------- /web/src/styles/reset.less: -------------------------------------------------------------------------------- 1 | html { 2 | box-sizing: border-box; 3 | } 4 | 5 | *, 6 | ::before, 7 | ::after { 8 | margin: 0; 9 | padding: 0; 10 | box-sizing: inherit; 11 | } 12 | 13 | 14 | a:hover, 15 | a:link, 16 | a:visited, 17 | a:active { 18 | text-decoration: none; 19 | } 20 | 21 | ol, 22 | ul { 23 | list-style: none; 24 | } 25 | 26 | input, 27 | textarea { 28 | outline: none; 29 | border: none; 30 | resize: none; 31 | } 32 | 33 | body { 34 | font-size: 14px; 35 | font-weight: 400; 36 | } 37 | -------------------------------------------------------------------------------- /web/src/utils/auth.ts: -------------------------------------------------------------------------------- 1 | const TokenKey = 'fast-token'; 2 | const TokenPrefix = 'Bearer '; 3 | const isLogin = () => { 4 | return !!localStorage.getItem(TokenKey); 5 | }; 6 | const getToken = () => { 7 | return localStorage.getItem(TokenKey); 8 | }; 9 | const setToken = (token: string) => { 10 | localStorage.setItem(TokenKey, token); 11 | }; 12 | const clearToken = () => { 13 | localStorage.removeItem(TokenKey); 14 | }; 15 | export { TokenPrefix, isLogin, getToken, setToken, clearToken }; 16 | -------------------------------------------------------------------------------- /web/src/utils/http/axios/index.ts: -------------------------------------------------------------------------------- 1 | import axios from 'axios'; 2 | import type { AxiosInstance, AxiosRequestConfig, AxiosResponse, AxiosError, InternalAxiosRequestConfig } from 'axios'; 3 | import { showMessage } from './status'; 4 | import { IResponse } from './type'; 5 | import { getToken } from '/@/utils/auth'; 6 | import { TokenPrefix } from '/@/utils/auth'; 7 | import {ADMIN_USER_TOKEN, USER_TOKEN, BASE_URL} from '/@/store/constants' 8 | 9 | const service: AxiosInstance = axios.create({ 10 | baseURL: BASE_URL + '', 11 | timeout: 15000, 12 | }); 13 | 14 | // axios实例拦截请求 15 | service.interceptors.request.use( 16 | (config: InternalAxiosRequestConfig) => { 17 | 18 | config.headers.ADMINTOKEN = localStorage.getItem(ADMIN_USER_TOKEN) 19 | config.headers.TOKEN = localStorage.getItem(USER_TOKEN) 20 | 21 | return config; 22 | }, 23 | (error: AxiosError) => { 24 | return Promise.reject(error); 25 | }, 26 | ); 27 | 28 | // axios实例拦截响应 29 | service.interceptors.response.use( 30 | (response: AxiosResponse) => { 31 | if(response.status == 200) { 32 | if(response.data.code == 0 || response.data.code == 200) { 33 | return response 34 | }else { 35 | return Promise.reject(response.data) 36 | } 37 | } else { 38 | return Promise.reject(response.data) 39 | } 40 | }, 41 | // 请求失败 42 | (error: any) => { 43 | console.log(error.response.status) 44 | if(error.response.status == 404) { 45 | // todo 46 | } else if(error.response.status == 403) { 47 | // todo 48 | } 49 | return Promise.reject(error) 50 | }, 51 | ); 52 | 53 | 54 | 55 | const request = (config: AxiosRequestConfig): Promise => { 56 | const conf = config; 57 | return new Promise((resolve, reject) => { 58 | service.request>(conf).then((res: AxiosResponse) => { 59 | const data = res.data 60 | resolve(data as T); 61 | }).catch(err => { 62 | reject(err) 63 | }); 64 | }); 65 | }; 66 | 67 | export function get(config: AxiosRequestConfig): Promise { 68 | return request({ ...config, method: 'GET' }); 69 | } 70 | 71 | export function post(config: AxiosRequestConfig): Promise { 72 | return request({ ...config, method: 'POST' }); 73 | } 74 | 75 | export default request; 76 | 77 | export type { AxiosInstance, AxiosResponse }; 78 | -------------------------------------------------------------------------------- /web/src/utils/http/axios/status.ts: -------------------------------------------------------------------------------- 1 | export const showMessage = (status: number | string): string => { 2 | let message = ''; 3 | switch (status) { 4 | case 400: 5 | message = '请求错误(400)'; 6 | break; 7 | case 401: 8 | message = '未授权,请重新登录(401)'; 9 | break; 10 | case 403: 11 | message = '拒绝访问(403)'; 12 | break; 13 | case 404: 14 | message = '请求出错(404)'; 15 | break; 16 | case 408: 17 | message = '请求超时(408)'; 18 | break; 19 | case 500: 20 | message = '服务器错误(500)'; 21 | break; 22 | case 501: 23 | message = '服务未实现(501)'; 24 | break; 25 | case 502: 26 | message = '网络错误(502)'; 27 | break; 28 | case 503: 29 | message = '服务不可用(503)'; 30 | break; 31 | case 504: 32 | message = '网络超时(504)'; 33 | break; 34 | case 505: 35 | message = 'HTTP版本不受支持(505)'; 36 | break; 37 | default: 38 | message = `连接出错(${status})!`; 39 | } 40 | return `${message},请检查网络或联系管理员!`; 41 | }; 42 | -------------------------------------------------------------------------------- /web/src/utils/http/axios/type.ts: -------------------------------------------------------------------------------- 1 | export interface RequestOptions { 2 | // Whether to process the request result 3 | isTransformResponse?: boolean; 4 | } 5 | 6 | // 返回res.data的interface 7 | export interface IResponse { 8 | code: number | string; 9 | result: T; 10 | message: string; 11 | status: string | number; 12 | } 13 | 14 | /**用户登录 */ 15 | export interface ILogin { 16 | /** 账户名称 */ 17 | username: string; 18 | /** 账户密码 */ 19 | password: string; 20 | } 21 | -------------------------------------------------------------------------------- /web/src/utils/index.ts: -------------------------------------------------------------------------------- 1 | // import { resolve } from 'path'; 2 | // const fs = require('fs'); 3 | // 4 | // function pathResolve(dir: string) { 5 | // return resolve(process.cwd(), '.', dir); 6 | // } 7 | // 8 | // export const getFolder = (path: any) => { 9 | // const components: Array = []; 10 | // const files = fs.readdirSync(path); 11 | // files.forEach(function (item: string) { 12 | // const stat = fs.lstatSync(path + '/' + item); 13 | // if (stat.isDirectory() === true && item != 'components') { 14 | // components.push(path + '/' + item); 15 | // components.push(pathResolve(path + '/' + item)); 16 | // } 17 | // }); 18 | // return components; 19 | // }; 20 | 21 | export function getFormatTime(dateTime,flag) { 22 | if(dateTime != null ) { 23 | //若传入的dateTime为字符串类型,需要进行转换成数值,若不是无需下面注释代码 24 | var time = parseInt(dateTime) 25 | var date = new Date(time); 26 | //获取年份 27 | var YY = date.getFullYear(); 28 | //获取月份 29 | var MM = (date.getMonth() + 1 < 10 ? '0'+(date.getMonth() + 1) : date.getMonth() + 1); 30 | //获取日期 31 | var DD = (date.getDate() < 10 ? '0'+date.getDate() : date.getDate()); 32 | if(flag) { //flag为true,显示时分秒格式 33 | //获取小时 34 | var hh = (date.getHours() < 10 ? '0'+date.getHours() : date.getHours()); 35 | //获取分 36 | var mm = (date.getMinutes() < 10 ? '0'+date.getMinutes() : date.getMinutes()); 37 | ///获取秒 38 | var ss = (date.getSeconds() < 10 ? '0'+date.getSeconds() : date.getSeconds()); 39 | //返回时间格式: 2020-11-09 13:14:52 40 | return YY + '-' + MM + '-' + DD + ' ' + hh + ':' + mm + ':' + ss; 41 | } else { 42 | //返回时间格式: 2020-11-09 43 | return YY + '-' + MM + '-' + DD; 44 | } 45 | } else { 46 | return ""; 47 | } 48 | } 49 | -------------------------------------------------------------------------------- /web/src/utils/result.ts: -------------------------------------------------------------------------------- 1 | import { Recoverable } from 'repl'; 2 | 3 | // 返回统一格式的接口数据类型定义 4 | export function successResult(result: T, { message = 'Request success' } = {}) { 5 | return { 6 | code: 200, 7 | result, 8 | message, 9 | status: 'ok', 10 | }; 11 | } 12 | export function errorResult(message = 'Request failed', { code = -1, result = null } = {}) { 13 | return { 14 | code, 15 | result, 16 | message, 17 | status: 'fail', 18 | }; 19 | } 20 | 21 | //返回分页数据 22 | export function pageSuccessResult(page: number, pageSize: number, list: T[], { message = 'ok' } = {}) { 23 | const pageData = pagination(page, pageSize, list); 24 | return { 25 | ...successResult({ 26 | items: pageData, 27 | total: list.length, 28 | }), 29 | message, 30 | }; 31 | } 32 | 33 | // 封装分页数据 34 | export function pagination(pageNo: number, pageSize: number, array: T[]): T[] { 35 | const offset = (pageNo - 1) * Number(pageSize); 36 | const res = 37 | offset + Number(pageSize) >= array.length ? array.slice(offset, array.length) : array.slice(offset, offset + Number(pageSize)); 38 | return res; 39 | } 40 | 41 | // 返回参数类型定义 42 | export interface requestParams { 43 | method: string; 44 | body: any; 45 | headers?: { authorization?: string }; 46 | query: any; 47 | } 48 | 49 | /** 50 | * @name getRequestToken 51 | * @description 通过request数据中获取token,具体情况根据接口规范修改 52 | */ 53 | export function getRequestToken({ headers }: requestParams): string | undefined { 54 | return headers?.authorization; 55 | } 56 | -------------------------------------------------------------------------------- /web/src/views/error-log.vue: -------------------------------------------------------------------------------- 1 | 36 | 37 | 123 | 124 | 142 | -------------------------------------------------------------------------------- /web/src/views/login-log.vue: -------------------------------------------------------------------------------- 1 | 36 | 37 | 112 | 113 | 131 | -------------------------------------------------------------------------------- /web/src/views/sys-info.vue: -------------------------------------------------------------------------------- 1 | 58 | 59 | 83 | 84 | 102 | -------------------------------------------------------------------------------- /web/stylelint.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | root: true, 3 | plugins: ['stylelint-order'], 4 | extends: ['stylelint-config-standard', 'stylelint-config-prettier'], 5 | customSyntax: 'postcss-html', 6 | rules: { 7 | 'function-no-unknown': null, 8 | 'selector-class-pattern': null, 9 | 'selector-pseudo-class-no-unknown': [ 10 | true, 11 | { 12 | ignorePseudoClasses: ['global'], 13 | }, 14 | ], 15 | 'selector-pseudo-element-no-unknown': [ 16 | true, 17 | { 18 | ignorePseudoElements: ['v-deep'], 19 | }, 20 | ], 21 | 'at-rule-no-unknown': [ 22 | true, 23 | { 24 | ignoreAtRules: ['tailwind', 'apply', 'variants', 'responsive', 'screen', 'function', 'if', 'each', 'include', 'mixin'], 25 | }, 26 | ], 27 | 'no-empty-source': null, 28 | 'string-quotes': null, 29 | 'named-grid-areas-no-invalid': null, 30 | 'unicode-bom': 'never', 31 | 'no-descending-specificity': null, 32 | 'font-family-no-missing-generic-family-keyword': null, 33 | 'declaration-colon-space-after': 'always-single-line', 34 | 'declaration-colon-space-before': 'never', 35 | // 'declaration-block-trailing-semicolon': 'always', 36 | 'rule-empty-line-before': [ 37 | 'always', 38 | { 39 | ignore: ['after-comment', 'first-nested'], 40 | }, 41 | ], 42 | 'unit-no-unknown': [true, { ignoreUnits: ['rpx'] }], 43 | 'order/order': [ 44 | [ 45 | 'dollar-variables', 46 | 'custom-properties', 47 | 'at-rules', 48 | 'declarations', 49 | { 50 | type: 'at-rule', 51 | name: 'supports', 52 | }, 53 | { 54 | type: 'at-rule', 55 | name: 'media', 56 | }, 57 | 'rules', 58 | ], 59 | { severity: 'warning' }, 60 | ], 61 | }, 62 | ignoreFiles: ['**/*.js', '**/*.jsx', '**/*.tsx', '**/*.ts'], 63 | overrides: [ 64 | { 65 | files: ['*.vue', '**/*.vue', '*.html', '**/*.html'], 66 | extends: ['stylelint-config-recommended'], 67 | rules: { 68 | 'keyframes-name-pattern': null, 69 | 'selector-pseudo-class-no-unknown': [ 70 | true, 71 | { 72 | ignorePseudoClasses: ['deep', 'global'], 73 | }, 74 | ], 75 | 'selector-pseudo-element-no-unknown': [ 76 | true, 77 | { 78 | ignorePseudoElements: ['v-deep', 'v-global', 'v-slotted'], 79 | }, 80 | ], 81 | }, 82 | }, 83 | { 84 | files: ['*.less', '**/*.less'], 85 | customSyntax: 'postcss-less', 86 | extends: ['stylelint-config-standard', 'stylelint-config-recommended-vue'], 87 | }, 88 | ], 89 | }; 90 | -------------------------------------------------------------------------------- /web/tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "esnext", 4 | "module": "esnext", 5 | "moduleResolution": "node", 6 | "strict": true, 7 | "noLib": false, 8 | "forceConsistentCasingInFileNames": true, 9 | "allowSyntheticDefaultImports": true, 10 | "strictFunctionTypes": false, 11 | "jsx": "preserve", 12 | "baseUrl": ".", 13 | "allowJs": true, 14 | "sourceMap": true, 15 | "esModuleInterop": true, 16 | "resolveJsonModule": true, 17 | "noUnusedLocals": false, 18 | "noUnusedParameters": false, 19 | "experimentalDecorators": true, 20 | "lib": ["dom", "esnext"], 21 | "noImplicitAny": false, 22 | "skipLibCheck": true, 23 | "types": ["vite/client"], 24 | "removeComments": true, 25 | "paths": { 26 | "/@/*": ["src/*"], 27 | "/#/*": ["types/*"] 28 | } 29 | }, 30 | "include": [ 31 | "tests/**/*.ts", 32 | "src/**/*.ts", 33 | "src/**/*.d.ts", 34 | "src/**/*.tsx", 35 | "src/**/*.vue", 36 | "types/**/*.d.ts", 37 | "types/**/*.ts", 38 | "build/**/*.ts", 39 | "build/**/*.d.ts", 40 | "mock/**/*.ts", 41 | "vite.config.ts" 42 | ], 43 | "exclude": ["node_modules", "tests/server/**/*.ts", "dist", "**/*.js"] 44 | } 45 | -------------------------------------------------------------------------------- /web/types/components.d.ts: -------------------------------------------------------------------------------- 1 | // generated by unplugin-vue-components 2 | // We suggest you to commit this file into source control 3 | // Read more: https://github.com/vuejs/core/pull/3399 4 | import '@vue/runtime-core' 5 | 6 | export {} 7 | 8 | declare module '@vue/runtime-core' { 9 | export interface GlobalComponents { 10 | AButton: typeof import('ant-design-vue/es')['Button'] 11 | ACard: typeof import('ant-design-vue/es')['Card'] 12 | ACol: typeof import('ant-design-vue/es')['Col'] 13 | AConfigProvider: typeof import('ant-design-vue/es')['ConfigProvider'] 14 | ADescriptions: typeof import('ant-design-vue/es')['Descriptions'] 15 | ADescriptionsItem: typeof import('ant-design-vue/es')['DescriptionsItem'] 16 | ADivider: typeof import('ant-design-vue/es')['Divider'] 17 | ADrawer: typeof import('ant-design-vue/es')['Drawer'] 18 | ADropdown: typeof import('ant-design-vue/es')['Dropdown'] 19 | AForm: typeof import('ant-design-vue/es')['Form'] 20 | AFormItem: typeof import('ant-design-vue/es')['FormItem'] 21 | AInput: typeof import('ant-design-vue/es')['Input'] 22 | AInputNumber: typeof import('ant-design-vue/es')['InputNumber'] 23 | AInputPassword: typeof import('ant-design-vue/es')['InputPassword'] 24 | AInputSearch: typeof import('ant-design-vue/es')['InputSearch'] 25 | ALayout: typeof import('ant-design-vue/es')['Layout'] 26 | ALayoutContent: typeof import('ant-design-vue/es')['LayoutContent'] 27 | ALayoutHeader: typeof import('ant-design-vue/es')['LayoutHeader'] 28 | ALayoutSider: typeof import('ant-design-vue/es')['LayoutSider'] 29 | AMenu: typeof import('ant-design-vue/es')['Menu'] 30 | AMenuItem: typeof import('ant-design-vue/es')['MenuItem'] 31 | AModal: typeof import('ant-design-vue/es')['Modal'] 32 | APagination: typeof import('ant-design-vue/es')['Pagination'] 33 | APopconfirm: typeof import('ant-design-vue/es')['Popconfirm'] 34 | ARow: typeof import('ant-design-vue/es')['Row'] 35 | ASelect: typeof import('ant-design-vue/es')['Select'] 36 | ASelectOption: typeof import('ant-design-vue/es')['SelectOption'] 37 | ASpace: typeof import('ant-design-vue/es')['Space'] 38 | ASpin: typeof import('ant-design-vue/es')['Spin'] 39 | ASubMenu: typeof import('ant-design-vue/es')['SubMenu'] 40 | ASwitch: typeof import('ant-design-vue/es')['Switch'] 41 | ATable: typeof import('ant-design-vue/es')['Table'] 42 | ATabPane: typeof import('ant-design-vue/es')['TabPane'] 43 | ATabs: typeof import('ant-design-vue/es')['Tabs'] 44 | ATag: typeof import('ant-design-vue/es')['Tag'] 45 | ATextarea: typeof import('ant-design-vue/es')['Textarea'] 46 | ATree: typeof import('ant-design-vue/es')['Tree'] 47 | ATreeSelect: typeof import('ant-design-vue/es')['TreeSelect'] 48 | ATreeSelectNode: typeof import('ant-design-vue/es')['TreeSelectNode'] 49 | AUpload: typeof import('ant-design-vue/es')['Upload'] 50 | AUploadDragger: typeof import('ant-design-vue/es')['UploadDragger'] 51 | RouterLink: typeof import('vue-router')['RouterLink'] 52 | RouterView: typeof import('vue-router')['RouterView'] 53 | } 54 | } 55 | -------------------------------------------------------------------------------- /web/types/env.d.ts: -------------------------------------------------------------------------------- 1 | /// 2 | 3 | declare module '*.vue' { 4 | import { DefineComponent } from 'vue'; 5 | const component: DefineComponent<{}, {}, any>; 6 | export default component; 7 | } 8 | 9 | declare module 'virtual:*' { 10 | const result: any; 11 | export default result; 12 | } 13 | -------------------------------------------------------------------------------- /web/vite.config.ts: -------------------------------------------------------------------------------- 1 | import { UserConfig, ConfigEnv } from 'vite'; 2 | import { createVitePlugins } from './build/vite/plugins'; 3 | import { resolve } from 'path'; 4 | import { VITE_PORT } from './build/constant'; 5 | 6 | function pathResolve(dir: string) { 7 | return resolve(process.cwd(), '.', dir); 8 | } 9 | 10 | // https://vitejs.dev/config/ 11 | export default ({ command }: ConfigEnv): UserConfig => { 12 | const isBuild = command === 'build'; 13 | let base: string; 14 | if (command === 'build') { 15 | base = '/'; 16 | } else { 17 | base = '/'; 18 | } 19 | return { 20 | base, 21 | publicDir: "public", //静态资源服务的文件夹 22 | resolve: { 23 | alias: [ 24 | { 25 | find: 'vue-i18n', 26 | replacement: 'vue-i18n/dist/vue-i18n.cjs.js', 27 | }, 28 | // 别名 /@/xxxx => src/xxxx 29 | { 30 | find: '/@', 31 | replacement: pathResolve('src') + '/', 32 | }, 33 | ], 34 | }, 35 | // plugins 36 | plugins: createVitePlugins(isBuild), 37 | 38 | // css 39 | css: {}, 40 | 41 | // server 42 | server: { 43 | hmr: { overlay: false }, // 禁用或配置 HMR 连接 设置 server.hmr.overlay 为 false 可以禁用服务器错误遮罩层 44 | // 服务配置 45 | port: VITE_PORT, // 类型: number 指定服务器端口; 46 | open: false, // 类型: boolean | string在服务器启动时自动在浏览器中打开应用程序; 47 | cors: true, // 类型: boolean | CorsOptions 为开发服务器配置 CORS。默认启用并允许任何源 48 | host: '0.0.0.0', // IP配置,支持从IP启动 49 | https: false, // 禁用https 50 | // proxy, 51 | }, 52 | }; 53 | }; 54 | --------------------------------------------------------------------------------