├── .gitignore ├── Readme.md ├── doc.md ├── doc └── python_adopt.docx ├── python_adopt.sql ├── server ├── manage.py ├── myapp │ ├── __init__.py │ ├── admin.py │ ├── apps.py │ ├── auth │ │ ├── MyRateThrottle.py │ │ ├── __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 │ │ ├── errorLog.py │ │ ├── loginLog.py │ │ ├── notice.py │ │ ├── opLog.py │ │ ├── order.py │ │ ├── overview.py │ │ ├── record.py │ │ ├── tag.py │ │ ├── thing.py │ │ └── user.py │ │ └── index │ │ ├── __init__.py │ │ ├── address.py │ │ ├── classification.py │ │ ├── comment.py │ │ ├── notice.py │ │ ├── order.py │ │ ├── tag.py │ │ ├── thing.py │ │ └── user.py ├── readme-doc.md ├── 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 │ └── 1687260782755.jpeg │ ├── 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 │ ├── 1687258713493.jpeg │ ├── 1687258782902.jpeg │ ├── 1687258812642.jpeg │ ├── 1687258953981.png │ ├── 1687260056932.jpeg │ ├── 1687260111383.jpeg │ ├── 1687260187342.jpeg │ ├── 1687260228638.jpeg │ ├── 1687260327436.jpeg │ ├── 1687260376491.jpeg │ ├── 1687260425529.jpeg │ ├── 1687260459725.jpeg │ ├── 1687260505646.jpeg │ ├── 17.jpg │ ├── 1706520451417.jpeg │ ├── 1706522032586.jpeg │ ├── 1706522180820.jpeg │ ├── 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 │ └── img │ ├── Wechat.jpeg │ ├── a.png │ ├── b.png │ └── weixin.png └── 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 │ ├── admin │ │ ├── ad.ts │ │ ├── classification.ts │ │ ├── comment.ts │ │ ├── log.ts │ │ ├── notice.ts │ │ ├── order.ts │ │ ├── overview.ts │ │ ├── tag.ts │ │ ├── thing.ts │ │ └── user.ts │ └── index │ │ ├── address.ts │ │ ├── classification.ts │ │ ├── comment.ts │ │ ├── notice.ts │ │ ├── order.ts │ │ ├── tag.ts │ │ ├── thing.ts │ │ ├── thingCollect.ts │ │ ├── thingWish.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 │ │ ├── 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 │ ├── constants.js │ ├── 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 │ ├── admin │ ├── ad.vue │ ├── admin-login.vue │ ├── classification.vue │ ├── comment.vue │ ├── error-log.vue │ ├── login-log.vue │ ├── main.vue │ ├── notice.vue │ ├── op-log.vue │ ├── order.vue │ ├── overview.vue │ ├── sys-info.vue │ ├── tag.vue │ ├── thing.vue │ └── user.vue │ └── index │ ├── components │ ├── content.vue │ ├── footer.vue │ ├── header.vue │ └── search-content-view.vue │ ├── confirm.vue │ ├── detail.vue │ ├── index.vue │ ├── login.vue │ ├── pay.vue │ ├── portal.vue │ ├── register.vue │ ├── search.vue │ ├── user │ ├── address-view.vue │ ├── collect-thing-view.vue │ ├── comment-view.vue │ ├── fans-view.vue │ ├── follow-view.vue │ ├── message-view.vue │ ├── mine-infos-view.vue │ ├── modal │ │ └── edit-address.vue │ ├── order-view.vue │ ├── push-view.vue │ ├── score-view.vue │ ├── security-view.vue │ ├── userinfo-edit-view.vue │ └── wish-thing-view.vue │ └── usercenter.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 | 2 | 3 | ### 功能介绍 4 | 5 | > 一直想做一款管理系统,看了很多优秀的开源项目但是发现没有合适的。于是利用空闲休息时间开始自己写了一套管理系统。现将部分源码开源,如需完整源码,可以联系客服微信购买:lengqin1024 6 | 7 | 整个平台包括前台和后台两个部分。 8 | 9 | - 前台功能包括:首页、宠物详情页、用户中心模块。 10 | - 后台功能包括:总览、领养管理、宠物管理、分类管理、标签管理、评论管理、用户管理、运营管理、日志管理、系统信息模块。 11 | 12 | 13 | 14 | ### 演示地址 15 | 16 | 前台地址: http://adopt.gitapp.cn 17 | 18 | 后台地址: http://adopt.gitapp.cn/admin 19 | 20 | 后台管理帐号: 21 | 22 | 用户名:admin123 23 | 密码:admin123 24 | 25 | ### 代码结构 26 | 27 | - server目录是后端代码 28 | - web目录是前端代码 29 | 30 | ### 部署运行 31 | 32 | #### 后端运行步骤 33 | 34 | (1) 安装python 3.8 35 | 36 | (2) 安装依赖。进入server目录下,执行 pip install -r requirements.txt 37 | 38 | (3) 安装mysql 5.7数据库,并创建数据库,命名为python_adopt,创建SQL如下: 39 | ``` 40 | CREATE DATABASE IF NOT EXISTS python_adopt DEFAULT CHARSET utf8 COLLATE utf8_general_ci 41 | ``` 42 | (4) 恢复shop.sql数据。在mysql下依次执行如下命令: 43 | 44 | ``` 45 | mysql> use python_adopt; 46 | mysql> source D:/xxx/xxx/python_adopt.sql; 47 | ``` 48 | 49 | (5) 启动django服务。在server目录下执行: 50 | ``` 51 | python manage.py runserver 52 | ``` 53 | 54 | #### 前端运行步骤 55 | 56 | (1) 安装node 16.14 57 | 58 | (2) 进入web目录下,安装依赖,执行: 59 | ``` 60 | npm install 61 | ``` 62 | (3) 运行项目 63 | ``` 64 | npm run dev 65 | ``` 66 | 67 | 68 | 69 | 70 | ### 待完善功能 71 | 72 | - 邮箱推送功能 73 | - 手机号绑定功能 74 | - 粉丝关注功能 75 | - 支付功能 76 | 77 | 78 | ### 开发文档 79 | 80 | [点击查看](doc.md) 81 | 82 | 83 | ### 参考论文 84 | 85 | [点击查看](doc/python_adopt.docx) 86 | 87 | ### 付费咨询 88 | 89 | 微信(lengqin1024) 90 | 91 | 92 | 93 | -------------------------------------------------------------------------------- /doc/python_adopt.docx: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/doc/python_adopt.docx -------------------------------------------------------------------------------- /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_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/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/MyRateThrottle.py: -------------------------------------------------------------------------------- 1 | from rest_framework.throttling import AnonRateThrottle 2 | 3 | 4 | class MyRateThrottle(AnonRateThrottle): 5 | THROTTLE_RATES = {"anon": "5/min"} 6 | -------------------------------------------------------------------------------- /server/myapp/auth/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/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 | users = User.objects.filter(admin_token=adminToken) 13 | """ 14 | 判定条件: 15 | 1. 传了adminToken 16 | 2. 查到了该帐号 17 | 3. 该帐号是管理员或演示帐号 18 | """ 19 | if not adminToken or len(users) == 0 or users[0].role == '2': 20 | raise exceptions.AuthenticationFailed("AUTH_FAIL_END") 21 | else: 22 | print('adminToken验证通过') 23 | 24 | 25 | # 前台接口认证 26 | class TokenAuthtication(BaseAuthentication): 27 | def authenticate(self, request): 28 | token = request.META.get("HTTP_TOKEN", "") 29 | if token is not None: 30 | print("检查token==>" + token) 31 | users = User.objects.filter(token=token) 32 | # print(users) 33 | """ 34 | 判定条件: 35 | 1. 传了token 36 | 2. 查到了该帐号 37 | 3. 该帐号是普通用户 38 | """ 39 | if not token or len(users) == 0 or (users[0].role in ['1', '3']): 40 | raise exceptions.AuthenticationFailed("AUTH_FAIL_FRONT") 41 | else: 42 | print('token验证通过') 43 | else: 44 | print("检查token==>token 为空") 45 | raise exceptions.AuthenticationFailed("AUTH_FAIL_FRONT") 46 | -------------------------------------------------------------------------------- /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_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/myapp/middlewares/__init__.py -------------------------------------------------------------------------------- /server/myapp/permission/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/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/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 | from myapp.views.index import * 3 | -------------------------------------------------------------------------------- /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.order import * 10 | from myapp.views.admin.opLog import * 11 | from myapp.views.admin.errorLog import * 12 | from myapp.views.admin.banner import * 13 | from myapp.views.admin.ad import * 14 | from myapp.views.admin.notice 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/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/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/myapp/views/index/__init__.py: -------------------------------------------------------------------------------- 1 | from myapp.views.index.classification import * 2 | from myapp.views.index.tag import * 3 | from myapp.views.index.user import * 4 | from myapp.views.index.thing import * 5 | from myapp.views.index.comment import * 6 | from myapp.views.index.order import * 7 | from myapp.views.index.notice import * 8 | from myapp.views.index.address import * 9 | -------------------------------------------------------------------------------- /server/myapp/views/index/classification.py: -------------------------------------------------------------------------------- 1 | # Create your views here. 2 | from django.db import connection 3 | from rest_framework.decorators import api_view 4 | 5 | from myapp.handler import APIResponse 6 | from myapp.models import Classification 7 | from myapp.serializers import ClassificationSerializer 8 | from myapp.utils import dict_fetchall 9 | 10 | 11 | @api_view(['GET']) 12 | def list_api(request): 13 | if request.method == 'GET': 14 | classifications = Classification.objects.all().order_by('-create_time') 15 | serializer = ClassificationSerializer(classifications, many=True) 16 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /server/myapp/views/index/notice.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 Notice 6 | from myapp.serializers import NoticeSerializer 7 | 8 | 9 | @api_view(['GET']) 10 | def list_api(request): 11 | if request.method == 'GET': 12 | notices = Notice.objects.all().order_by('-create_time') 13 | serializer = NoticeSerializer(notices, many=True) 14 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 15 | 16 | -------------------------------------------------------------------------------- /server/myapp/views/index/tag.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 Tag 6 | from myapp.serializers import TagSerializer 7 | 8 | 9 | @api_view(['GET']) 10 | def list_api(request): 11 | if request.method == 'GET': 12 | tags = Tag.objects.all().order_by('-create_time') 13 | serializer = TagSerializer(tags, many=True) 14 | return APIResponse(code=0, msg='查询成功', data=serializer.data) 15 | 16 | -------------------------------------------------------------------------------- /server/readme.md: -------------------------------------------------------------------------------- 1 | ### 后端部署步骤 2 | 3 | > 部署过程中,如遇问题可咨询作者:lengqin1024(微信) 4 | 5 | 1. 安装mysql数据库,启动服务 6 | 2. 打开cmd命令行,进入mysql,并新建数据库 7 | ``` 8 | mysql -u root -p 9 | CREATE DATABASE IF NOT EXISTS python_adopt DEFAULT CHARSET utf8 COLLATE utf8_general_ci; 10 | ``` 11 | 3. 恢复sql数据 12 | ``` 13 | use python_adopt 14 | source python_adopt.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 0.0.0.0:9003 26 | ``` 27 | 7. 后期维护改动 28 | 29 | 将修改的py文件覆盖服务器的py文件即可,重启django 30 | 31 | ### 删除数据库 32 | 33 | drop database if exists python_adopt; 34 | 35 | ### 创建数据库 36 | 37 | CREATE DATABASE IF NOT EXISTS python_adopt 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/shop.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 | update接口的时候,如果model里面存在多对多字段,则需要设置explode 87 | 88 | 89 | 90 | 91 | 92 | 93 | -------------------------------------------------------------------------------- /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_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/ad/1674045266113.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1674045282581.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/ad/1674045282581.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1674045308177.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/ad/1674045308177.png -------------------------------------------------------------------------------- /server/upload/ad/1674045324510.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/ad/1674045324510.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1684565423182.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/ad/1684565423182.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1684565863904.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/ad/1684565863904.jpeg -------------------------------------------------------------------------------- /server/upload/ad/1684565876995.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/ad/1684565876995.png -------------------------------------------------------------------------------- /server/upload/ad/1687260782755.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/ad/1687260782755.jpeg -------------------------------------------------------------------------------- /server/upload/avatar/1676553050529.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/avatar/1676553050529.png -------------------------------------------------------------------------------- /server/upload/avatar/1676553366217.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/avatar/1676553366217.png -------------------------------------------------------------------------------- /server/upload/avatar/1676553498600.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/avatar/1676553498600.jpeg -------------------------------------------------------------------------------- /server/upload/avatar/1676553815688.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/avatar/1676553815688.jpeg -------------------------------------------------------------------------------- /server/upload/avatar/1677240189427.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/avatar/1677240189427.png -------------------------------------------------------------------------------- /server/upload/avatar/1677982820781.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/avatar/1677982820781.jpeg -------------------------------------------------------------------------------- /server/upload/avatar/1679146350134.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/avatar/1679146350134.jpeg -------------------------------------------------------------------------------- /server/upload/avatar/1684593239449.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/avatar/1684593239449.png -------------------------------------------------------------------------------- /server/upload/avatar/1684593453676.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/avatar/1684593453676.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673963977440.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673963977440.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673964384835.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673964384835.png -------------------------------------------------------------------------------- /server/upload/banner/1673964652167.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673964652167.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965110189.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673965110189.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965198155.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673965198155.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965389141.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673965389141.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965574311.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673965574311.png -------------------------------------------------------------------------------- /server/upload/banner/1673965709533.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673965709533.jpeg -------------------------------------------------------------------------------- /server/upload/banner/1673965718720.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673965718720.png -------------------------------------------------------------------------------- /server/upload/banner/1673965728690.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/banner/1673965728690.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1.jpg -------------------------------------------------------------------------------- /server/upload/cover/10.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/10.jpg -------------------------------------------------------------------------------- /server/upload/cover/11.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/11.jpg -------------------------------------------------------------------------------- /server/upload/cover/12.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/12.jpg -------------------------------------------------------------------------------- /server/upload/cover/13.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/13.jpg -------------------------------------------------------------------------------- /server/upload/cover/14.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/14.jpg -------------------------------------------------------------------------------- /server/upload/cover/15.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/15.jpg -------------------------------------------------------------------------------- /server/upload/cover/16.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/16.jpg -------------------------------------------------------------------------------- /server/upload/cover/1672749055571.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1672749055571.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1674044230851.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1674044230851.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1676186518276.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676186518276.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1676186872895.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676186872895.png -------------------------------------------------------------------------------- /server/upload/cover/1676186935002.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676186935002.png -------------------------------------------------------------------------------- /server/upload/cover/1676188277099.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676188277099.png -------------------------------------------------------------------------------- /server/upload/cover/1676188355688.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676188355688.png -------------------------------------------------------------------------------- /server/upload/cover/1676381084256.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676381084256.png -------------------------------------------------------------------------------- /server/upload/cover/1676381091144.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676381091144.png -------------------------------------------------------------------------------- /server/upload/cover/1676381097051.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676381097051.png -------------------------------------------------------------------------------- /server/upload/cover/1676381103032.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676381103032.png -------------------------------------------------------------------------------- /server/upload/cover/1676381110015.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1676381110015.png -------------------------------------------------------------------------------- /server/upload/cover/1677500674281.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677500674281.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677501266461.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677501266461.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677501470234.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677501470234.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677501544737.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677501544737.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505180730.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505180730.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505298772.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505298772.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505357042.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505357042.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505364969.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505364969.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505380921.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505380921.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505393025.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505393025.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505410960.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505410960.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505421920.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505421920.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505436478.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505436478.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505452753.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505452753.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505579480.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505579480.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505616285.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505616285.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505626565.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505626565.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505637425.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505637425.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505648826.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505648826.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505659291.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505659291.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505667178.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505667178.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505685641.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505685641.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505695894.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505695894.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505706333.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505706333.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505876732.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505876732.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505884200.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505884200.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505890616.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505890616.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505897079.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505897079.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505910282.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505910282.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505919134.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505919134.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505928898.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505928898.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505937048.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505937048.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505945207.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505945207.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677505953782.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677505953782.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677588447622.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677588447622.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1677677497401.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1677677497401.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1678529114670.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1678529114670.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679056479437.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679056479437.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679056528013.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679056528013.png -------------------------------------------------------------------------------- /server/upload/cover/1679056561868.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679056561868.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679056587496.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679056587496.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679058036014.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679058036014.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679058043365.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679058043365.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679229610791.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679229610791.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679229780971.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679229780971.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679229853940.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679229853940.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679229963230.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679229963230.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230045590.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230045590.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230092657.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230092657.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230147996.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230147996.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230364390.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230364390.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230543572.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230543572.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230586243.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230586243.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230641879.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230641879.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230894621.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230894621.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230935716.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230935716.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679230989243.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679230989243.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231022326.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679231022326.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231052822.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679231052822.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231084420.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679231084420.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231113572.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679231113572.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231151929.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679231151929.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231184020.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679231184020.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231221940.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679231221940.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679231465845.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679231465845.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232418133.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232418133.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232448857.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232448857.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232480948.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232480948.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232514230.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232514230.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232543082.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232543082.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232587234.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232587234.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232615578.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232615578.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232650073.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232650073.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232676818.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232676818.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232717023.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232717023.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679232756075.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679232756075.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315151478.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315151478.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315191893.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315191893.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315240787.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315240787.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315276060.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315276060.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315310720.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315310720.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315343720.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315343720.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315365210.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315365210.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315396953.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315396953.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315437571.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315437571.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315478327.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315478327.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315749022.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315749022.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315761797.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315761797.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315803245.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315803245.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315842977.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315842977.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315883620.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315883620.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315915467.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315915467.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315954931.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315954931.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679315985852.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679315985852.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316037517.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316037517.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316072493.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316072493.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316422812.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316422812.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316453474.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316453474.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316484842.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316484842.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316513693.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316513693.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316533369.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316533369.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316566135.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316566135.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316605104.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316605104.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316637185.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316637185.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316666891.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316666891.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679316698087.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679316698087.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403034316.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403034316.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403102225.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403102225.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403138846.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403138846.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403158073.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403158073.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403193320.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403193320.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403239138.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403239138.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403285129.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403285129.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403331202.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403331202.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403370669.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403370669.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403403756.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403403756.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403503106.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403503106.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403631942.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403631942.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403697740.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403697740.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679403736341.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679403736341.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1679488092121.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1679488092121.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1684567758685.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1684567758685.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1684567804066.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1684567804066.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1684567891591.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1684567891591.png -------------------------------------------------------------------------------- /server/upload/cover/1687258713493.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687258713493.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687258782902.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687258782902.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687258812642.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687258812642.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687258953981.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687258953981.png -------------------------------------------------------------------------------- /server/upload/cover/1687260056932.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687260056932.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687260111383.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687260111383.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687260187342.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687260187342.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687260228638.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687260228638.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687260327436.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687260327436.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687260376491.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687260376491.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687260425529.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687260425529.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687260459725.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687260459725.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1687260505646.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1687260505646.jpeg -------------------------------------------------------------------------------- /server/upload/cover/17.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/17.jpg -------------------------------------------------------------------------------- /server/upload/cover/1706520451417.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1706520451417.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1706522032586.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1706522032586.jpeg -------------------------------------------------------------------------------- /server/upload/cover/1706522180820.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/1706522180820.jpeg -------------------------------------------------------------------------------- /server/upload/cover/18.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/18.jpg -------------------------------------------------------------------------------- /server/upload/cover/19.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/19.jpg -------------------------------------------------------------------------------- /server/upload/cover/2.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/2.jpg -------------------------------------------------------------------------------- /server/upload/cover/20.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/20.jpg -------------------------------------------------------------------------------- /server/upload/cover/21.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/21.jpg -------------------------------------------------------------------------------- /server/upload/cover/22.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/22.jpg -------------------------------------------------------------------------------- /server/upload/cover/23.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/23.jpg -------------------------------------------------------------------------------- /server/upload/cover/24.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/24.jpg -------------------------------------------------------------------------------- /server/upload/cover/25.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/25.jpg -------------------------------------------------------------------------------- /server/upload/cover/26.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/26.jpg -------------------------------------------------------------------------------- /server/upload/cover/27.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/27.jpg -------------------------------------------------------------------------------- /server/upload/cover/28.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/28.jpg -------------------------------------------------------------------------------- /server/upload/cover/29.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/29.jpg -------------------------------------------------------------------------------- /server/upload/cover/3.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/3.jpg -------------------------------------------------------------------------------- /server/upload/cover/30.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/30.jpg -------------------------------------------------------------------------------- /server/upload/cover/4.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/4.jpg -------------------------------------------------------------------------------- /server/upload/cover/5.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/5.jpg -------------------------------------------------------------------------------- /server/upload/cover/6.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/6.jpg -------------------------------------------------------------------------------- /server/upload/cover/7.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/7.jpg -------------------------------------------------------------------------------- /server/upload/cover/8.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/8.jpg -------------------------------------------------------------------------------- /server/upload/cover/9.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/cover/9.jpg -------------------------------------------------------------------------------- /server/upload/img/Wechat.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/img/Wechat.jpeg -------------------------------------------------------------------------------- /server/upload/img/a.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/img/a.png -------------------------------------------------------------------------------- /server/upload/img/b.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/img/b.png -------------------------------------------------------------------------------- /server/upload/img/weixin.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/server/upload/img/weixin.png -------------------------------------------------------------------------------- /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 | 6 | 1. 修改constants.ts中的BASE_URL 7 | 2. vite build 8 | 3. 将dist部署到nginx 9 | 10 | 11 | ### 配置解释 12 | 13 | 1. env.development 开发环境配置 14 | 2. eslintrc.js 代码规范化提示 15 | 3. vite.config.js vite 开发服务器配置 16 | 17 | 18 | 19 | ### 开发步骤 20 | 21 | 共分三步:开发接口、开发view、开发路由 22 | 23 | 第一步:开发接口 24 | 25 | 在api文件夹下,新建请求文件,然后写入api请求代码,如下 26 | 27 | ``` 28 | import { get, post } from '/@/utils/http/axios'; 29 | 30 | enum URL { 31 | list = '/myapp/admin/notice/list', 32 | create = '/myapp/admin/notice/create', 33 | update = '/myapp/admin/notice/update', 34 | delete = '/myapp/admin/notice/delete', 35 | } 36 | 37 | const listApi = async (params: any) => get({ url: URL.list, params: params, data: {}, headers: {} }); 38 | const createApi = async (data: any) => 39 | post({ 40 | url: URL.create, 41 | params: {}, 42 | data: data, 43 | headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' }, 44 | }); 45 | const updateApi = async (params: any, data: any) => 46 | post({ 47 | url: URL.update, 48 | params: params, 49 | data: data, 50 | headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' }, 51 | }); 52 | const deleteApi = async (params: any) => post({ url: URL.delete, params: params, headers: {} }); 53 | 54 | export { listApi, createApi, updateApi, deleteApi }; 55 | 56 | ``` 57 | 58 | 第二步:开发view页面 59 | 60 | 在view文件夹下,新增页面vue文件,然后写入页面代码,比如user.vue 61 | 62 | 第三步:设置路由 63 | 64 | 在router的root.js文件里面配置路由地址。如下 65 | 66 | ``` 67 | { path: 'overview', name: 'overview', component: () => import('/@/views/overview.vue') }, 68 | { path: 'asset', name: 'asset', component: () => import('/@/views/asset.vue') }, 69 | { path: 'thing', name: 'thing', component: () => import('/@/views/thing.vue') }, 70 | { path: 'comment', name: 'comment', component: () => import('/@/views/comment.vue') }, 71 | { path: 'user', name: 'user', component: () => import('/@/views/user.vue') }, 72 | ``` 73 | 74 | 即可 75 | 76 | ### 常见问题 77 | 78 | ### 变量 79 | https://blog.csdn.net/qq_41636947/article/details/117907448 80 | 81 | ### antd的css引入方式 82 | 在index.html里面引入的cdn 83 | 84 | ### cdn 85 | https://cdn.jsdelivr.net/npm/ant-design-vue@3.2.20/dist/ 86 | https://cdn.staticfile.org/ant-design-vue/3.2.20/antd.min.css 87 | 88 | ### public文件夹内容在build后会自动打到dist中 89 | -------------------------------------------------------------------------------- /web/build/constant.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * @name Config 3 | * @description 项目配置 4 | */ 5 | 6 | 7 | // 本地服务端口 8 | export const VITE_PORT = 8008; 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 | 20 | 21 | 22 | 23 | 24 |
25 | 26 | 27 | 28 | -------------------------------------------------------------------------------- /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_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/public/favicon.ico -------------------------------------------------------------------------------- /web/public/images/admin-login-bg.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/public/images/admin-login-bg.jpg -------------------------------------------------------------------------------- /web/public/images/bg2.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/public/images/bg2.jpg -------------------------------------------------------------------------------- /web/public/images/demo.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/public/images/demo.jpg -------------------------------------------------------------------------------- /web/src/App.vue: -------------------------------------------------------------------------------- 1 | 6 | 11 | 12 | 19 | -------------------------------------------------------------------------------- /web/src/api/admin/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/admin/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/admin/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/admin/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/admin/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/admin/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/admin/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/admin/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/admin/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, 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/admin/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/api/index/address.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/index/address/list', 5 | create = '/myapp/index/address/create', 6 | update = '/myapp/index/address/update', 7 | delete = '/myapp/index/address/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/index/classification.ts: -------------------------------------------------------------------------------- 1 | import { get, post } from '/@/utils/http/axios'; 2 | enum URL { 3 | list = '/myapp/index/classification/list', 4 | } 5 | 6 | const listApi = async (params: any) => get({ url: URL.list, params: params, data: {}, headers: {} }); 7 | 8 | export { listApi}; 9 | -------------------------------------------------------------------------------- /web/src/api/index/comment.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | create = '/myapp/index/comment/create', 5 | listThingComments = '/myapp/index/comment/list', 6 | listUserComments = '/myapp/index/comment/listMyComments', 7 | like = '/myapp/index/comment/like' 8 | } 9 | 10 | const createApi = async (data: any) => post({ 11 | url: URL.create, 12 | params: {}, 13 | data: data, 14 | headers: {'Content-Type': 'multipart/form-data;charset=utf-8'} 15 | }); 16 | const listThingCommentsApi = async (params: any) => get({url: URL.listThingComments, params: params, data: {}, headers: {}}); 17 | const listUserCommentsApi = async (params: any) => get({url: URL.listUserComments, params: params, data: {}, headers: {}}); 18 | const likeApi = async (params: any) => post({url: URL.like, params: params, headers: {}}); 19 | 20 | export {createApi, listThingCommentsApi,listUserCommentsApi, likeApi}; 21 | -------------------------------------------------------------------------------- /web/src/api/index/notice.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/index/notice/list_api', 5 | } 6 | 7 | const listApi = async (params: any) => 8 | get({url: URL.list, params: params, data: {}, headers: {}}); 9 | 10 | export {listApi}; 11 | -------------------------------------------------------------------------------- /web/src/api/index/order.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | create='/myapp/index/order/create', 5 | cancelUserOrder = '/myapp/index/order/cancel_order', 6 | userOrderList = '/myapp/index/order/list', 7 | } 8 | 9 | const createApi = async (data: any) => 10 | post({url: URL.create, data: data, headers: {}}); 11 | 12 | const userOrderListApi = async (params: any) => 13 | get({url: URL.userOrderList, params: params, data: {}, headers: {}}); 14 | 15 | const cancelUserOrderApi = async (params: any) => 16 | post({url: URL.cancelUserOrder, params: params, headers: {}}); 17 | 18 | export {createApi, userOrderListApi, cancelUserOrderApi}; 19 | -------------------------------------------------------------------------------- /web/src/api/index/tag.ts: -------------------------------------------------------------------------------- 1 | import {get, post} from '/@/utils/http/axios'; 2 | 3 | enum URL { 4 | list = '/myapp/index/tag/list', 5 | } 6 | 7 | const listApi = async (params: any) => 8 | get({url: URL.list, params: params, data: {}, headers: {}}); 9 | 10 | export {listApi}; 11 | -------------------------------------------------------------------------------- /web/src/api/index/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/index/thing/list', 7 | detail = '/myapp/index/thing/detail', 8 | addWishUser = '/myapp/index/thing/addWishUser', 9 | addCollectUser = '/myapp/index/thing/addCollectUser', 10 | getCollectThingList = '/myapp/index/thing/getCollectThingList', 11 | getWishThingList = '/myapp/index/thing/getWishThingList', 12 | removeCollectUser = '/myapp/index/thing/removeCollectUser', 13 | removeWishUser = '/myapp/index/thing/removeWishUser' 14 | } 15 | 16 | const listApi = async (params: any) => get({ url: URL.list, params: params, data: {}, headers: {} }); 17 | const detailApi = async (params: any) => get({ url: URL.detail, params: params, headers: {} }); 18 | const addWishUserApi = async (params: any) => post({ url: URL.addWishUser, params: params, headers: {} }); 19 | const addCollectUserApi = async (params: any) => post({ url: URL.addCollectUser, params: params, headers: {} }); 20 | const getCollectThingListApi = async (params: any) => get({ url: URL.getCollectThingList, params: params, headers: {} }); 21 | const getWishThingListApi = async (params: any) => get({ url: URL.getWishThingList, params: params, headers: {} }); 22 | 23 | const removeCollectUserApi = async (params: any) => post({ url: URL.removeCollectUser, params: params, headers: {} }); 24 | const removeWishUserApi = async (params: any) => post({ url: URL.removeWishUser, params: params, headers: {} }); 25 | 26 | 27 | export { listApi, detailApi, addWishUserApi,addCollectUserApi, getCollectThingListApi, 28 | getWishThingListApi, removeCollectUserApi, removeWishUserApi }; 29 | -------------------------------------------------------------------------------- /web/src/api/index/thingCollect.ts: -------------------------------------------------------------------------------- 1 | 2 | import { get, post } from '/@/utils/http/axios'; 3 | import { UserState } from '/@/store/modules/user/types'; 4 | 5 | enum URL { 6 | userCollectList = '/api/thingCollect/getUserCollectList', 7 | collect = '/api/thingCollect/collect', 8 | unCollect = '/api/thingCollect/unCollect', 9 | } 10 | 11 | const collectApi = async (data: any) => post({ url: URL.collect, params: {}, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 12 | const unCollectApi = async (params: any) => post({ url: URL.unCollect, params: params, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 13 | const userCollectListApi = async (params: any) => get({ url: URL.userCollectList, params: params }); 14 | 15 | export { collectApi, unCollectApi, userCollectListApi }; 16 | -------------------------------------------------------------------------------- /web/src/api/index/thingWish.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 | userWishList = '/api/thingWish/getUserWishList', 7 | wish = '/api/thingWish/wish', 8 | unWish = '/api/thingWish/unWish', 9 | } 10 | 11 | const wishApi = async (data: any) => post({ url: URL.wish, params: {}, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 12 | const unWishApi = async (params: any) => post({ url: URL.unWish, params: params, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 13 | const userWishListApi = async (params: any) => get({ url: URL.userWishList, params: params }); 14 | 15 | export { wishApi, unWishApi, userWishListApi }; 16 | -------------------------------------------------------------------------------- /web/src/api/index/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 | userLogin = '/myapp/index/user/login', 7 | userRegister = '/myapp/index/user/register', 8 | detail = '/myapp/index/user/info', 9 | updateUserPwd = '/myapp/index/user/updatePwd', 10 | updateUserInfo = '/myapp/index/user/update' 11 | } 12 | interface LoginRes { 13 | token: string; 14 | } 15 | 16 | export interface LoginData { 17 | username: string; 18 | password: string; 19 | } 20 | 21 | const detailApi = async (params: any) => get({ url: URL.detail, params: params, data: {}, headers: {} }); 22 | const userLoginApi = async (data: LoginData) => post({ url: URL.userLogin, data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 23 | const userRegisterApi = async (data: any) => post({ url: URL.userRegister, params: {}, data: data }); 24 | const updateUserPwdApi = async (params: any, data:any) => post({ url: URL.updateUserPwd, params: params, data:data }); 25 | const updateUserInfoApi = async (params: any,data: any) => post({ url: URL.updateUserInfo, params:params, data: data, headers: { 'Content-Type': 'multipart/form-data;charset=utf-8' } }); 26 | 27 | export { detailApi, userLoginApi, userRegisterApi, updateUserPwdApi, updateUserInfoApi}; 28 | -------------------------------------------------------------------------------- /web/src/assets/fonts/Blimone-ExtraBold.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/src/assets/fonts/Blimone-ExtraBold.woff -------------------------------------------------------------------------------- /web/src/assets/fonts/Blimone-ExtraLight.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/src/assets/fonts/Blimone-ExtraLight.woff -------------------------------------------------------------------------------- /web/src/assets/fonts/Blimone-Light.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/src/assets/fonts/Blimone-Light.woff -------------------------------------------------------------------------------- /web/src/assets/fonts/Blimone-Regular.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/src/assets/fonts/Blimone-Regular.woff -------------------------------------------------------------------------------- /web/src/assets/icons/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/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/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_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/src/assets/images/avatar.jpg -------------------------------------------------------------------------------- /web/src/assets/images/banner-02.webp: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/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_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/src/assets/images/k-logo.png -------------------------------------------------------------------------------- /web/src/assets/images/login-banner.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/src/assets/images/login-banner.png -------------------------------------------------------------------------------- /web/src/assets/images/login.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/web/src/assets/images/login.png -------------------------------------------------------------------------------- /web/src/assets/images/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/geeeeeeeek/python_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/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_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/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_adopt/08bf66bd3fab5a991a0433cd53c9eca4265bc7d6/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 | // next() 39 | } 40 | 41 | /** 前台路由 **/ 42 | if (to.path.startsWith('/index')) { 43 | if (localStorage.getItem(USER_TOKEN)) { 44 | if (to.path === loginRoutePath) { 45 | next({ path: '/' }) 46 | } else { 47 | next() 48 | } 49 | } else { 50 | if (allowList.includes(to.name)) { 51 | // 在免登录名单,直接进入 52 | next() 53 | } else { 54 | next({ path: loginRoutePath, query: { redirect: to.fullPath } }) 55 | } 56 | } 57 | // next() 58 | } 59 | 60 | }); 61 | 62 | router.afterEach((_to) => { 63 | // 回到顶部 64 | document.getElementById("html")?.scrollTo(0, 0) 65 | }); 66 | 67 | export default router; 68 | -------------------------------------------------------------------------------- /web/src/store/constants.js: -------------------------------------------------------------------------------- 1 | const BASE_URL = 'http://127.0.0.1:8000' 2 | // const BASE_URL = 'https://adopt.gitapp.cn/api/' 3 | 4 | const USER_ID = 'user_id' 5 | const USER_NAME = 'user_name' 6 | const USER_TOKEN = 'user_token' 7 | 8 | const ADMIN_USER_ID = 'admin_user_id' 9 | const ADMIN_USER_NAME = 'admin_user_name' 10 | const ADMIN_USER_TOKEN = 'admin_user_token' 11 | 12 | 13 | export {BASE_URL, USER_TOKEN, USER_NAME, USER_ID, ADMIN_USER_ID,ADMIN_USER_NAME,ADMIN_USER_TOKEN } 14 | -------------------------------------------------------------------------------- /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'; 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/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: import.meta.env.BASE_URL + '', 11 | baseURL: BASE_URL + '', 12 | timeout: 15000, 13 | }); 14 | 15 | // axios实例拦截请求 16 | service.interceptors.request.use( 17 | (config: InternalAxiosRequestConfig) => { 18 | 19 | config.headers.ADMINTOKEN = localStorage.getItem(ADMIN_USER_TOKEN) 20 | config.headers.TOKEN = localStorage.getItem(USER_TOKEN) 21 | 22 | return config; 23 | }, 24 | (error: AxiosError) => { 25 | return Promise.reject(error); 26 | }, 27 | ); 28 | 29 | // axios实例拦截响应 30 | service.interceptors.response.use( 31 | (response: AxiosResponse) => { 32 | if(response.status == 200) { 33 | if(response.data.code == 0 || response.data.code == 200) { 34 | return response 35 | }else { 36 | return Promise.reject(response.data) 37 | } 38 | } else { 39 | return Promise.reject(response.data) 40 | } 41 | }, 42 | // 请求失败 43 | (error: any) => { 44 | console.log(error.response.status) 45 | if(error.response.status == 404) { 46 | // todo 47 | } else if(error.response.status == 403) { 48 | // todo 49 | } 50 | return Promise.reject(error) 51 | }, 52 | ); 53 | 54 | 55 | 56 | const request = (config: AxiosRequestConfig): Promise => { 57 | const conf = config; 58 | return new Promise((resolve, reject) => { 59 | service.request>(conf).then((res: AxiosResponse) => { 60 | const data = res.data 61 | resolve(data as T); 62 | }).catch(err => { 63 | reject(err) 64 | }); 65 | }); 66 | }; 67 | 68 | export function get(config: AxiosRequestConfig): Promise { 69 | return request({ ...config, method: 'GET' }); 70 | } 71 | 72 | export function post(config: AxiosRequestConfig): Promise { 73 | return request({ ...config, method: 'POST' }); 74 | } 75 | 76 | export default request; 77 | 78 | export type { AxiosInstance, AxiosResponse }; 79 | -------------------------------------------------------------------------------- /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/admin/sys-info.vue: -------------------------------------------------------------------------------- 1 | 55 | 56 | 80 | 81 | 99 | -------------------------------------------------------------------------------- /web/src/views/index/components/footer.vue: -------------------------------------------------------------------------------- 1 | 19 | 20 | 39 | 40 | 94 | -------------------------------------------------------------------------------- /web/src/views/index/index.vue: -------------------------------------------------------------------------------- 1 | 4 | 5 | 9 | -------------------------------------------------------------------------------- /web/src/views/index/portal.vue: -------------------------------------------------------------------------------- 1 | 8 | 25 | 28 | -------------------------------------------------------------------------------- /web/src/views/index/search.vue: -------------------------------------------------------------------------------- 1 | 10 | 11 | 17 | 18 | 25 | -------------------------------------------------------------------------------- /web/src/views/index/user/fans-view.vue: -------------------------------------------------------------------------------- 1 | 16 | 17 | 27 | 104 | -------------------------------------------------------------------------------- /web/src/views/index/user/follow-view.vue: -------------------------------------------------------------------------------- 1 | 17 | 18 | 28 | 105 | -------------------------------------------------------------------------------- /web/src/views/index/user/score-view.vue: -------------------------------------------------------------------------------- 1 | 9 | 10 | 35 | 68 | -------------------------------------------------------------------------------- /web/src/views/index/usercenter.vue: -------------------------------------------------------------------------------- 1 | 19 | 37 | 59 | -------------------------------------------------------------------------------- /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/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 | --------------------------------------------------------------------------------