├── .gitignore ├── .vscode └── launch.json ├── Dockerfile ├── LICENSE ├── Pipfile ├── Pipfile.lock ├── README.md ├── app ├── __init__.py ├── config.py ├── log.py ├── main.py ├── models │ ├── __init__.py │ ├── vip_video_parse.py │ └── website_navigation.py ├── routes │ ├── __init__.py │ ├── home.py │ ├── log_list.py │ ├── robot.py │ └── robot_handler │ │ ├── __init__.py │ │ ├── image_handler.py │ │ └── text_handler.py ├── static │ ├── css │ │ ├── bootstrap.css │ │ ├── bootstrap.min.css │ │ └── style.css │ ├── img │ │ ├── 56logo.png │ │ ├── acfun.png │ │ ├── beian.png │ │ ├── bilibili.png │ │ ├── favicon.png │ │ ├── fengxing.gif │ │ ├── hunantvlogo.png │ │ ├── iqiyi.png │ │ ├── letvlogo.png │ │ ├── logo-collapsed@2x.png │ │ ├── logo@2x.png │ │ ├── logo_dark@2x.png │ │ ├── qqlogo.png │ │ ├── sohulogo.png │ │ ├── tudoulogo.png │ │ ├── wasulogo.png │ │ ├── webstack_banner_cn.png │ │ ├── webstack_icon_producthunt.png │ │ ├── yinyuetailogo.png │ │ ├── ykcloud.png │ │ └── youkulogo.png │ └── js │ │ ├── bootstrap.min.js │ │ └── jquery.min.js ├── templates │ ├── index.html │ └── log_list.html ├── third_party │ ├── __init__.py │ ├── movie.py │ └── ocr.py └── util │ ├── __init__.py │ └── requests_helper.py ├── docker-compose.yaml ├── docs └── image │ └── coderbox.jpg ├── gunicorn.conf.py ├── nginx_flask.conf ├── requirements.txt ├── script ├── compose-deploy.sh ├── docker-deploy.sh └── run.sh └── supervisord.conf /.gitignore: -------------------------------------------------------------------------------- 1 | __pycache__ 2 | logs 3 | *.db 4 | migrations 5 | .env 6 | -------------------------------------------------------------------------------- /.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "name": "Python: Flask", 9 | "type": "python", 10 | "request": "launch", 11 | "module": "flask", 12 | "env": { 13 | "FLASK_APP": "app:create_app('development')", 14 | "FLASK_ENV": "development" 15 | }, 16 | "args": [ 17 | "run", 18 | "-p", 19 | "8080" 20 | ], 21 | "jinja": true 22 | } 23 | ] 24 | } -------------------------------------------------------------------------------- /Dockerfile: -------------------------------------------------------------------------------- 1 | # syntax=docker/dockerfile:1 2 | FROM python:3.7-slim-buster 3 | 4 | RUN mkdir /deploy 5 | # 设置工作目录(后续都不需要指定完整路径了,默认工作目录/deploy) 6 | WORKDIR /deploy 7 | 8 | # 复制源码 (使用映射代替copy) 9 | #COPY ./app ./deploy/app 10 | 11 | # 复制发布相关脚本 12 | COPY requirements.txt requirements.txt 13 | COPY gunicorn.conf.py gunicorn.conf.py 14 | 15 | # 安装依赖 16 | RUN pip3 install --upgrade pip -i https://pypi.tuna.tsinghua.edu.cn/simple 17 | RUN pip3 install -r requirements.txt -i https://pypi.tuna.tsinghua.edu.cn/simple 18 | RUN pip3 install gunicorn -i https://pypi.tuna.tsinghua.edu.cn/simple 19 | RUN pip3 install gevent -i https://pypi.tuna.tsinghua.edu.cn/simple 20 | 21 | # 更新apt源 22 | RUN sed -i 's/deb.debian.org/mirrors.ustc.edu.cn/g' /etc/apt/sources.list 23 | RUN apt-get update && apt-get install -y nginx supervisor 24 | 25 | # 配置nginx 26 | RUN rm /etc/nginx/sites-enabled/default 27 | COPY nginx_flask.conf /etc/nginx/sites-available/ 28 | RUN ln -s /etc/nginx/sites-available/nginx_flask.conf /etc/nginx/sites-enabled/nginx_flask.conf 29 | # 是否后台启动:决定启动nginx命令是否block,因为supervisor不能监控后台进程,command 不能为后台运行命令 30 | # 用supervisor启动nginx的话需要关掉daemon 31 | RUN echo "daemon off;" >> /etc/nginx/nginx.conf 32 | 33 | # supervisord https://www.jianshu.com/p/0b9054b33db3 34 | RUN mkdir -p /var/log/supervisor 35 | COPY supervisord.conf /etc/supervisor/conf.d/supervisord.conf 36 | 37 | # 复制运行脚本 38 | COPY ./script/run.sh ./run.sh 39 | RUN chmod +x ./run.sh 40 | 41 | # 设置环境变量(给flask迁移db用) 42 | ENV FLASK_APP="app:create_app('production')" 43 | ENV FLASK_ENV="production" 44 | 45 | # 运行(最后这条CMD命令需要阻塞,否则docker启动后接着退出) 46 | CMD ["./run.sh"] 47 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2021 Barry 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /Pipfile: -------------------------------------------------------------------------------- 1 | [[source]] 2 | url = "https://pypi.tuna.tsinghua.edu.cn/simple" 3 | verify_ssl = true 4 | name = "pypi" 5 | 6 | [packages] 7 | flask = "1.1.2" 8 | flask-apscheduler = "1.11.0" 9 | flask-sqlalchemy = "2.4.4" 10 | flask-migrate = "2.7.0" 11 | flask-executor = "0.9.3" 12 | pymysql = "1.0.2" 13 | python-dotenv = "0.19.0" 14 | requests = "2.25.1" 15 | werobot = "1.13.1" 16 | baidu-aip = "4.15.4" 17 | lxml = "4.2.1" 18 | 19 | [dev-packages] 20 | autopep8 = "*" 21 | 22 | [requires] 23 | python_version = "3.7" 24 | -------------------------------------------------------------------------------- /Pipfile.lock: -------------------------------------------------------------------------------- 1 | { 2 | "_meta": { 3 | "hash": { 4 | "sha256": "678ea63e2ec259a838f62e2d20b73e1ff16f94fdd783372d44b8590fd1e8df38" 5 | }, 6 | "pipfile-spec": 6, 7 | "requires": { 8 | "python_version": "3.7" 9 | }, 10 | "sources": [ 11 | { 12 | "name": "pypi", 13 | "url": "https://pypi.tuna.tsinghua.edu.cn/simple", 14 | "verify_ssl": true 15 | } 16 | ] 17 | }, 18 | "default": { 19 | "alembic": { 20 | "hashes": [ 21 | "sha256:a21fedebb3fb8f6bbbba51a11114f08c78709377051384c9c5ead5705ee93a51", 22 | "sha256:e78be5b919f5bb184e3e0e2dd1ca986f2362e29a2bc933c446fe89f39dbe4e9c" 23 | ], 24 | "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5'", 25 | "version": "==1.6.5" 26 | }, 27 | "apscheduler": { 28 | "hashes": [ 29 | "sha256:1cab7f2521e107d07127b042155b632b7a1cd5e02c34be5a28ff62f77c900c6a", 30 | "sha256:c06cc796d5bb9eb3c4f77727f6223476eb67749e7eea074d1587550702a7fbe3" 31 | ], 32 | "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4'", 33 | "version": "==3.7.0" 34 | }, 35 | "baidu-aip": { 36 | "hashes": [ 37 | "sha256:c7f03c671027dcae1ccaa60631ec5811c8e9e51e2ebb505a748a213d0f0b5b88" 38 | ], 39 | "index": "pypi", 40 | "version": "==2.2.18.0" 41 | }, 42 | "bottle": { 43 | "hashes": [ 44 | "sha256:a9d73ffcbc6a1345ca2d7949638db46349f5b2b77dac65d6494d45c23628da2c", 45 | "sha256:f6b8a34fe9aa406f9813c02990db72ca69ce6a158b5b156d2c41f345016a723d" 46 | ], 47 | "version": "==0.12.19" 48 | }, 49 | "certifi": { 50 | "hashes": [ 51 | "sha256:2bbf76fd432960138b3ef6dda3dde0544f27cbf8546c458e60baf371917ba9ee", 52 | "sha256:50b1e4f8446b06f41be7dd6338db18e0990601dce795c2b1686458aa7e8fa7d8" 53 | ], 54 | "version": "==2021.5.30" 55 | }, 56 | "charset-normalizer": { 57 | "hashes": [ 58 | "sha256:0c8911edd15d19223366a194a513099a302055a962bca2cec0f54b8b63175d8b", 59 | "sha256:f23667ebe1084be45f6ae0538e4a5a865206544097e4e8bbcacf42cd02a348f3" 60 | ], 61 | "markers": "python_version >= '3'", 62 | "version": "==2.0.4" 63 | }, 64 | "click": { 65 | "hashes": [ 66 | "sha256:8c04c11192119b1ef78ea049e0a6f0463e4c48ef00a30160c704337586f3ad7a", 67 | "sha256:fba402a4a47334742d782209a7c79bc448911afe1149d07bdabdf480b3e2f4b6" 68 | ], 69 | "markers": "python_version >= '3.6'", 70 | "version": "==8.0.1" 71 | }, 72 | "colorama": { 73 | "hashes": [ 74 | "sha256:5941b2b48a20143d2267e95b1c2a7603ce057ee39fd88e7329b0c292aa16869b", 75 | "sha256:9f47eda37229f68eee03b24b9748937c7dc3868f906e8ba69fbcbdd3bc5dc3e2" 76 | ], 77 | "markers": "platform_system == 'Windows'", 78 | "version": "==0.4.4" 79 | }, 80 | "flask": { 81 | "hashes": [ 82 | "sha256:1c4c257b1892aec1398784c63791cbaa43062f1f7aeb555c4da961b20ee68f55", 83 | "sha256:a6209ca15eb63fc9385f38e452704113d679511d9574d09b2cf9183ae7d20dc9" 84 | ], 85 | "index": "pypi", 86 | "version": "==2.0.1" 87 | }, 88 | "flask-apscheduler": { 89 | "hashes": [ 90 | "sha256:b9fe174b90d201d8beeba5522b023208f7bb6e2583fc02fea4be4bce5ee8f9e5" 91 | ], 92 | "index": "pypi", 93 | "version": "==1.12.2" 94 | }, 95 | "flask-executor": { 96 | "hashes": [ 97 | "sha256:11fab654cb0acfc8eefbff74f6b916622f18fa1eb9466daeac1d105c159fd728", 98 | "sha256:31b4a1141515214abc67b5f35507e5625ca33894f897afd422d5bfe223804242" 99 | ], 100 | "index": "pypi", 101 | "version": "==0.9.4" 102 | }, 103 | "flask-migrate": { 104 | "hashes": [ 105 | "sha256:57d6060839e3a7f150eaab6fe4e726d9e3e7cffe2150fb223d73f92421c6d1d9", 106 | "sha256:a6498706241aba6be7a251078de9cf166d74307bca41a4ca3e403c9d39e2f897" 107 | ], 108 | "index": "pypi", 109 | "version": "==3.1.0" 110 | }, 111 | "flask-sqlalchemy": { 112 | "hashes": [ 113 | "sha256:2bda44b43e7cacb15d4e05ff3cc1f8bc97936cc464623424102bfc2c35e95912", 114 | "sha256:f12c3d4cc5cc7fdcc148b9527ea05671718c3ea45d50c7e732cceb33f574b390" 115 | ], 116 | "index": "pypi", 117 | "version": "==2.5.1" 118 | }, 119 | "greenlet": { 120 | "hashes": [ 121 | "sha256:03f28a5ea20201e70ab70518d151116ce939b412961c33827519ce620957d44c", 122 | "sha256:06d7ac89e6094a0a8f8dc46aa61898e9e1aec79b0f8b47b2400dd51a44dbc832", 123 | "sha256:06ecb43b04480e6bafc45cb1b4b67c785e183ce12c079473359e04a709333b08", 124 | "sha256:096cb0217d1505826ba3d723e8981096f2622cde1eb91af9ed89a17c10aa1f3e", 125 | "sha256:0c557c809eeee215b87e8a7cbfb2d783fb5598a78342c29ade561440abae7d22", 126 | "sha256:0de64d419b1cb1bfd4ea544bedea4b535ef3ae1e150b0f2609da14bbf48a4a5f", 127 | "sha256:14927b15c953f8f2d2a8dffa224aa78d7759ef95284d4c39e1745cf36e8cdd2c", 128 | "sha256:16183fa53bc1a037c38d75fdc59d6208181fa28024a12a7f64bb0884434c91ea", 129 | "sha256:206295d270f702bc27dbdbd7651e8ebe42d319139e0d90217b2074309a200da8", 130 | "sha256:22002259e5b7828b05600a762579fa2f8b33373ad95a0ee57b4d6109d0e589ad", 131 | "sha256:2325123ff3a8ecc10ca76f062445efef13b6cf5a23389e2df3c02a4a527b89bc", 132 | "sha256:258f9612aba0d06785143ee1cbf2d7361801c95489c0bd10c69d163ec5254a16", 133 | "sha256:3096286a6072553b5dbd5efbefc22297e9d06a05ac14ba017233fedaed7584a8", 134 | "sha256:3d13da093d44dee7535b91049e44dd2b5540c2a0e15df168404d3dd2626e0ec5", 135 | "sha256:408071b64e52192869129a205e5b463abda36eff0cebb19d6e63369440e4dc99", 136 | "sha256:598bcfd841e0b1d88e32e6a5ea48348a2c726461b05ff057c1b8692be9443c6e", 137 | "sha256:5d928e2e3c3906e0a29b43dc26d9b3d6e36921eee276786c4e7ad9ff5665c78a", 138 | "sha256:5f75e7f237428755d00e7460239a2482fa7e3970db56c8935bd60da3f0733e56", 139 | "sha256:60848099b76467ef09b62b0f4512e7e6f0a2c977357a036de602b653667f5f4c", 140 | "sha256:6b1d08f2e7f2048d77343279c4d4faa7aef168b3e36039cba1917fffb781a8ed", 141 | "sha256:70bd1bb271e9429e2793902dfd194b653221904a07cbf207c3139e2672d17959", 142 | "sha256:76ed710b4e953fc31c663b079d317c18f40235ba2e3d55f70ff80794f7b57922", 143 | "sha256:7920e3eccd26b7f4c661b746002f5ec5f0928076bd738d38d894bb359ce51927", 144 | "sha256:7db68f15486d412b8e2cfcd584bf3b3a000911d25779d081cbbae76d71bd1a7e", 145 | "sha256:8833e27949ea32d27f7e96930fa29404dd4f2feb13cce483daf52e8842ec246a", 146 | "sha256:944fbdd540712d5377a8795c840a97ff71e7f3221d3fddc98769a15a87b36131", 147 | "sha256:9a6b035aa2c5fcf3dbbf0e3a8a5bc75286fc2d4e6f9cfa738788b433ec894919", 148 | "sha256:9bdcff4b9051fb1aa4bba4fceff6a5f770c6be436408efd99b76fc827f2a9319", 149 | "sha256:a9017ff5fc2522e45562882ff481128631bf35da444775bc2776ac5c61d8bcae", 150 | "sha256:aa4230234d02e6f32f189fd40b59d5a968fe77e80f59c9c933384fe8ba535535", 151 | "sha256:ad80bb338cf9f8129c049837a42a43451fc7c8b57ad56f8e6d32e7697b115505", 152 | "sha256:adb94a28225005890d4cf73648b5131e885c7b4b17bc762779f061844aabcc11", 153 | "sha256:b3090631fecdf7e983d183d0fad7ea72cfb12fa9212461a9b708ff7907ffff47", 154 | "sha256:b33b51ab057f8a20b497ffafdb1e79256db0c03ef4f5e3d52e7497200e11f821", 155 | "sha256:b97c9a144bbeec7039cca44df117efcbeed7209543f5695201cacf05ba3b5857", 156 | "sha256:be13a18cec649ebaab835dff269e914679ef329204704869f2f167b2c163a9da", 157 | "sha256:be9768e56f92d1d7cd94185bab5856f3c5589a50d221c166cc2ad5eb134bd1dc", 158 | "sha256:c1580087ab493c6b43e66f2bdd165d9e3c1e86ef83f6c2c44a29f2869d2c5bd5", 159 | "sha256:c35872b2916ab5a240d52a94314c963476c989814ba9b519bc842e5b61b464bb", 160 | "sha256:c70c7dd733a4c56838d1f1781e769081a25fade879510c5b5f0df76956abfa05", 161 | "sha256:c767458511a59f6f597bfb0032a1c82a52c29ae228c2c0a6865cfeaeaac4c5f5", 162 | "sha256:c87df8ae3f01ffb4483c796fe1b15232ce2b219f0b18126948616224d3f658ee", 163 | "sha256:ca1c4a569232c063615f9e70ff9a1e2fee8c66a6fb5caf0f5e8b21a396deec3e", 164 | "sha256:cc407b68e0a874e7ece60f6639df46309376882152345508be94da608cc0b831", 165 | "sha256:da862b8f7de577bc421323714f63276acb2f759ab8c5e33335509f0b89e06b8f", 166 | "sha256:dfe7eac0d253915116ed0cd160a15a88981a1d194c1ef151e862a5c7d2f853d3", 167 | "sha256:ed1377feed808c9c1139bdb6a61bcbf030c236dd288d6fca71ac26906ab03ba6", 168 | "sha256:f42ad188466d946f1b3afc0a9e1a266ac8926461ee0786c06baac6bd71f8a6f3", 169 | "sha256:f92731609d6625e1cc26ff5757db4d32b6b810d2a3363b0ff94ff573e5901f6f" 170 | ], 171 | "markers": "python_version >= '3'", 172 | "version": "==1.1.0" 173 | }, 174 | "idna": { 175 | "hashes": [ 176 | "sha256:14475042e284991034cb48e06f6851428fb14c4dc953acd9be9a5e95c7b6dd7a", 177 | "sha256:467fbad99067910785144ce333826c71fb0e63a425657295239737f7ecd125f3" 178 | ], 179 | "markers": "python_version >= '3'", 180 | "version": "==3.2" 181 | }, 182 | "importlib-metadata": { 183 | "hashes": [ 184 | "sha256:0645585859e9a6689c523927a5032f2ba5919f1f7d0e84bd4533312320de1ff9", 185 | "sha256:51c6635429c77cf1ae634c997ff9e53ca3438b495f10a55ba28594dd69764a8b" 186 | ], 187 | "markers": "python_version < '3.8'", 188 | "version": "==4.6.3" 189 | }, 190 | "itsdangerous": { 191 | "hashes": [ 192 | "sha256:5174094b9637652bdb841a3029700391451bd092ba3db90600dea710ba28e97c", 193 | "sha256:9e724d68fc22902a1435351f84c3fb8623f303fffcc566a4cb952df8c572cff0" 194 | ], 195 | "markers": "python_version >= '3.6'", 196 | "version": "==2.0.1" 197 | }, 198 | "jinja2": { 199 | "hashes": [ 200 | "sha256:1f06f2da51e7b56b8f238affdd6b4e2c61e39598a378cc49345bc1bd42a978a4", 201 | "sha256:703f484b47a6af502e743c9122595cc812b0271f661722403114f71a79d0f5a4" 202 | ], 203 | "markers": "python_version >= '3.6'", 204 | "version": "==3.0.1" 205 | }, 206 | "lxml": { 207 | "hashes": [ 208 | "sha256:079f3ae844f38982d156efce585bc540c16a926d4436712cf4baee0cce487a3d", 209 | "sha256:0fbcf5565ac01dff87cbfc0ff323515c823081c5777a9fc7703ff58388c258c3", 210 | "sha256:122fba10466c7bd4178b07dba427aa516286b846b2cbd6f6169141917283aae2", 211 | "sha256:1b38116b6e628118dea5b2186ee6820ab138dbb1e24a13e478490c7db2f326ae", 212 | "sha256:1b7584d421d254ab86d4f0b13ec662a9014397678a7c4265a02a6d7c2b18a75f", 213 | "sha256:26e761ab5b07adf5f555ee82fb4bfc35bf93750499c6c7614bd64d12aaa67927", 214 | "sha256:289e9ca1a9287f08daaf796d96e06cb2bc2958891d7911ac7cae1c5f9e1e0ee3", 215 | "sha256:2a9d50e69aac3ebee695424f7dbd7b8c6d6eb7de2a2eb6b0f6c7db6aa41e02b7", 216 | "sha256:3082c518be8e97324390614dacd041bb1358c882d77108ca1957ba47738d9d59", 217 | "sha256:33bb934a044cf32157c12bfcfbb6649807da20aa92c062ef51903415c704704f", 218 | "sha256:3439c71103ef0e904ea0a1901611863e51f50b5cd5e8654a151740fde5e1cade", 219 | "sha256:36108c73739985979bf302006527cf8a20515ce444ba916281d1c43938b8bb96", 220 | "sha256:39b78571b3b30645ac77b95f7c69d1bffc4cf8c3b157c435a34da72e78c82468", 221 | "sha256:4289728b5e2000a4ad4ab8da6e1db2e093c63c08bdc0414799ee776a3f78da4b", 222 | "sha256:4bff24dfeea62f2e56f5bab929b4428ae6caba2d1eea0c2d6eb618e30a71e6d4", 223 | "sha256:4c61b3a0db43a1607d6264166b230438f85bfed02e8cff20c22e564d0faff354", 224 | "sha256:542d454665a3e277f76954418124d67516c5f88e51a900365ed54a9806122b83", 225 | "sha256:5a0a14e264069c03e46f926be0d8919f4105c1623d620e7ec0e612a2e9bf1c04", 226 | "sha256:5c8c163396cc0df3fd151b927e74f6e4acd67160d6c33304e805b84293351d16", 227 | "sha256:66e575c62792c3f9ca47cb8b6fab9e35bab91360c783d1606f758761810c9791", 228 | "sha256:6f12e1427285008fd32a6025e38e977d44d6382cf28e7201ed10d6c1698d2a9a", 229 | "sha256:74f7d8d439b18fa4c385f3f5dfd11144bb87c1da034a466c5b5577d23a1d9b51", 230 | "sha256:7610b8c31688f0b1be0ef882889817939490a36d0ee880ea562a4e1399c447a1", 231 | "sha256:76fa7b1362d19f8fbd3e75fe2fb7c79359b0af8747e6f7141c338f0bee2f871a", 232 | "sha256:7728e05c35412ba36d3e9795ae8995e3c86958179c9770e65558ec3fdfd3724f", 233 | "sha256:8157dadbb09a34a6bd95a50690595e1fa0af1a99445e2744110e3dca7831c4ee", 234 | "sha256:820628b7b3135403540202e60551e741f9b6d3304371712521be939470b454ec", 235 | "sha256:884ab9b29feaca361f7f88d811b1eea9bfca36cf3da27768d28ad45c3ee6f969", 236 | "sha256:89b8b22a5ff72d89d48d0e62abb14340d9e99fd637d046c27b8b257a01ffbe28", 237 | "sha256:92e821e43ad382332eade6812e298dc9701c75fe289f2a2d39c7960b43d1e92a", 238 | "sha256:b007cbb845b28db4fb8b6a5cdcbf65bacb16a8bd328b53cbc0698688a68e1caa", 239 | "sha256:bc4313cbeb0e7a416a488d72f9680fffffc645f8a838bd2193809881c67dd106", 240 | "sha256:bccbfc27563652de7dc9bdc595cb25e90b59c5f8e23e806ed0fd623755b6565d", 241 | "sha256:c47ff7e0a36d4efac9fd692cfa33fbd0636674c102e9e8d9b26e1b93a94e7617", 242 | "sha256:c4f05c5a7c49d2fb70223d0d5bcfbe474cf928310ac9fa6a7c6dddc831d0b1d4", 243 | "sha256:cdaf11d2bd275bf391b5308f86731e5194a21af45fbaaaf1d9e8147b9160ea92", 244 | "sha256:ce256aaa50f6cc9a649c51be3cd4ff142d67295bfc4f490c9134d0f9f6d58ef0", 245 | "sha256:d2e35d7bf1c1ac8c538f88d26b396e73dd81440d59c1ef8522e1ea77b345ede4", 246 | "sha256:d916d31fd85b2f78c76400d625076d9124de3e4bda8b016d25a050cc7d603f24", 247 | "sha256:df7c53783a46febb0e70f6b05df2ba104610f2fb0d27023409734a3ecbb78fb2", 248 | "sha256:e1cbd3f19a61e27e011e02f9600837b921ac661f0c40560eefb366e4e4fb275e", 249 | "sha256:efac139c3f0bf4f0939f9375af4b02c5ad83a622de52d6dfa8e438e8e01d0eb0", 250 | "sha256:efd7a09678fd8b53117f6bae4fa3825e0a22b03ef0a932e070c0bdbb3a35e654", 251 | "sha256:f2380a6376dfa090227b663f9678150ef27543483055cc327555fb592c5967e2", 252 | "sha256:f8380c03e45cf09f8557bdaa41e1fa7c81f3ae22828e1db470ab2a6c96d8bc23", 253 | "sha256:f90ba11136bfdd25cae3951af8da2e95121c9b9b93727b1b896e3fa105b2f586" 254 | ], 255 | "index": "pypi", 256 | "version": "==4.6.3" 257 | }, 258 | "mako": { 259 | "hashes": [ 260 | "sha256:17831f0b7087c313c0ffae2bcbbd3c1d5ba9eeac9c38f2eb7b50e8c99fe9d5ab", 261 | "sha256:aea166356da44b9b830c8023cd9b557fa856bd8b4035d6de771ca027dfc5cc6e" 262 | ], 263 | "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", 264 | "version": "==1.1.4" 265 | }, 266 | "markupsafe": { 267 | "hashes": [ 268 | "sha256:01a9b8ea66f1658938f65b93a85ebe8bc016e6769611be228d797c9d998dd298", 269 | "sha256:023cb26ec21ece8dc3907c0e8320058b2e0cb3c55cf9564da612bc325bed5e64", 270 | "sha256:0446679737af14f45767963a1a9ef7620189912317d095f2d9ffa183a4d25d2b", 271 | "sha256:0717a7390a68be14b8c793ba258e075c6f4ca819f15edfc2a3a027c823718567", 272 | "sha256:0955295dd5eec6cb6cc2fe1698f4c6d84af2e92de33fbcac4111913cd100a6ff", 273 | "sha256:10f82115e21dc0dfec9ab5c0223652f7197feb168c940f3ef61563fc2d6beb74", 274 | "sha256:1d609f577dc6e1aa17d746f8bd3c31aa4d258f4070d61b2aa5c4166c1539de35", 275 | "sha256:2ef54abee730b502252bcdf31b10dacb0a416229b72c18b19e24a4509f273d26", 276 | "sha256:3c112550557578c26af18a1ccc9e090bfe03832ae994343cfdacd287db6a6ae7", 277 | "sha256:47ab1e7b91c098ab893b828deafa1203de86d0bc6ab587b160f78fe6c4011f75", 278 | "sha256:49e3ceeabbfb9d66c3aef5af3a60cc43b85c33df25ce03d0031a608b0a8b2e3f", 279 | "sha256:4efca8f86c54b22348a5467704e3fec767b2db12fc39c6d963168ab1d3fc9135", 280 | "sha256:53edb4da6925ad13c07b6d26c2a852bd81e364f95301c66e930ab2aef5b5ddd8", 281 | "sha256:594c67807fb16238b30c44bdf74f36c02cdf22d1c8cda91ef8a0ed8dabf5620a", 282 | "sha256:611d1ad9a4288cf3e3c16014564df047fe08410e628f89805e475368bd304914", 283 | "sha256:6557b31b5e2c9ddf0de32a691f2312a32f77cd7681d8af66c2692efdbef84c18", 284 | "sha256:693ce3f9e70a6cf7d2fb9e6c9d8b204b6b39897a2c4a1aa65728d5ac97dcc1d8", 285 | "sha256:6a7fae0dd14cf60ad5ff42baa2e95727c3d81ded453457771d02b7d2b3f9c0c2", 286 | "sha256:6c4ca60fa24e85fe25b912b01e62cb969d69a23a5d5867682dd3e80b5b02581d", 287 | "sha256:7d91275b0245b1da4d4cfa07e0faedd5b0812efc15b702576d103293e252af1b", 288 | "sha256:905fec760bd2fa1388bb5b489ee8ee5f7291d692638ea5f67982d968366bef9f", 289 | "sha256:97383d78eb34da7e1fa37dd273c20ad4320929af65d156e35a5e2d89566d9dfb", 290 | "sha256:984d76483eb32f1bcb536dc27e4ad56bba4baa70be32fa87152832cdd9db0833", 291 | "sha256:a30e67a65b53ea0a5e62fe23682cfe22712e01f453b95233b25502f7c61cb415", 292 | "sha256:ab3ef638ace319fa26553db0624c4699e31a28bb2a835c5faca8f8acf6a5a902", 293 | "sha256:b2f4bf27480f5e5e8ce285a8c8fd176c0b03e93dcc6646477d4630e83440c6a9", 294 | "sha256:b7f2d075102dc8c794cbde1947378051c4e5180d52d276987b8d28a3bd58c17d", 295 | "sha256:be98f628055368795d818ebf93da628541e10b75b41c559fdf36d104c5787066", 296 | "sha256:d7f9850398e85aba693bb640262d3611788b1f29a79f0c93c565694658f4071f", 297 | "sha256:f5653a225f31e113b152e56f154ccbe59eeb1c7487b39b9d9f9cdb58e6c79dc5", 298 | "sha256:f826e31d18b516f653fe296d967d700fddad5901ae07c622bb3705955e1faa94", 299 | "sha256:f8ba0e8349a38d3001fae7eadded3f6606f0da5d748ee53cc1dab1d6527b9509", 300 | "sha256:f9081981fe268bd86831e5c75f7de206ef275defcb82bc70740ae6dc507aee51", 301 | "sha256:fa130dd50c57d53368c9d59395cb5526eda596d3ffe36666cd81a44d56e48872" 302 | ], 303 | "markers": "python_version >= '3.6'", 304 | "version": "==2.0.1" 305 | }, 306 | "pymysql": { 307 | "hashes": [ 308 | "sha256:41fc3a0c5013d5f039639442321185532e3e2c8924687abe6537de157d403641", 309 | "sha256:816927a350f38d56072aeca5dfb10221fe1dc653745853d30a216637f5d7ad36" 310 | ], 311 | "index": "pypi", 312 | "version": "==1.0.2" 313 | }, 314 | "python-dateutil": { 315 | "hashes": [ 316 | "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86", 317 | "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9" 318 | ], 319 | "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", 320 | "version": "==2.8.2" 321 | }, 322 | "python-dotenv": { 323 | "hashes": [ 324 | "sha256:aae25dc1ebe97c420f50b81fb0e5c949659af713f31fdb63c749ca68748f34b1", 325 | "sha256:f521bc2ac9a8e03c736f62911605c5d83970021e3fa95b37d769e2bbbe9b6172" 326 | ], 327 | "index": "pypi", 328 | "version": "==0.19.0" 329 | }, 330 | "python-editor": { 331 | "hashes": [ 332 | "sha256:1bf6e860a8ad52a14c3ee1252d5dc25b2030618ed80c022598f00176adc8367d", 333 | "sha256:51fda6bcc5ddbbb7063b2af7509e43bd84bfc32a4ff71349ec7847713882327b", 334 | "sha256:5f98b069316ea1c2ed3f67e7f5df6c0d8f10b689964a4a811ff64f0106819ec8" 335 | ], 336 | "version": "==1.0.4" 337 | }, 338 | "pytz": { 339 | "hashes": [ 340 | "sha256:83a4a90894bf38e243cf052c8b58f381bfe9a7a483f6a9cab140bc7f702ac4da", 341 | "sha256:eb10ce3e7736052ed3623d49975ce333bcd712c7bb19a58b9e2089d4057d0798" 342 | ], 343 | "version": "==2021.1" 344 | }, 345 | "requests": { 346 | "hashes": [ 347 | "sha256:6c1246513ecd5ecd4528a0906f910e8f0f9c6b8ec72030dc9fd154dc1a6efd24", 348 | "sha256:b8aa58f8cf793ffd8782d3d8cb19e66ef36f7aba4353eec859e74678b01b07a7" 349 | ], 350 | "index": "pypi", 351 | "version": "==2.26.0" 352 | }, 353 | "six": { 354 | "hashes": [ 355 | "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926", 356 | "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254" 357 | ], 358 | "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", 359 | "version": "==1.16.0" 360 | }, 361 | "sqlalchemy": { 362 | "hashes": [ 363 | "sha256:09dbb4bc01a734ccddbf188deb2a69aede4b3c153a72b6d5c6900be7fb2945b1", 364 | "sha256:12bac5fa1a6ea870bdccb96fe01610641dd44ebe001ed91ef7fcd980e9702db5", 365 | "sha256:1fdae7d980a2fa617d119d0dc13ecb5c23cc63a8b04ffcb5298f2c59d86851e9", 366 | "sha256:26daa429f039e29b1e523bf763bfab17490556b974c77b5ca7acb545b9230e9a", 367 | "sha256:36a089dc604032d41343d86290ce85d4e6886012eea73faa88001260abf5ff81", 368 | "sha256:39b5d36ab71f73c068cdcf70c38075511de73616e6c7fdd112d6268c2704d9f5", 369 | "sha256:4014978de28163cd8027434916a92d0f5bb1a3a38dff5e8bf8bff4d9372a9117", 370 | "sha256:44d23ea797a5e0be71bc5454b9ae99158ea0edc79e2393c6e9a2354de88329c0", 371 | "sha256:488608953385d6c127d2dcbc4b11f8d7f2f30b89f6bd27c01b042253d985cc2f", 372 | "sha256:5102b9face693e8b2db3b2539c7e1a5d9a5b4dc0d79967670626ffd2f710d6e6", 373 | "sha256:5908ea6c652a050d768580d01219c98c071e71910ab8e7b42c02af4010608397", 374 | "sha256:5d856cc50fd26fc8dd04892ed5a5a3d7eeb914fea2c2e484183e2d84c14926e0", 375 | "sha256:68393d3fd31469845b6ba11f5b4209edbea0b58506be0e077aafbf9aa2e21e11", 376 | "sha256:6a16c7c4452293da5143afa3056680db2d187b380b3ef4d470d4e29885720de3", 377 | "sha256:756f5d2f5b92d27450167247fb574b09c4cd192a3f8c2e493b3e518a204ee543", 378 | "sha256:891927a49b2363a4199763a9d436d97b0b42c65922a4ea09025600b81a00d17e", 379 | "sha256:9bfe882d5a1bbde0245dca0bd48da0976bd6634cf2041d2fdf0417c5463e40e5", 380 | "sha256:9fcbb4b4756b250ed19adc5e28c005b8ed56fdb5c21efa24c6822c0575b4964d", 381 | "sha256:a00d9c6d3a8afe1d1681cd8a5266d2f0ed684b0b44bada2ca82403b9e8b25d39", 382 | "sha256:a5e14cb0c0a4ac095395f24575a0e7ab5d1be27f5f9347f1762f21505e3ba9f1", 383 | "sha256:b48148ceedfb55f764562e04c00539bb9ea72bf07820ca15a594a9a049ff6b0e", 384 | "sha256:b7fb937c720847879c7402fe300cfdb2aeff22349fa4ea3651bca4e2d6555939", 385 | "sha256:bc34a007e604091ca3a4a057525efc4cefd2b7fe970f44d20b9cfa109ab1bddb", 386 | "sha256:c9373ef67a127799027091fa53449125351a8c943ddaa97bec4e99271dbb21f4", 387 | "sha256:d09a760b0a045b4d799102ae7965b5491ccf102123f14b2a8cc6c01d1021a2d9", 388 | "sha256:ec1be26cdccd60d180359a527d5980d959a26269a2c7b1b327a1eea0cab37ed8", 389 | "sha256:eedd76f135461cf237534a6dc0d1e0f6bb88a1dc193678fab48a11d223462da5", 390 | "sha256:f028ef6a1d828bc754852a022b2160e036202ac8658a6c7d34875aafd14a9a15", 391 | "sha256:f814d80844969b0d22ea63663da4de5ca1c434cfbae226188901e5d368792c17", 392 | "sha256:fd2102a8f8a659522719ed73865dff3d3cc76eb0833039dc473e0ad3041d04be" 393 | ], 394 | "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5'", 395 | "version": "==1.4.22" 396 | }, 397 | "typing-extensions": { 398 | "hashes": [ 399 | "sha256:0ac0f89795dd19de6b97debb0c6af1c70987fd80a2d62d1958f7e56fcc31b497", 400 | "sha256:50b6f157849174217d0656f99dc82fe932884fb250826c18350e159ec6cdf342", 401 | "sha256:779383f6086d90c99ae41cf0ff39aac8a7937a9283ce0a414e5dd782f4c94a84" 402 | ], 403 | "markers": "python_version < '3.8'", 404 | "version": "==3.10.0.0" 405 | }, 406 | "tzlocal": { 407 | "hashes": [ 408 | "sha256:643c97c5294aedc737780a49d9df30889321cbe1204eac2c2ec6134035a92e44", 409 | "sha256:e2cb6c6b5b604af38597403e9852872d7f534962ae2954c7f35efcb1ccacf4a4" 410 | ], 411 | "version": "==2.1" 412 | }, 413 | "urllib3": { 414 | "hashes": [ 415 | "sha256:39fb8672126159acb139a7718dd10806104dec1e2f0f6c88aab05d17df10c8d4", 416 | "sha256:f57b4c16c62fa2760b7e3d97c35b255512fb6b59a259730f36ba32ce9f8e342f" 417 | ], 418 | "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4'", 419 | "version": "==1.26.6" 420 | }, 421 | "werkzeug": { 422 | "hashes": [ 423 | "sha256:1de1db30d010ff1af14a009224ec49ab2329ad2cde454c8a708130642d579c42", 424 | "sha256:6c1ec500dcdba0baa27600f6a22f6333d8b662d22027ff9f6202e3367413caa8" 425 | ], 426 | "markers": "python_version >= '3.6'", 427 | "version": "==2.0.1" 428 | }, 429 | "werobot": { 430 | "hashes": [ 431 | "sha256:8c3bba57f3f8d8b4866c119b3eb8e2d6c6c0525ff7b96eb207d81a6d350aaae5", 432 | "sha256:b145502b7a9b9f30b2669db04c812bbca3a27a9822ff41e322ee458ea3123b28" 433 | ], 434 | "index": "pypi", 435 | "version": "==1.13.1" 436 | }, 437 | "xmltodict": { 438 | "hashes": [ 439 | "sha256:50d8c638ed7ecb88d90561beedbf720c9b4e851a9fa6c47ebd64e99d166d8a21", 440 | "sha256:8bbcb45cc982f48b2ca8fe7e7827c5d792f217ecf1792626f808bf41c3b86051" 441 | ], 442 | "version": "==0.12.0" 443 | }, 444 | "zipp": { 445 | "hashes": [ 446 | "sha256:957cfda87797e389580cb8b9e3870841ca991e2125350677b2ca83a0e99390a3", 447 | "sha256:f5812b1e007e48cff63449a5e9f4e7ebea716b4111f9c4f9a645f91d579bf0c4" 448 | ], 449 | "markers": "python_version >= '3.6'", 450 | "version": "==3.5.0" 451 | } 452 | }, 453 | "develop": { 454 | "autopep8": { 455 | "hashes": [ 456 | "sha256:276ced7e9e3cb22e5d7c14748384a5cf5d9002257c0ed50c0e075b68011bb6d0", 457 | "sha256:aa213493c30dcdac99537249ee65b24af0b2c29f2e83cd8b3f68760441ed0db9" 458 | ], 459 | "index": "pypi", 460 | "version": "==1.5.7" 461 | }, 462 | "pycodestyle": { 463 | "hashes": [ 464 | "sha256:514f76d918fcc0b55c6680472f0a37970994e07bbb80725808c17089be302068", 465 | "sha256:c389c1d06bf7904078ca03399a4816f974a1d590090fecea0c63ec26ebaf1cef" 466 | ], 467 | "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", 468 | "version": "==2.7.0" 469 | }, 470 | "toml": { 471 | "hashes": [ 472 | "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b", 473 | "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f" 474 | ], 475 | "markers": "python_version >= '2.6' and python_version not in '3.0, 3.1, 3.2, 3.3'", 476 | "version": "==0.10.2" 477 | } 478 | } 479 | } 480 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # werobot 2 | 我的微信公众号后台 3 | 4 | 主要功能(期待更多功能可以直接提建议给公众号): 5 | - vip视频解析 6 | - 电影搜索 7 | - 图片转文字 8 | - 技术文章 9 | 10 | 微信扫码即可体验: 11 | 12 | ![](docs/image/coderbox.jpg) 13 | 14 | # 开发相关 15 | 主要使用python语言开发,使用flask、werobot等开源库。 16 | 17 | 基于我的另一个开源项目[flask-scaffolding](https://github.com/barry-ran/flask-scaffolding)作为脚手架,可以实现flask项目的快速开发与部署。 18 | 19 | 如果希望本地调试该项目,除了参考[flask-scaffolding](https://github.com/barry-ran/flask-scaffolding)配置开发环境并初始化数据库,还需要在app目录提供.env文件: 20 | ``` 21 | # flask 密钥(开发可选) 22 | SECRET_KEY="****" 23 | 24 | # 数据库连接信息(必须) 25 | DEV_DATABASE_URL='****' 26 | 27 | # 微信公众号相关密钥(必须) 28 | WEROBOT_ID='****' 29 | WEROBOT_SECRET='****' 30 | 31 | # 百度ai的图片转文字接口需要(不调试该功能可以不提供) 32 | OCR_APP_ID='****' 33 | OCR_API_KEY='****' 34 | OCR_SECRET_KEY='****' 35 | ``` 36 | -------------------------------------------------------------------------------- /app/__init__.py: -------------------------------------------------------------------------------- 1 | from flask import Flask 2 | from . import routes, models 3 | from .config import config 4 | from .log import Log 5 | 6 | # flask run命令会自动调用create_app函数 https://dormousehole.readthedocs.io/en/latest/cli.html 7 | # 通过flask run启动时,可以通过设置FLASK_APP=app:create_app('development')来指定create_app的参数 8 | def create_app(config_name='default'): 9 | app = Flask(__name__) 10 | app.config.from_object(config[config_name]) 11 | config[config_name].init_app(app) 12 | 13 | Log.init_app(app) 14 | 15 | models.init_app(app) 16 | routes.init_app(app) 17 | 18 | Log.logger().info('create_app:(%s)', config_name) 19 | 20 | return app -------------------------------------------------------------------------------- /app/config.py: -------------------------------------------------------------------------------- 1 | import os 2 | from dotenv import load_dotenv 3 | 4 | basedir = os.path.abspath(os.path.dirname(__file__)) 5 | 6 | dotenv_path = os.path.join(basedir, '.env') 7 | load_dotenv(dotenv_path) 8 | 9 | 10 | class Config: 11 | # falsk的内置config key: https://flask.palletsprojects.com/en/2.0.x/config/ 12 | SECRET_KEY = os.environ.get('SECRET_KEY') or os.urandom(16) 13 | SQLALCHEMY_DATABASE_URI = os.environ.get( 14 | 'DEV_DATABASE_URL') or 'no found DEV_DATABASE_URL on .env' 15 | 16 | SQLALCHEMY_COMMIT_ON_TEARDOWN = True 17 | SQLALCHEMY_TRACK_MODIFICATIONS = True 18 | 19 | PRODUCTION_CONFIG = False 20 | 21 | # 公众号配置 22 | WEROBOT_ID = os.environ.get( 23 | 'WEROBOT_ID') or 'no found WEROBOT_ID on .env' 24 | WEROBOT_SECRET = os.environ.get( 25 | 'WEROBOT_SECRET') or 'no found WEROBOT_SECRET on .env' 26 | 27 | # 阿里ocr api 28 | OCR_APP_ID = os.environ.get( 29 | 'OCR_APP_ID') or 'no found OCR_APP_ID on .env' 30 | OCR_API_KEY = os.environ.get( 31 | 'OCR_API_KEY') or 'no found OCR_API_KEY on .env' 32 | OCR_SECRET_KEY = os.environ.get( 33 | 'OCR_SECRET_KEY') or 'no found OCR_SECRET_KEY on .env' 34 | 35 | # init_app是为了在初始化app时附加一些额外配置用的 36 | 37 | @ classmethod 38 | def init_app(cls, app): 39 | 40 | pass 41 | 42 | 43 | class DevelopmentConfig(Config): 44 | SQLALCHEMY_ECHO = True 45 | PRODUCTION_CONFIG = False 46 | 47 | 48 | class ProductionConfig(Config): 49 | SQLALCHEMY_ECHO = False 50 | PRODUCTION_CONFIG = True 51 | 52 | 53 | config = { 54 | 'development': DevelopmentConfig, 55 | 'production': ProductionConfig, 56 | 'default': DevelopmentConfig 57 | } 58 | -------------------------------------------------------------------------------- /app/log.py: -------------------------------------------------------------------------------- 1 | import os 2 | import logging 3 | from logging import handlers 4 | from werkzeug.exceptions import InternalServerError 5 | 6 | basedir = os.path.abspath(os.path.dirname(__file__)) 7 | 8 | def handle_error(error): 9 | Log.logger().error(error) 10 | return error 11 | 12 | class Log: 13 | LOG_PATH = os.path.join(basedir, 'logs') 14 | LOG_NAME = os.path.join(LOG_PATH, 'log.txt') 15 | LOG_LEVEL = 'INFO' 16 | 17 | current_app = None 18 | 19 | @staticmethod 20 | def init_app(app): 21 | Log.current_app = app 22 | if not os.path.exists(Log.LOG_PATH): 23 | os.makedirs(Log.LOG_PATH) 24 | 25 | # 根据时间重命名log 26 | file_handler = logging.handlers.TimedRotatingFileHandler(Log.LOG_NAME, when='D', interval=1, backupCount=0, encoding='utf-8') 27 | file_handler.suffix = '%Y-%m-%d.log' 28 | # 单独设置handler的日志级别:低于该级别则该handler不处理(一个logger可以有多个handler) 29 | # file_handler用来写入文件 30 | file_handler.setLevel(Log.LOG_LEVEL) 31 | 32 | fmt = '%(asctime)s-%(levelname)s-%(filename)s-%(funcName)s-%(lineno)s: %(message)s' 33 | formatter = logging.Formatter(fmt) 34 | file_handler.setFormatter(formatter) 35 | 36 | # 设置logger的日志级别:大于等于该级别才会交给handler处理 37 | app.logger.setLevel('DEBUG') 38 | app.logger.addHandler(file_handler) 39 | 40 | # DEBUG模式下不会走到handle_error 41 | app.register_error_handler(InternalServerError, handle_error) 42 | 43 | @staticmethod 44 | def logger(): 45 | return Log.current_app.logger -------------------------------------------------------------------------------- /app/main.py: -------------------------------------------------------------------------------- 1 | from flask import Flask, render_template 2 | 3 | # 用flask run命令启动app/main时会自动探测到我们创建的这个app对象来启动服务器 4 | # 更多flask run的知识可以看这里:https://dormousehole.readthedocs.io/en/latest/cli.html 5 | app = Flask(__name__) 6 | 7 | @app.route("/") 8 | def index(): 9 | return render_template('index.html') 10 | 11 | if __name__ == "__main__": 12 | # 仅仅直接python main.py运行的时候才会通过这里启动服务器 13 | app.run(port=5000, debug=True) -------------------------------------------------------------------------------- /app/models/__init__.py: -------------------------------------------------------------------------------- 1 | 2 | from flask_sqlalchemy import SQLAlchemy 3 | from flask_migrate import Migrate 4 | 5 | db = SQLAlchemy() 6 | 7 | 8 | def init_app(app): 9 | migrate = Migrate(app, db) 10 | db.init_app(app) 11 | migrate.init_app(app) 12 | 13 | return db 14 | -------------------------------------------------------------------------------- /app/models/vip_video_parse.py: -------------------------------------------------------------------------------- 1 | from . import db 2 | 3 | 4 | class VipVideoParse(db.Model): 5 | id = db.Column(db.Integer, primary_key=True) 6 | parse_url = db.Column(db.String(1024), index=True, unique=True) 7 | network_quality = db.Column(db.Integer, index=True, default=10) 8 | 9 | @classmethod 10 | def get_rand_parse_url(cls, rand): 11 | db_urls = cls.query.order_by(VipVideoParse.network_quality).limit(rand) 12 | urls = [] 13 | for url in db_urls: 14 | urls.append(url.parse_url) 15 | 16 | return urls 17 | 18 | # 告诉 Python 如何打印这个类的对象。我们将用它来调试 19 | def __repr__(self): 20 | return '' % (self.parse_url) 21 | -------------------------------------------------------------------------------- /app/models/website_navigation.py: -------------------------------------------------------------------------------- 1 | from . import db 2 | 3 | 4 | class WebsiteNavigation(db.Model): 5 | id = db.Column(db.Integer, primary_key=True) 6 | url = db.Column(db.String(1024), index=True, unique=True) 7 | logo_url = db.Column(db.String(1024), index=True) 8 | main_type = db.Column(db.String(50), index=True) 9 | second_type = db.Column(db.String(50), index=True) 10 | search_json = db.Column(db.String(1024), index=True, default='') 11 | network_quality = db.Column(db.Integer, index=True, default=10) 12 | __table_args__ = { 13 | "mysql_charset": "utf8" 14 | } 15 | 16 | @classmethod 17 | def get_rand_online_movie_website(cls, rand): 18 | websites = cls.query.filter(WebsiteNavigation.second_type == '在线电影').order_by( 19 | WebsiteNavigation.network_quality).limit(rand) 20 | urls = [] 21 | for website in websites: 22 | urls.append(website) 23 | return urls 24 | -------------------------------------------------------------------------------- /app/routes/__init__.py: -------------------------------------------------------------------------------- 1 | from .home import home_bp 2 | from .log_list import log_list_bp 3 | 4 | from flask_executor import Executor 5 | executor = Executor() 6 | 7 | 8 | def init_app(app): 9 | executor.init_app(app) 10 | 11 | app.register_blueprint(home_bp) 12 | app.register_blueprint(log_list_bp) 13 | 14 | with app.app_context(): 15 | from werobot.contrib.flask import make_view 16 | from .robot import my_werobot 17 | app.add_url_rule(rule='/robot/', # WeRoBot 挂载地址 18 | endpoint='werobot', # Flask 的 endpoint 19 | view_func=make_view(my_werobot), 20 | methods=['GET', 'POST']) 21 | -------------------------------------------------------------------------------- /app/routes/home.py: -------------------------------------------------------------------------------- 1 | import random 2 | from flask import Blueprint, render_template 3 | from ..log import Log 4 | from ..models.vip_video_parse import VipVideoParse 5 | 6 | home_bp = Blueprint('home_bp', __name__) 7 | 8 | 9 | @home_bp.route('/', methods=['GET', 'POST']) 10 | def index(): 11 | default_url_name = ['http://vip.jlsprh.com/?url=', '默认'] 12 | urls_name = get_urls_name() 13 | if len(urls_name) != 0: 14 | default_url_name = urls_name[random.randint(0, len(urls_name) - 1)] 15 | 16 | return render_template('index.html', 17 | default_parse_url=default_url_name[0], 18 | video_url='http://v.youku.com/v_show/id_XMjcxMjY5NjM1Ng==.html', 19 | url_list=urls_name) 20 | 21 | 22 | def get_urls_name(): 23 | urls = VipVideoParse.get_rand_parse_url(10) 24 | urls_name = [] 25 | count = 0 26 | for url in urls: 27 | urls_name.append([url, '推荐接口{}'.format(count)]) 28 | count = count + 1 29 | Log.logger().info('get_urls_name:' + url) 30 | 31 | return urls_name 32 | -------------------------------------------------------------------------------- /app/routes/log_list.py: -------------------------------------------------------------------------------- 1 | import os 2 | from flask import Blueprint, render_template, send_from_directory, current_app 3 | from ..log import Log 4 | 5 | # https://blog.csdn.net/kaever/article/details/116312794?spm=1001.2014.3001.5501 6 | 7 | log_list_bp = Blueprint('log_list_bp', __name__) 8 | 9 | basedir = os.path.abspath(os.path.dirname(__file__)) 10 | logs_dir = os.path.join(basedir, '../logs') 11 | 12 | 13 | @log_list_bp.route('/logs', methods=['GET', 'POST']) 14 | def log_list(): 15 | if current_app.config['PRODUCTION_CONFIG']: 16 | return 'no log on production' 17 | 18 | names = os.listdir(logs_dir) 19 | files = {} 20 | for name in names: 21 | path = os.path.join('/logs', name) 22 | Log.logger().info("list log file:%s", path) 23 | files.update({name: path}) 24 | 25 | return render_template('log_list.html', files=files) 26 | 27 | 28 | @log_list_bp.route('/logs/') 29 | def download(filename): 30 | Log.logger().info("send file:%s:%s", logs_dir, filename) 31 | return send_from_directory(logs_dir, filename) 32 | -------------------------------------------------------------------------------- /app/routes/robot.py: -------------------------------------------------------------------------------- 1 | from werobot import WeRoBot 2 | import werobot 3 | from flask import current_app 4 | from .robot_handler.text_handler import TextHandler 5 | from .robot_handler.image_handler import ImageHandler 6 | from ..log import Log 7 | 8 | my_werobot = WeRoBot(token='werobot') 9 | my_werobot.config["APP_ID"] = current_app.config['WEROBOT_ID'] 10 | my_werobot.config["APP_SECRET"] = current_app.config['WEROBOT_SECRET'] 11 | werobot.logger.enable_pretty_logging(Log.logger(), level='info') 12 | 13 | TextHandler.init_handler(my_werobot) 14 | ImageHandler.init_handler(my_werobot) 15 | 16 | # 自定义菜单 17 | my_werobot.client.create_menu({ 18 | "button": [ 19 | { 20 | "name": "影音娱乐", 21 | "sub_button": [ 22 | { 23 | "type": "click", 24 | "name": "搜电影", 25 | "key": "CLICK_MENU_SEARCH_MOVIE" 26 | }, 27 | { 28 | "type": "view", 29 | "name": "影视VIP接口", 30 | "url": "http://129.211.167.170/" 31 | } 32 | ] 33 | }, 34 | { 35 | "name": "实用工具", 36 | "sub_button": [ 37 | { 38 | "type": "click", 39 | "name": "图片转文字", 40 | "key": "CLICK_MENU_OCR" 41 | }, 42 | ] 43 | }, 44 | { 45 | "name": "技术博客", 46 | "sub_button": [ 47 | { 48 | "type": "view", 49 | "name": "我的开源作品", 50 | "url": "https://github.com/barry-ran" 51 | }, 52 | ] 53 | }, 54 | ]}) 55 | 56 | 57 | @my_werobot.key_click("CLICK_MENU_OCR") 58 | def menu_ocr(message, session): 59 | ImageHandler.set_image_handler_to_ocr(session) 60 | ImageHandler.update_ocr_timeout(session) 61 | my_werobot.client.send_text_message(message.source, '进入图片转文字模式,可以直接发图片给我') 62 | 63 | 64 | @my_werobot.key_click("CLICK_MENU_SEARCH_MOVIE") 65 | def menu_ocr(message, session): 66 | TextHandler.set_text_handler_to_search_movie(session) 67 | my_werobot.client.send_text_message( 68 | message.source, '欢迎使用电影搜索功能,可以直接发电影名给我') 69 | 70 | 71 | @my_werobot.text 72 | def text_handler(message, session): 73 | return TextHandler.text_handler(message, session) 74 | 75 | 76 | @my_werobot.image 77 | def image_handler(message, session): 78 | return ImageHandler.image_handler(message, session) 79 | -------------------------------------------------------------------------------- /app/routes/robot_handler/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/routes/robot_handler/__init__.py -------------------------------------------------------------------------------- /app/routes/robot_handler/image_handler.py: -------------------------------------------------------------------------------- 1 | from enum import IntEnum 2 | from datetime import datetime, timedelta 3 | from .. import executor 4 | from ...third_party.ocr import ocr_aip_url 5 | from ...log import Log 6 | 7 | 8 | class ImageHandlerState(IntEnum): 9 | NULL = 1 10 | OCR = 2 11 | 12 | 13 | class ImageHandler(object): 14 | NULL_IMAGE_STATE_TIP = '请在菜单中选择要执行的操作' 15 | 16 | my_werobot = None 17 | 18 | @staticmethod 19 | def init_handler(my_werobot): 20 | ImageHandler.my_werobot = my_werobot 21 | 22 | @staticmethod 23 | def image_handler(message, session): 24 | if not 'image_handler_state' in session: 25 | return ImageHandler.NULL_IMAGE_STATE_TIP 26 | 27 | if session['image_handler_state'] == ImageHandlerState.OCR: 28 | return ImageHandler.ocr_image_handler(message, session) 29 | 30 | return ImageHandler.NULL_IMAGE_STATE_TIP 31 | 32 | @staticmethod 33 | def set_image_handler_to_ocr(session): 34 | ImageHandler.set_image_handler_state(session, ImageHandlerState.OCR) 35 | 36 | @staticmethod 37 | def set_image_handler_state(session, state): 38 | session['image_handler_state'] = state 39 | 40 | @staticmethod 41 | def update_ocr_timeout(session): 42 | session['image_handler_timeout'] = ( 43 | datetime.now() + timedelta(minutes=5)).timestamp() 44 | 45 | @staticmethod 46 | def ocr_image_handler(message, session): 47 | # 已经超时 48 | if not 'image_handler_timeout' in session: 49 | return ImageHandler.NULL_IMAGE_STATE_TIP 50 | 51 | # 判断是否超时 52 | timestamp = session['image_handler_timeout'] 53 | timeout = datetime.fromtimestamp(timestamp) 54 | now_time = datetime.now() 55 | if now_time > timeout: 56 | del session['image_handler_timeout'] 57 | return ImageHandler.NULL_IMAGE_STATE_TIP 58 | 59 | # 没有超时,则更新时间 60 | ImageHandler.update_ocr_timeout(session) 61 | 62 | # 交由线程去执行耗时任务 63 | executor.submit(ImageHandler.request_ocr, message, session) 64 | return '正在识别...' 65 | 66 | @staticmethod 67 | def request_ocr(message, session): 68 | source = message.source 69 | 70 | Log.logger().info('start ocr') 71 | text = ocr_aip_url(message.img) 72 | # os.remove(img_path) 73 | Log.logger().info('end ocr') 74 | 75 | # 微信公众号限制回复最多2048字节:len(text.encode()) 76 | if len(text.encode()) <= 2048: 77 | ImageHandler.my_werobot.client.send_text_message(source, text) 78 | return 79 | 80 | Log.logger().info('send long text') 81 | # 大于2048字节则按照字符裁剪:utf8一般2-4个字节表示一个字符,这里按照三个字节一个字符估算 82 | text_len = len(text) 83 | cut_limit = int(2048/3 - 10) 84 | pos = 0 85 | while True: 86 | if pos + cut_limit < text_len: 87 | ImageHandler.my_werobot.client.send_text_message( 88 | source, text[pos:pos+cut_limit]) 89 | pos = pos + cut_limit 90 | else: 91 | ImageHandler.my_werobot.client.send_text_message( 92 | source, text[pos:]) 93 | break 94 | 95 | 96 | if __name__ == "__main__": 97 | print('hello') 98 | -------------------------------------------------------------------------------- /app/routes/robot_handler/text_handler.py: -------------------------------------------------------------------------------- 1 | 2 | from enum import IntEnum 3 | from flask import current_app 4 | from .. import executor 5 | from ...log import Log 6 | from ...third_party.movie import Movie 7 | 8 | 9 | class TextHandlerState(IntEnum): 10 | NULL = 1 11 | SEARCH_MOVIE = 2 12 | 13 | 14 | class TextHandler(object): 15 | NULL_TEXT_STATE_TIP = '请在菜单中选择要执行的操作' 16 | 17 | my_werobot = None 18 | 19 | @staticmethod 20 | def init_handler(my_werobot): 21 | TextHandler.my_werobot = my_werobot 22 | 23 | @staticmethod 24 | def text_handler(message, session): 25 | Log.logger().info('text handler:{}'.format(message.content)) 26 | 27 | if not 'text_handler_state' in session: 28 | return TextHandler.NULL_TEXT_STATE_TIP 29 | 30 | try: 31 | if session['text_handler_state'] == TextHandlerState.SEARCH_MOVIE: 32 | return TextHandler.search_movie_handler(message, session) 33 | except Exception as e: 34 | Log.logger().exception(e) 35 | return str(e) 36 | 37 | return TextHandler.NULL_TEXT_STATE_TIP 38 | 39 | @staticmethod 40 | def set_text_handler_to_search_movie(session): 41 | TextHandler.set_text_handler_state( 42 | session, TextHandlerState.SEARCH_MOVIE) 43 | 44 | @staticmethod 45 | def set_text_handler_state(session, state): 46 | session['text_handler_state'] = state 47 | 48 | @staticmethod 49 | def search_movie_handler(message, session): 50 | TextHandler.set_text_handler_state(session, TextHandlerState.NULL) 51 | # 交由线程去执行耗时任务 52 | executor.submit(TextHandler.search_movie, message, session) 53 | return '正在搜索...' 54 | 55 | @staticmethod 56 | def search_movie(message, session): 57 | Log.logger().info('begin search movie:{}'.format(message.content)) 58 | movies = Movie.search(message.content) 59 | Log.logger().info('end search movie:{}'.format(message.content)) 60 | text = "以下结果来源网络,如有侵权,请联系公众号删除(建议复制到浏览器打开):\n" 61 | 62 | for movie in movies: 63 | text += movie.title + ':' + movie.url + '\n' 64 | # 微信公众号限制回复最多2048字节:len(text.encode()) 65 | if len(text.encode()) >= 2048: 66 | break 67 | Log.logger().info('send result:{}'.format(text)) 68 | TextHandler.my_werobot.client.send_text_message(message.source, text) 69 | -------------------------------------------------------------------------------- /app/static/css/style.css: -------------------------------------------------------------------------------- 1 | /************************************************** 2 | * 北漂鱼Vip细节新版样式文件 3 | * 编写:Beipy(http://www.beipy.com) 4 | * 时间:2017年7月16日 5 | * 前端样式为北漂鱼原创设计UI图样设计 6 | * 所有代码均独立编写,谢绝一切形式的盗站行为!! 7 | *************************************************/ 8 | 9 | 10 | /* 畅言去广告(支持评论区跟右下角浮窗广告) */ 11 | #MZAD_POP_PLACEHOLDER,#pop_ad{margin-top:-250px!important;transform:scale(0)} 12 | #feedAv{transform:scale(0);margin-top:-150px!important} 13 | .module-cmt-box{padding:10px 0!important;margin-top:-40px!important} 14 | .header-login{left:90px!important;border:0!important;border-radius:0!important;margin-top:81pt!important;border-radius:21px!important} 15 | .post-wrap-border-l,.post-wrap-border-r,.post-wrap-border-t-l,.post-wrap-border-t-r{display:none} 16 | .post-wrap-main{border:0!important} 17 | .post-wrap-w{background:#f0f0f0;border-radius:6px;} 18 | .btn-fw{background:#0093ff url(../img/release.svg) center no-repeat!important;width:80px!important;height:34px!important;border-radius:15px;margin-top:10px!important;margin-right:10px!important;background-size:30px!important;box-shadow:0 2px 6px rgba(0,0,0,.2);-webkit-transition:.3s;transition:.3s} 19 | .btn-fw:hover{box-shadow:0 6px 10px rgba(0,0,0,.2)} 20 | .block-head-w{margin-top:-20px!important} 21 | .section-service-w{height:0;opacity:0} 22 | .head-img-w{margin:0!important} 23 | .head-img-w img{width:25px!important;height:25px!important} 24 | .head-img-w img:hover{transition:all .3s;transform:scale(3)} 25 | .head-img-w{top:138px!important;left:95px!important} 26 | .wrap-action-gw{border-bottom:1px solid #dee4e9!important;padding-top:30px!important;padding-bottom:1pc!important} 27 | .cmt-list-number,.title-name-gw-tag,.type-lists,.wrap-name-w{display:none!important} 28 | .cmt-list-type{margin:0!important} 29 | .build-floor-gw{background:#f0f0f0!important} 30 | .block-cont-gw{padding:20px 0!important;border:0!important} 31 | .section-list-w{width:95%!important;margin-left:2%!important} 32 | .head-img-gw img:hover{transition:all .3s;transform:scale(.9)} 33 | 34 | 35 | * { 36 | margin: 0; 37 | padding: 0; 38 | font-family: Arial, "Microsoft Yahei", sans-serif, "宋体"; 39 | } 40 | a{ 41 | text-decoration: none; 42 | } 43 | 44 | /*body滚动条开始*/ 45 | 46 | body::-webkit-scrollbar { 47 | width: 5px; 48 | height: 5px; 49 | } 50 | 51 | body::-webkit-scrollbar-button { 52 | width: 0; 53 | height: 0; 54 | background-color: royalblue; 55 | } 56 | 57 | body::-webkit-scrollbar-track { 58 | background: #ccc; 59 | border-radius: 5px; 60 | } 61 | 62 | body::-webkit-scrollbar-thumb { 63 | background: #0093ff; 64 | border-radius: 5px; 65 | } 66 | 67 | body::-webkit-scrollbar-corner { 68 | background: #82AFFF; 69 | } 70 | 71 | body::-webkit-scrollbar-resizer { 72 | background: #FF0BEE; 73 | } 74 | 75 | .btnColor-jb {} 76 | /*body滚动结束*/ 77 | 78 | header { 79 | background: #333333; 80 | } 81 | 82 | header a { 83 | color: #fff; 84 | } 85 | 86 | header .container { 87 | padding: 0; 88 | } 89 | 90 | header nav.navbar { 91 | margin-bottom: 0; 92 | } 93 | 94 | header nav.navbar-default { 95 | background-color: transparent; 96 | border: 0; 97 | } 98 | 99 | header .navbar-brand { 100 | padding: 0px 0px; 101 | margin: 4px 10px; 102 | } 103 | 104 | .icon-bar { 105 | background-color: #fff !important; 106 | } 107 | 108 | header .navbar-toggle { 109 | border: none; 110 | } 111 | 112 | .navbar-default .navbar-toggle:hover, 113 | .navbar-default .navbar-toggle:focus { 114 | background-color: #333333; 115 | } 116 | 117 | .navbar-default .navbar-nav>li>a:hover, 118 | .navbar-default .navbar-nav>li>a:focus { 119 | color: #fff; 120 | background-color: transparent; 121 | } 122 | .logo { 123 | display: inline-block; 124 | width: 150px; 125 | height: 42px; 126 | float: left; 127 | background-position: -183px 0; 128 | background-image: url(../img/jingling.png); 129 | background-repeat: no-repeat; 130 | } 131 | /*video-box样式*/ 132 | .video-box { 133 | background-image: url(../img/mbj.jpg); 134 | background-repeat: no-repeat; 135 | background-size: cover; 136 | width: 100%; 137 | margin-top: 50px; 138 | padding-right: 8px; 139 | padding-left: 8px; 140 | position: relative; 141 | } 142 | 143 | .video-box>.container { 144 | padding-right: 0px; 145 | padding-left: 0px; 146 | } 147 | 148 | #palybox { 149 | width: 100%; 150 | height: 224px; 151 | background: url(../img/palyboxBg.png); 152 | background-repeat: no-repeat; 153 | background-size: cover; 154 | } 155 | 156 | .tit-name { 157 | text-align: center; 158 | line-height: 35px; 159 | margin-top: 20px; 160 | vertical-align: middle; 161 | color: #3376f3; 162 | background: #0f1856; 163 | vertical-align: middle; 164 | text-overflow: ellipsis; 165 | white-space: nowrap; 166 | word-wrap: normal; 167 | } 168 | 169 | .tit-name h1 { 170 | display: inline-block; 171 | font-size: 16px; 172 | width: 70%; 173 | overflow: hidden; 174 | margin-top: 0px; 175 | margin-bottom: 0px; 176 | vertical-align: middle; 177 | text-overflow: ellipsis; 178 | white-space: nowrap; 179 | word-wrap: normal; 180 | } 181 | 182 | 183 | /*输入input样式*/ 184 | 185 | .url-text { 186 | height: 35px; 187 | border: 1px solid #060c37; 188 | background: #060c37; 189 | color: #3376f3; 190 | } 191 | 192 | .url-box { 193 | background: #0f1856; 194 | padding: 10px 5px; 195 | margin-top: -6px; 196 | } 197 | 198 | div.url-box .input-group-addon { 199 | padding: 0; 200 | border: none; 201 | } 202 | 203 | .url-text:focus { 204 | border-color: #66afe9; 205 | outline: 0; 206 | -webkit-box-shadow: inset 0 1px 1px rgba(0, 0, 0, .075), 0 0 8px rgba(102, 175, 233, .6); 207 | box-shadow: inset 0 1px 1px rgba(0, 0, 0, .075), 0 0 8px rgba(102, 175, 233, .6); 208 | } 209 | 210 | .btn-play { 211 | height: 35px; 212 | padding: 0 10px; 213 | color: #fff; 214 | border: none; 215 | background: -webkit-linear-gradient(left, #19c3ff, #0093ff); 216 | /* Safari 5.1 - 6.0 */ 217 | background: -o-linear-gradient(left, #19c3ff, #0093ff); 218 | /* Opera 11.1 - 12.0 */ 219 | background: -moz-linear-gradient(left, #19c3ff, #0093ff); 220 | /* Firefox 3.6 - 15 */ 221 | background: linear-gradient(left, #19c3ff, #0093ff); 222 | /* 标准的语法 */ 223 | } 224 | 225 | div.url-box .input-group-btn:last-child>.btn:not(:last-child):not(.dropdown-toggle) { 226 | border-top-right-radius: 4px; 227 | border-bottom-right-radius: 4px; 228 | } 229 | 230 | div.url-box .btn-play:focus { 231 | outline-style: none; 232 | color: #FFFFFF; 233 | } 234 | 235 | div.url-box .btn-play:active { 236 | color: #0093ff; 237 | background: -webkit-linear-gradient(left, #ffb199, #fa709a); 238 | /* Safari 5.1 - 6.0 */ 239 | background: -o-linear-gradient(left, #ffb199, #fa709a); 240 | /* Opera 11.1 - 12.0 */ 241 | background: -moz-linear-gradient(left, #ffb199, #fa709a); 242 | /* Firefox 3.6 - 15 */ 243 | background: linear-gradient(left, #ffb199, #fa709a); 244 | /* 标准的语法 */ 245 | } 246 | 247 | div.url-box .btn-play:hover { 248 | color: #FFFFFF; 249 | } 250 | 251 | .tit-gg { 252 | text-align: center; 253 | line-height: 35px; 254 | vertical-align: middle; 255 | color: #3376f3; 256 | background: #0f1856; 257 | vertical-align: middle; 258 | text-overflow: ellipsis; 259 | white-space: nowrap; 260 | word-wrap: normal; 261 | } 262 | .guanggaoBox{ 263 | margin: 40px 0 40px 0; 264 | } 265 | .guanggaoBox img{ 266 | width: 100%; 267 | } 268 | /*logo平台*/ 269 | .title-tit { 270 | font-family: "微软雅黑"; 271 | padding: 20px 0 10px; 272 | } 273 | .title-tit h4 { 274 | display: inline-block; 275 | border-left: 5px solid #0093FF; 276 | padding-left: 10px; 277 | font-size: 16px; 278 | line-height: 1rem; 279 | font-weight: bold; 280 | } 281 | .title-tit p { 282 | display: inline-block; 283 | font-size: 12px; 284 | color: #555555; 285 | } 286 | .logo-lie { 287 | background: #fff; 288 | text-align: center; 289 | padding: 4px 10px; 290 | border-radius: 4px; 291 | border: 1px solid #EEEEEE; 292 | -webkit-transition: border linear .2s, box-shadow linear .2s; 293 | -moz-transition: border linear .2s, box-shadow linear .2s; 294 | -o-transition: border linear .2s, box-shadow linear .2s; 295 | transition: border linear .2s, box-shadow linear .2s; 296 | } 297 | 298 | .logo-lie a { 299 | text-align: center; 300 | } 301 | 302 | .logo-box>.container>.row>.col-xs-4 { 303 | padding-right: 5px; 304 | padding-left: 5px; 305 | margin-bottom: 5px; 306 | } 307 | 308 | .logo-box>.container>.row>.col-lg-12 { 309 | padding-right: 0px; 310 | padding-left: 0px; 311 | } 312 | /*评论区*/ 313 | .cy-box .container{ 314 | padding-right: 0px; 315 | padding-left: 0px; 316 | } 317 | 318 | /*页尾*/ 319 | .footer{ 320 | margin-top: 30px; 321 | background:#26282c; 322 | } 323 | .branding_link{ 324 | display: none; 325 | text-align: center; 326 | } 327 | .footer-tit{ 328 | text-align: center; 329 | color: #888; 330 | } 331 | .toot-tto{ 332 | margin-bottom:3em; 333 | } 334 | .foot-tt{ 335 | color: #777777; 336 | } 337 | .counter{ 338 | margin: 15px 0; 339 | color: rgba(255, 255, 255, 0.7); 340 | } 341 | /*侧滑样式*/ 342 | .ch1{ 343 | display: none; 344 | } 345 | /*屏幕最小768px匹配样式*/ 346 | 347 | @media (min-width: 768px) { 348 | a{ 349 | text-decoration: none; 350 | } 351 | a:hover{ 352 | text-decoration: none; 353 | } 354 | 355 | /*头部导航样式*/ 356 | header { 357 | background: #fff; 358 | background: #FFFFFF; 359 | width: 100%; 360 | height: 70px; 361 | box-shadow: 1px 1px 1px rgba(0, 0, 0, 0.27); 362 | -webkit-box-shadow: 1px 1px 1px rgba(0, 0, 0, 0.27); 363 | } 364 | header a { 365 | color: #333; 366 | } 367 | header .navbar>.container-fluid .navbar-brand { 368 | margin-left: 0px; 369 | } 370 | header .container-fluid { 371 | padding-right: 0px; 372 | padding-left: 0px; 373 | } 374 | header .navbar-brand { 375 | margin: 14px 0px; 376 | margin-left: 0px; 377 | line-height: 40px; 378 | } 379 | .video-box{ 380 | margin-top: 70px; 381 | } 382 | .nav .open>a, 383 | .nav .open>a:hover, 384 | .nav .open>a:focus { 385 | border-color: #0093ff; 386 | } 387 | .logo { 388 | background-position: 0px 0; 389 | } 390 | header .navbar-nav>li>a { 391 | line-height: 40px; 392 | color: #333 !important; 393 | padding-bottom: 11px; 394 | border-bottom: 4px solid #fff; 395 | transition: all 0.5s; 396 | -webkit-transition: all 0.5s; 397 | } 398 | header .navbar-nav>li>a:hover { 399 | color: #0093ff !important; 400 | border-bottom: 4px solid #0093ff; 401 | } 402 | header .active { 403 | background-color: transparent; 404 | } 405 | header .navbar-nav>li>a:focus { 406 | background-color: transparent !important; 407 | } 408 | header ul.dropdown-menu { 409 | border: 1px solid #0093FF; 410 | box-shadow: 0 3px 6px rgba(0, 0, 0, .175); 411 | } 412 | .weibo-box { 413 | width: 300px; 414 | height: 530px; 415 | } 416 | .weibo-tit { 417 | text-align: center; 418 | } 419 | /*video-box样式*/ 420 | .video-box { 421 | background: url(../img/video.jpg); 422 | background-repeat: no-repeat; 423 | background-size: cover; 424 | width: 100%; 425 | } 426 | /*palybox视频窗口*/ 427 | #palybox { 428 | height: 659px; 429 | } 430 | .tit-name { 431 | text-align: left; 432 | margin-top: 30px; 433 | padding: 10px 20px; 434 | color: #fff; 435 | /*background: #fff;*/ 436 | /*color: #333;*/ 437 | border-radius: 5px 5px 0 0; 438 | background: rgba(0,0,0,.49); 439 | } 440 | .tit-name h1{ 441 | font-size: 1.2em; 442 | } 443 | 444 | .tit-name span { 445 | color: #b2b2b2; 446 | font-size: 1em; 447 | vertical-align: middle; 448 | } 449 | .url-box { 450 | background: #fff; 451 | padding: 15px 15px; 452 | } 453 | .url-text { 454 | height: 40px; 455 | background: #f2f3f5; 456 | border: 1px solid #f2f3f5; 457 | border-right: 0px; 458 | border-radius: 20px 0 0 20px; 459 | } 460 | .btn-play { 461 | height: 40px; 462 | color: #fff; 463 | padding: 0 40px; 464 | border-radius: 20px; 465 | } 466 | div.url-box .btn-play:hover { 467 | color: #FFFFFF; 468 | background: -webkit-linear-gradient(left, #ffb199, #fa709a); 469 | /* Safari 5.1 - 6.0 */ 470 | background: -o-linear-gradient(left, #ffb199, #fa709a); 471 | /* Opera 11.1 - 12.0 */ 472 | background: -moz-linear-gradient(left, #ffb199, #fa709a); 473 | /* Firefox 3.6 - 15 */ 474 | background: linear-gradient(left, #ffb199, #fa709a); 475 | /* 标准的语法 */ 476 | } 477 | .url-c { 478 | border-radius: 0px 0 0 0px; 479 | padding: 0 15px; 480 | box-shadow: 0px 1px 2px rgba(153, 153, 153, 0.32) inset; 481 | } 482 | div.url-box .input-group-btn:last-child>.btn:not(:last-child):not(.dropdown-toggle) { 483 | border-top-right-radius: 20px; 484 | border-bottom-right-radius: 20px; 485 | } 486 | .tit-gg { 487 | text-align: left; 488 | border-top: 1px solid #F5F5F5; 489 | padding: 10px 20px; 490 | background: #fff; 491 | color: #999; 492 | } 493 | /*logo平台*/ 494 | .logo-box>.container { 495 | padding-right: 0px; 496 | padding-left: 0px; 497 | } 498 | .logo-box>.container>.row { 499 | margin-right: 0px; 500 | margin-left: 0px; 501 | } 502 | .title-tit { 503 | font-family: "微软雅黑"; 504 | padding: 50px 0 10px; 505 | margin-bottom: 10px; 506 | } 507 | .title-tit h4 { 508 | display: inline-block; 509 | border-left: 5px solid #0093FF; 510 | padding-left: 10px; 511 | font-size: 16px; 512 | font-weight: bold; 513 | } 514 | .title-tit p { 515 | padding-left: 20px; 516 | display: inline-block; 517 | font-size: 12px; 518 | color: #555555; 519 | } 520 | .logo-box>.col-sm-2 { 521 | padding-right: 0px; 522 | padding-left: 0px; 523 | } 524 | .logo-box>.container>.row>.col-xs-4 { 525 | margin-bottom: 15px; 526 | } 527 | .logo-lie { 528 | padding: 10px 15px; 529 | } 530 | .logo-lie:hover { 531 | border: 1px solid #0093FF; 532 | box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px rgba(0, 147, 255, .6); 533 | } 534 | /*页尾*/ 535 | .footer{ 536 | background:#333; 537 | } 538 | .footer a:hover{ 539 | color: #0093ff; 540 | } 541 | .branding_link{ 542 | display: block; 543 | text-align: center; 544 | padding: 10px 0; 545 | } 546 | /*侧滑*/ 547 | .ch1 { 548 | display: block; 549 | position: fixed; 550 | width: 50px; 551 | z-index: 999; 552 | left: 10%; 553 | bottom: 30%; 554 | } 555 | 556 | .ch1 a { 557 | display: inline-block; 558 | } 559 | 560 | 561 | .ch { 562 | position: fixed; 563 | width: 50px; 564 | z-index: 999; 565 | right: 15px; 566 | bottom: 15px; 567 | } 568 | .rollbar-weibo { 569 | background: url(../img/ce.png) no-repeat scroll -156px 0px; 570 | } 571 | 572 | .rollbar-weibo, 573 | .rollbar-item, 574 | .rollbar-weix { 575 | text-align: center; 576 | height: 50px; 577 | border-radius: 0px; 578 | background-color: #666; 579 | color: #fff; 580 | opacity: 0.5; 581 | cursor: pointer; 582 | -webkit-transition: all .3s ease-in-out; 583 | -moz-transition: all .3s ease-in-out; 584 | transition: all .3s ease-in-out; 585 | } 586 | 587 | .rollbar-weibo a, 588 | .rollbar-item a { 589 | width: 50px; 590 | height: 50px; 591 | display: inline-block; 592 | } 593 | 594 | .rollbar-item { 595 | background: url(../img/ce.png) no-repeat scroll -208px 0px; 596 | } 597 | 598 | .rollbar-item:hover, 599 | .rollbar-weibo:hover, 600 | .rollbar-weix:hover { 601 | color: #fff; 602 | opacity: 0.9; 603 | } 604 | 605 | .rollbar-weix { 606 | background: url(../img/ce.png) no-repeat scroll -104px 0px; 607 | } 608 | 609 | .rollbar-weix:hover .weixinx { 610 | display: block; 611 | } 612 | 613 | .weixinx { 614 | display: none; 615 | width: 164px; 616 | height: 206px; 617 | position: absolute; 618 | left: -164px; 619 | bottom: 0; 620 | -webkit-transition: all .3s ease-in-out; 621 | -moz-transition: all .3s ease-in-out; 622 | transition: all .3s ease-in-out; 623 | } 624 | 625 | .weixinx { 626 | background: url(../img/ce.png) no-repeat scroll 0px -52px; 627 | } 628 | } 629 | -------------------------------------------------------------------------------- /app/static/img/56logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/56logo.png -------------------------------------------------------------------------------- /app/static/img/acfun.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/acfun.png -------------------------------------------------------------------------------- /app/static/img/beian.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/beian.png -------------------------------------------------------------------------------- /app/static/img/bilibili.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/bilibili.png -------------------------------------------------------------------------------- /app/static/img/favicon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/favicon.png -------------------------------------------------------------------------------- /app/static/img/fengxing.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/fengxing.gif -------------------------------------------------------------------------------- /app/static/img/hunantvlogo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/hunantvlogo.png -------------------------------------------------------------------------------- /app/static/img/iqiyi.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/iqiyi.png -------------------------------------------------------------------------------- /app/static/img/letvlogo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/letvlogo.png -------------------------------------------------------------------------------- /app/static/img/logo-collapsed@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/logo-collapsed@2x.png -------------------------------------------------------------------------------- /app/static/img/logo@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/logo@2x.png -------------------------------------------------------------------------------- /app/static/img/logo_dark@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/logo_dark@2x.png -------------------------------------------------------------------------------- /app/static/img/qqlogo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/qqlogo.png -------------------------------------------------------------------------------- /app/static/img/sohulogo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/sohulogo.png -------------------------------------------------------------------------------- /app/static/img/tudoulogo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/tudoulogo.png -------------------------------------------------------------------------------- /app/static/img/wasulogo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/wasulogo.png -------------------------------------------------------------------------------- /app/static/img/webstack_banner_cn.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/webstack_banner_cn.png -------------------------------------------------------------------------------- /app/static/img/webstack_icon_producthunt.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/webstack_icon_producthunt.png -------------------------------------------------------------------------------- /app/static/img/yinyuetailogo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/yinyuetailogo.png -------------------------------------------------------------------------------- /app/static/img/ykcloud.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/ykcloud.png -------------------------------------------------------------------------------- /app/static/img/youkulogo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/static/img/youkulogo.png -------------------------------------------------------------------------------- /app/static/js/bootstrap.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap v3.3.7 (http://getbootstrap.com) 3 | * Copyright 2011-2016 Twitter, Inc. 4 | * Licensed under the MIT license 5 | */ 6 | if("undefined"==typeof jQuery)throw new Error("Bootstrap's JavaScript requires jQuery");+function(a){"use strict";var b=a.fn.jquery.split(" ")[0].split(".");if(b[0]<2&&b[1]<9||1==b[0]&&9==b[1]&&b[2]<1||b[0]>3)throw new Error("Bootstrap's JavaScript requires jQuery version 1.9.1 or higher, but lower than version 4")}(jQuery),+function(a){"use strict";function b(){var a=document.createElement("bootstrap"),b={WebkitTransition:"webkitTransitionEnd",MozTransition:"transitionend",OTransition:"oTransitionEnd otransitionend",transition:"transitionend"};for(var c in b)if(void 0!==a.style[c])return{end:b[c]};return!1}a.fn.emulateTransitionEnd=function(b){var c=!1,d=this;a(this).one("bsTransitionEnd",function(){c=!0});var e=function(){c||a(d).trigger(a.support.transition.end)};return setTimeout(e,b),this},a(function(){a.support.transition=b(),a.support.transition&&(a.event.special.bsTransitionEnd={bindType:a.support.transition.end,delegateType:a.support.transition.end,handle:function(b){if(a(b.target).is(this))return b.handleObj.handler.apply(this,arguments)}})})}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var c=a(this),e=c.data("bs.alert");e||c.data("bs.alert",e=new d(this)),"string"==typeof b&&e[b].call(c)})}var c='[data-dismiss="alert"]',d=function(b){a(b).on("click",c,this.close)};d.VERSION="3.3.7",d.TRANSITION_DURATION=150,d.prototype.close=function(b){function c(){g.detach().trigger("closed.bs.alert").remove()}var e=a(this),f=e.attr("data-target");f||(f=e.attr("href"),f=f&&f.replace(/.*(?=#[^\s]*$)/,""));var g=a("#"===f?[]:f);b&&b.preventDefault(),g.length||(g=e.closest(".alert")),g.trigger(b=a.Event("close.bs.alert")),b.isDefaultPrevented()||(g.removeClass("in"),a.support.transition&&g.hasClass("fade")?g.one("bsTransitionEnd",c).emulateTransitionEnd(d.TRANSITION_DURATION):c())};var e=a.fn.alert;a.fn.alert=b,a.fn.alert.Constructor=d,a.fn.alert.noConflict=function(){return a.fn.alert=e,this},a(document).on("click.bs.alert.data-api",c,d.prototype.close)}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.button"),f="object"==typeof b&&b;e||d.data("bs.button",e=new c(this,f)),"toggle"==b?e.toggle():b&&e.setState(b)})}var c=function(b,d){this.$element=a(b),this.options=a.extend({},c.DEFAULTS,d),this.isLoading=!1};c.VERSION="3.3.7",c.DEFAULTS={loadingText:"loading..."},c.prototype.setState=function(b){var c="disabled",d=this.$element,e=d.is("input")?"val":"html",f=d.data();b+="Text",null==f.resetText&&d.data("resetText",d[e]()),setTimeout(a.proxy(function(){d[e](null==f[b]?this.options[b]:f[b]),"loadingText"==b?(this.isLoading=!0,d.addClass(c).attr(c,c).prop(c,!0)):this.isLoading&&(this.isLoading=!1,d.removeClass(c).removeAttr(c).prop(c,!1))},this),0)},c.prototype.toggle=function(){var a=!0,b=this.$element.closest('[data-toggle="buttons"]');if(b.length){var c=this.$element.find("input");"radio"==c.prop("type")?(c.prop("checked")&&(a=!1),b.find(".active").removeClass("active"),this.$element.addClass("active")):"checkbox"==c.prop("type")&&(c.prop("checked")!==this.$element.hasClass("active")&&(a=!1),this.$element.toggleClass("active")),c.prop("checked",this.$element.hasClass("active")),a&&c.trigger("change")}else this.$element.attr("aria-pressed",!this.$element.hasClass("active")),this.$element.toggleClass("active")};var d=a.fn.button;a.fn.button=b,a.fn.button.Constructor=c,a.fn.button.noConflict=function(){return a.fn.button=d,this},a(document).on("click.bs.button.data-api",'[data-toggle^="button"]',function(c){var d=a(c.target).closest(".btn");b.call(d,"toggle"),a(c.target).is('input[type="radio"], input[type="checkbox"]')||(c.preventDefault(),d.is("input,button")?d.trigger("focus"):d.find("input:visible,button:visible").first().trigger("focus"))}).on("focus.bs.button.data-api blur.bs.button.data-api",'[data-toggle^="button"]',function(b){a(b.target).closest(".btn").toggleClass("focus",/^focus(in)?$/.test(b.type))})}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.carousel"),f=a.extend({},c.DEFAULTS,d.data(),"object"==typeof b&&b),g="string"==typeof b?b:f.slide;e||d.data("bs.carousel",e=new c(this,f)),"number"==typeof b?e.to(b):g?e[g]():f.interval&&e.pause().cycle()})}var c=function(b,c){this.$element=a(b),this.$indicators=this.$element.find(".carousel-indicators"),this.options=c,this.paused=null,this.sliding=null,this.interval=null,this.$active=null,this.$items=null,this.options.keyboard&&this.$element.on("keydown.bs.carousel",a.proxy(this.keydown,this)),"hover"==this.options.pause&&!("ontouchstart"in document.documentElement)&&this.$element.on("mouseenter.bs.carousel",a.proxy(this.pause,this)).on("mouseleave.bs.carousel",a.proxy(this.cycle,this))};c.VERSION="3.3.7",c.TRANSITION_DURATION=600,c.DEFAULTS={interval:5e3,pause:"hover",wrap:!0,keyboard:!0},c.prototype.keydown=function(a){if(!/input|textarea/i.test(a.target.tagName)){switch(a.which){case 37:this.prev();break;case 39:this.next();break;default:return}a.preventDefault()}},c.prototype.cycle=function(b){return b||(this.paused=!1),this.interval&&clearInterval(this.interval),this.options.interval&&!this.paused&&(this.interval=setInterval(a.proxy(this.next,this),this.options.interval)),this},c.prototype.getItemIndex=function(a){return this.$items=a.parent().children(".item"),this.$items.index(a||this.$active)},c.prototype.getItemForDirection=function(a,b){var c=this.getItemIndex(b),d="prev"==a&&0===c||"next"==a&&c==this.$items.length-1;if(d&&!this.options.wrap)return b;var e="prev"==a?-1:1,f=(c+e)%this.$items.length;return this.$items.eq(f)},c.prototype.to=function(a){var b=this,c=this.getItemIndex(this.$active=this.$element.find(".item.active"));if(!(a>this.$items.length-1||a<0))return this.sliding?this.$element.one("slid.bs.carousel",function(){b.to(a)}):c==a?this.pause().cycle():this.slide(a>c?"next":"prev",this.$items.eq(a))},c.prototype.pause=function(b){return b||(this.paused=!0),this.$element.find(".next, .prev").length&&a.support.transition&&(this.$element.trigger(a.support.transition.end),this.cycle(!0)),this.interval=clearInterval(this.interval),this},c.prototype.next=function(){if(!this.sliding)return this.slide("next")},c.prototype.prev=function(){if(!this.sliding)return this.slide("prev")},c.prototype.slide=function(b,d){var e=this.$element.find(".item.active"),f=d||this.getItemForDirection(b,e),g=this.interval,h="next"==b?"left":"right",i=this;if(f.hasClass("active"))return this.sliding=!1;var j=f[0],k=a.Event("slide.bs.carousel",{relatedTarget:j,direction:h});if(this.$element.trigger(k),!k.isDefaultPrevented()){if(this.sliding=!0,g&&this.pause(),this.$indicators.length){this.$indicators.find(".active").removeClass("active");var l=a(this.$indicators.children()[this.getItemIndex(f)]);l&&l.addClass("active")}var m=a.Event("slid.bs.carousel",{relatedTarget:j,direction:h});return a.support.transition&&this.$element.hasClass("slide")?(f.addClass(b),f[0].offsetWidth,e.addClass(h),f.addClass(h),e.one("bsTransitionEnd",function(){f.removeClass([b,h].join(" ")).addClass("active"),e.removeClass(["active",h].join(" ")),i.sliding=!1,setTimeout(function(){i.$element.trigger(m)},0)}).emulateTransitionEnd(c.TRANSITION_DURATION)):(e.removeClass("active"),f.addClass("active"),this.sliding=!1,this.$element.trigger(m)),g&&this.cycle(),this}};var d=a.fn.carousel;a.fn.carousel=b,a.fn.carousel.Constructor=c,a.fn.carousel.noConflict=function(){return a.fn.carousel=d,this};var e=function(c){var d,e=a(this),f=a(e.attr("data-target")||(d=e.attr("href"))&&d.replace(/.*(?=#[^\s]+$)/,""));if(f.hasClass("carousel")){var g=a.extend({},f.data(),e.data()),h=e.attr("data-slide-to");h&&(g.interval=!1),b.call(f,g),h&&f.data("bs.carousel").to(h),c.preventDefault()}};a(document).on("click.bs.carousel.data-api","[data-slide]",e).on("click.bs.carousel.data-api","[data-slide-to]",e),a(window).on("load",function(){a('[data-ride="carousel"]').each(function(){var c=a(this);b.call(c,c.data())})})}(jQuery),+function(a){"use strict";function b(b){var c,d=b.attr("data-target")||(c=b.attr("href"))&&c.replace(/.*(?=#[^\s]+$)/,"");return a(d)}function c(b){return this.each(function(){var c=a(this),e=c.data("bs.collapse"),f=a.extend({},d.DEFAULTS,c.data(),"object"==typeof b&&b);!e&&f.toggle&&/show|hide/.test(b)&&(f.toggle=!1),e||c.data("bs.collapse",e=new d(this,f)),"string"==typeof b&&e[b]()})}var d=function(b,c){this.$element=a(b),this.options=a.extend({},d.DEFAULTS,c),this.$trigger=a('[data-toggle="collapse"][href="#'+b.id+'"],[data-toggle="collapse"][data-target="#'+b.id+'"]'),this.transitioning=null,this.options.parent?this.$parent=this.getParent():this.addAriaAndCollapsedClass(this.$element,this.$trigger),this.options.toggle&&this.toggle()};d.VERSION="3.3.7",d.TRANSITION_DURATION=350,d.DEFAULTS={toggle:!0},d.prototype.dimension=function(){var a=this.$element.hasClass("width");return a?"width":"height"},d.prototype.show=function(){if(!this.transitioning&&!this.$element.hasClass("in")){var b,e=this.$parent&&this.$parent.children(".panel").children(".in, .collapsing");if(!(e&&e.length&&(b=e.data("bs.collapse"),b&&b.transitioning))){var f=a.Event("show.bs.collapse");if(this.$element.trigger(f),!f.isDefaultPrevented()){e&&e.length&&(c.call(e,"hide"),b||e.data("bs.collapse",null));var g=this.dimension();this.$element.removeClass("collapse").addClass("collapsing")[g](0).attr("aria-expanded",!0),this.$trigger.removeClass("collapsed").attr("aria-expanded",!0),this.transitioning=1;var h=function(){this.$element.removeClass("collapsing").addClass("collapse in")[g](""),this.transitioning=0,this.$element.trigger("shown.bs.collapse")};if(!a.support.transition)return h.call(this);var i=a.camelCase(["scroll",g].join("-"));this.$element.one("bsTransitionEnd",a.proxy(h,this)).emulateTransitionEnd(d.TRANSITION_DURATION)[g](this.$element[0][i])}}}},d.prototype.hide=function(){if(!this.transitioning&&this.$element.hasClass("in")){var b=a.Event("hide.bs.collapse");if(this.$element.trigger(b),!b.isDefaultPrevented()){var c=this.dimension();this.$element[c](this.$element[c]())[0].offsetHeight,this.$element.addClass("collapsing").removeClass("collapse in").attr("aria-expanded",!1),this.$trigger.addClass("collapsed").attr("aria-expanded",!1),this.transitioning=1;var e=function(){this.transitioning=0,this.$element.removeClass("collapsing").addClass("collapse").trigger("hidden.bs.collapse")};return a.support.transition?void this.$element[c](0).one("bsTransitionEnd",a.proxy(e,this)).emulateTransitionEnd(d.TRANSITION_DURATION):e.call(this)}}},d.prototype.toggle=function(){this[this.$element.hasClass("in")?"hide":"show"]()},d.prototype.getParent=function(){return a(this.options.parent).find('[data-toggle="collapse"][data-parent="'+this.options.parent+'"]').each(a.proxy(function(c,d){var e=a(d);this.addAriaAndCollapsedClass(b(e),e)},this)).end()},d.prototype.addAriaAndCollapsedClass=function(a,b){var c=a.hasClass("in");a.attr("aria-expanded",c),b.toggleClass("collapsed",!c).attr("aria-expanded",c)};var e=a.fn.collapse;a.fn.collapse=c,a.fn.collapse.Constructor=d,a.fn.collapse.noConflict=function(){return a.fn.collapse=e,this},a(document).on("click.bs.collapse.data-api",'[data-toggle="collapse"]',function(d){var e=a(this);e.attr("data-target")||d.preventDefault();var f=b(e),g=f.data("bs.collapse"),h=g?"toggle":e.data();c.call(f,h)})}(jQuery),+function(a){"use strict";function b(b){var c=b.attr("data-target");c||(c=b.attr("href"),c=c&&/#[A-Za-z]/.test(c)&&c.replace(/.*(?=#[^\s]*$)/,""));var d=c&&a(c);return d&&d.length?d:b.parent()}function c(c){c&&3===c.which||(a(e).remove(),a(f).each(function(){var d=a(this),e=b(d),f={relatedTarget:this};e.hasClass("open")&&(c&&"click"==c.type&&/input|textarea/i.test(c.target.tagName)&&a.contains(e[0],c.target)||(e.trigger(c=a.Event("hide.bs.dropdown",f)),c.isDefaultPrevented()||(d.attr("aria-expanded","false"),e.removeClass("open").trigger(a.Event("hidden.bs.dropdown",f)))))}))}function d(b){return this.each(function(){var c=a(this),d=c.data("bs.dropdown");d||c.data("bs.dropdown",d=new g(this)),"string"==typeof b&&d[b].call(c)})}var e=".dropdown-backdrop",f='[data-toggle="dropdown"]',g=function(b){a(b).on("click.bs.dropdown",this.toggle)};g.VERSION="3.3.7",g.prototype.toggle=function(d){var e=a(this);if(!e.is(".disabled, :disabled")){var f=b(e),g=f.hasClass("open");if(c(),!g){"ontouchstart"in document.documentElement&&!f.closest(".navbar-nav").length&&a(document.createElement("div")).addClass("dropdown-backdrop").insertAfter(a(this)).on("click",c);var h={relatedTarget:this};if(f.trigger(d=a.Event("show.bs.dropdown",h)),d.isDefaultPrevented())return;e.trigger("focus").attr("aria-expanded","true"),f.toggleClass("open").trigger(a.Event("shown.bs.dropdown",h))}return!1}},g.prototype.keydown=function(c){if(/(38|40|27|32)/.test(c.which)&&!/input|textarea/i.test(c.target.tagName)){var d=a(this);if(c.preventDefault(),c.stopPropagation(),!d.is(".disabled, :disabled")){var e=b(d),g=e.hasClass("open");if(!g&&27!=c.which||g&&27==c.which)return 27==c.which&&e.find(f).trigger("focus"),d.trigger("click");var h=" li:not(.disabled):visible a",i=e.find(".dropdown-menu"+h);if(i.length){var j=i.index(c.target);38==c.which&&j>0&&j--,40==c.which&&jdocument.documentElement.clientHeight;this.$element.css({paddingLeft:!this.bodyIsOverflowing&&a?this.scrollbarWidth:"",paddingRight:this.bodyIsOverflowing&&!a?this.scrollbarWidth:""})},c.prototype.resetAdjustments=function(){this.$element.css({paddingLeft:"",paddingRight:""})},c.prototype.checkScrollbar=function(){var a=window.innerWidth;if(!a){var b=document.documentElement.getBoundingClientRect();a=b.right-Math.abs(b.left)}this.bodyIsOverflowing=document.body.clientWidth
',trigger:"hover focus",title:"",delay:0,html:!1,container:!1,viewport:{selector:"body",padding:0}},c.prototype.init=function(b,c,d){if(this.enabled=!0,this.type=b,this.$element=a(c),this.options=this.getOptions(d),this.$viewport=this.options.viewport&&a(a.isFunction(this.options.viewport)?this.options.viewport.call(this,this.$element):this.options.viewport.selector||this.options.viewport),this.inState={click:!1,hover:!1,focus:!1},this.$element[0]instanceof document.constructor&&!this.options.selector)throw new Error("`selector` option must be specified when initializing "+this.type+" on the window.document object!");for(var e=this.options.trigger.split(" "),f=e.length;f--;){var g=e[f];if("click"==g)this.$element.on("click."+this.type,this.options.selector,a.proxy(this.toggle,this));else if("manual"!=g){var h="hover"==g?"mouseenter":"focusin",i="hover"==g?"mouseleave":"focusout";this.$element.on(h+"."+this.type,this.options.selector,a.proxy(this.enter,this)),this.$element.on(i+"."+this.type,this.options.selector,a.proxy(this.leave,this))}}this.options.selector?this._options=a.extend({},this.options,{trigger:"manual",selector:""}):this.fixTitle()},c.prototype.getDefaults=function(){return c.DEFAULTS},c.prototype.getOptions=function(b){return b=a.extend({},this.getDefaults(),this.$element.data(),b),b.delay&&"number"==typeof b.delay&&(b.delay={show:b.delay,hide:b.delay}),b},c.prototype.getDelegateOptions=function(){var b={},c=this.getDefaults();return this._options&&a.each(this._options,function(a,d){c[a]!=d&&(b[a]=d)}),b},c.prototype.enter=function(b){var c=b instanceof this.constructor?b:a(b.currentTarget).data("bs."+this.type);return c||(c=new this.constructor(b.currentTarget,this.getDelegateOptions()),a(b.currentTarget).data("bs."+this.type,c)),b instanceof a.Event&&(c.inState["focusin"==b.type?"focus":"hover"]=!0),c.tip().hasClass("in")||"in"==c.hoverState?void(c.hoverState="in"):(clearTimeout(c.timeout),c.hoverState="in",c.options.delay&&c.options.delay.show?void(c.timeout=setTimeout(function(){"in"==c.hoverState&&c.show()},c.options.delay.show)):c.show())},c.prototype.isInStateTrue=function(){for(var a in this.inState)if(this.inState[a])return!0;return!1},c.prototype.leave=function(b){var c=b instanceof this.constructor?b:a(b.currentTarget).data("bs."+this.type);if(c||(c=new this.constructor(b.currentTarget,this.getDelegateOptions()),a(b.currentTarget).data("bs."+this.type,c)),b instanceof a.Event&&(c.inState["focusout"==b.type?"focus":"hover"]=!1),!c.isInStateTrue())return clearTimeout(c.timeout),c.hoverState="out",c.options.delay&&c.options.delay.hide?void(c.timeout=setTimeout(function(){"out"==c.hoverState&&c.hide()},c.options.delay.hide)):c.hide()},c.prototype.show=function(){var b=a.Event("show.bs."+this.type);if(this.hasContent()&&this.enabled){this.$element.trigger(b);var d=a.contains(this.$element[0].ownerDocument.documentElement,this.$element[0]);if(b.isDefaultPrevented()||!d)return;var e=this,f=this.tip(),g=this.getUID(this.type);this.setContent(),f.attr("id",g),this.$element.attr("aria-describedby",g),this.options.animation&&f.addClass("fade");var h="function"==typeof this.options.placement?this.options.placement.call(this,f[0],this.$element[0]):this.options.placement,i=/\s?auto?\s?/i,j=i.test(h);j&&(h=h.replace(i,"")||"top"),f.detach().css({top:0,left:0,display:"block"}).addClass(h).data("bs."+this.type,this),this.options.container?f.appendTo(this.options.container):f.insertAfter(this.$element),this.$element.trigger("inserted.bs."+this.type);var k=this.getPosition(),l=f[0].offsetWidth,m=f[0].offsetHeight;if(j){var n=h,o=this.getPosition(this.$viewport);h="bottom"==h&&k.bottom+m>o.bottom?"top":"top"==h&&k.top-mo.width?"left":"left"==h&&k.left-lg.top+g.height&&(e.top=g.top+g.height-i)}else{var j=b.left-f,k=b.left+f+c;jg.right&&(e.left=g.left+g.width-k)}return e},c.prototype.getTitle=function(){var a,b=this.$element,c=this.options;return a=b.attr("data-original-title")||("function"==typeof c.title?c.title.call(b[0]):c.title)},c.prototype.getUID=function(a){do a+=~~(1e6*Math.random());while(document.getElementById(a));return a},c.prototype.tip=function(){if(!this.$tip&&(this.$tip=a(this.options.template),1!=this.$tip.length))throw new Error(this.type+" `template` option must consist of exactly 1 top-level element!");return this.$tip},c.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".tooltip-arrow")},c.prototype.enable=function(){this.enabled=!0},c.prototype.disable=function(){this.enabled=!1},c.prototype.toggleEnabled=function(){this.enabled=!this.enabled},c.prototype.toggle=function(b){var c=this;b&&(c=a(b.currentTarget).data("bs."+this.type),c||(c=new this.constructor(b.currentTarget,this.getDelegateOptions()),a(b.currentTarget).data("bs."+this.type,c))),b?(c.inState.click=!c.inState.click,c.isInStateTrue()?c.enter(c):c.leave(c)):c.tip().hasClass("in")?c.leave(c):c.enter(c)},c.prototype.destroy=function(){var a=this;clearTimeout(this.timeout),this.hide(function(){a.$element.off("."+a.type).removeData("bs."+a.type),a.$tip&&a.$tip.detach(),a.$tip=null,a.$arrow=null,a.$viewport=null,a.$element=null})};var d=a.fn.tooltip;a.fn.tooltip=b,a.fn.tooltip.Constructor=c,a.fn.tooltip.noConflict=function(){return a.fn.tooltip=d,this}}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.popover"),f="object"==typeof b&&b;!e&&/destroy|hide/.test(b)||(e||d.data("bs.popover",e=new c(this,f)),"string"==typeof b&&e[b]())})}var c=function(a,b){this.init("popover",a,b)};if(!a.fn.tooltip)throw new Error("Popover requires tooltip.js"/*tpa=http://img.imsxm.com/wp-content/themes/Kratos/js/Popover requires tooltip.js*/);c.VERSION="3.3.7",c.DEFAULTS=a.extend({},a.fn.tooltip.Constructor.DEFAULTS,{placement:"right",trigger:"click",content:"",template:''}),c.prototype=a.extend({},a.fn.tooltip.Constructor.prototype),c.prototype.constructor=c,c.prototype.getDefaults=function(){return c.DEFAULTS},c.prototype.setContent=function(){var a=this.tip(),b=this.getTitle(),c=this.getContent();a.find(".popover-title")[this.options.html?"html":"text"](b),a.find(".popover-content").children().detach().end()[this.options.html?"string"==typeof c?"html":"append":"text"](c),a.removeClass("fade top bottom left right in"),a.find(".popover-title").html()||a.find(".popover-title").hide()},c.prototype.hasContent=function(){return this.getTitle()||this.getContent()},c.prototype.getContent=function(){var a=this.$element,b=this.options;return a.attr("data-content")||("function"==typeof b.content?b.content.call(a[0]):b.content)},c.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".arrow")};var d=a.fn.popover;a.fn.popover=b,a.fn.popover.Constructor=c,a.fn.popover.noConflict=function(){return a.fn.popover=d,this}}(jQuery),+function(a){"use strict";function b(c,d){this.$body=a(document.body),this.$scrollElement=a(a(c).is(document.body)?window:c),this.options=a.extend({},b.DEFAULTS,d),this.selector=(this.options.target||"")+" .nav li > a",this.offsets=[],this.targets=[],this.activeTarget=null,this.scrollHeight=0,this.$scrollElement.on("scroll.bs.scrollspy",a.proxy(this.process,this)),this.refresh(),this.process()}function c(c){return this.each(function(){var d=a(this),e=d.data("bs.scrollspy"),f="object"==typeof c&&c;e||d.data("bs.scrollspy",e=new b(this,f)),"string"==typeof c&&e[c]()})}b.VERSION="3.3.7",b.DEFAULTS={offset:10},b.prototype.getScrollHeight=function(){return this.$scrollElement[0].scrollHeight||Math.max(this.$body[0].scrollHeight,document.documentElement.scrollHeight)},b.prototype.refresh=function(){var b=this,c="offset",d=0;this.offsets=[],this.targets=[],this.scrollHeight=this.getScrollHeight(),a.isWindow(this.$scrollElement[0])||(c="position",d=this.$scrollElement.scrollTop()),this.$body.find(this.selector).map(function(){var b=a(this),e=b.data("target")||b.attr("href"),f=/^#./.test(e)&&a(e);return f&&f.length&&f.is(":visible")&&[[f[c]().top+d,e]]||null}).sort(function(a,b){return a[0]-b[0]}).each(function(){b.offsets.push(this[0]),b.targets.push(this[1])})},b.prototype.process=function(){var a,b=this.$scrollElement.scrollTop()+this.options.offset,c=this.getScrollHeight(),d=this.options.offset+c-this.$scrollElement.height(),e=this.offsets,f=this.targets,g=this.activeTarget;if(this.scrollHeight!=c&&this.refresh(),b>=d)return g!=(a=f[f.length-1])&&this.activate(a);if(g&&b=e[a]&&(void 0===e[a+1]||b .dropdown-menu > .active").removeClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",!1),b.addClass("active").find('[data-toggle="tab"]').attr("aria-expanded",!0),h?(b[0].offsetWidth,b.addClass("in")):b.removeClass("fade"),b.parent(".dropdown-menu").length&&b.closest("li.dropdown").addClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",!0),e&&e()}var g=d.find("> .active"),h=e&&a.support.transition&&(g.length&&g.hasClass("fade")||!!d.find("> .fade").length);g.length&&h?g.one("bsTransitionEnd",f).emulateTransitionEnd(c.TRANSITION_DURATION):f(),g.removeClass("in")};var d=a.fn.tab;a.fn.tab=b,a.fn.tab.Constructor=c,a.fn.tab.noConflict=function(){return a.fn.tab=d,this};var e=function(c){c.preventDefault(),b.call(a(this),"show")};a(document).on("click.bs.tab.data-api",'[data-toggle="tab"]',e).on("click.bs.tab.data-api",'[data-toggle="pill"]',e)}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.affix"),f="object"==typeof b&&b;e||d.data("bs.affix",e=new c(this,f)),"string"==typeof b&&e[b]()})}var c=function(b,d){this.options=a.extend({},c.DEFAULTS,d),this.$target=a(this.options.target).on("scroll.bs.affix.data-api",a.proxy(this.checkPosition,this)).on("click.bs.affix.data-api",a.proxy(this.checkPositionWithEventLoop,this)),this.$element=a(b),this.affixed=null,this.unpin=null,this.pinnedOffset=null,this.checkPosition()};c.VERSION="3.3.7",c.RESET="affix affix-top affix-bottom",c.DEFAULTS={offset:0,target:window},c.prototype.getState=function(a,b,c,d){var e=this.$target.scrollTop(),f=this.$element.offset(),g=this.$target.height();if(null!=c&&"top"==this.affixed)return e=a-d&&"bottom"},c.prototype.getPinnedOffset=function(){if(this.pinnedOffset)return this.pinnedOffset;this.$element.removeClass(c.RESET).addClass("affix");var a=this.$target.scrollTop(),b=this.$element.offset();return this.pinnedOffset=b.top-a},c.prototype.checkPositionWithEventLoop=function(){setTimeout(a.proxy(this.checkPosition,this),1)},c.prototype.checkPosition=function(){if(this.$element.is(":visible")){var b=this.$element.height(),d=this.options.offset,e=d.top,f=d.bottom,g=Math.max(a(document).height(),a(document.body).height());"object"!=typeof d&&(f=e=d),"function"==typeof e&&(e=d.top(this.$element)),"function"==typeof f&&(f=d.bottom(this.$element));var h=this.getState(g,b,e,f);if(this.affixed!=h){null!=this.unpin&&this.$element.css("top","");var i="affix"+(h?"-"+h:""),j=a.Event(i+".bs.affix");if(this.$element.trigger(j),j.isDefaultPrevented())return;this.affixed=h,this.unpin="bottom"==h?this.getPinnedOffset():null,this.$element.removeClass(c.RESET).addClass(i).trigger(i.replace("affix","affixed")+".bs.affix")}"bottom"==h&&this.$element.offset({top:g-b-f})}};var d=a.fn.affix;a.fn.affix=b,a.fn.affix.Constructor=c,a.fn.affix.noConflict=function(){return a.fn.affix=d,this},a(window).on("load",function(){a('[data-spy="affix"]').each(function(){var c=a(this),d=c.data();d.offset=d.offset||{},null!=d.offsetBottom&&(d.offset.bottom=d.offsetBottom),null!=d.offsetTop&&(d.offset.top=d.offsetTop),b.call(c,d)})})}(jQuery); -------------------------------------------------------------------------------- /app/static/js/jquery.min.js: -------------------------------------------------------------------------------- 1 | /*! jQuery v1.12.2 | (c) jQuery Foundation | jquery.org/license */ 2 | !function(a,b){"object"==typeof module&&"object"==typeof module.exports?module.exports=a.document?b(a,!0):function(a){if(!a.document)throw new Error("jQuery requires a window with a document");return b(a)}:b(a)}("undefined"!=typeof window?window:this,function(a,b){var c=[],d=a.document,e=c.slice,f=c.concat,g=c.push,h=c.indexOf,i={},j=i.toString,k=i.hasOwnProperty,l={},m="1.12.2",n=function(a,b){return new n.fn.init(a,b)},o=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,p=/^-ms-/,q=/-([\da-z])/gi,r=function(a,b){return b.toUpperCase()};n.fn=n.prototype={jquery:m,constructor:n,selector:"",length:0,toArray:function(){return e.call(this)},get:function(a){return null!=a?0>a?this[a+this.length]:this[a]:e.call(this)},pushStack:function(a){var b=n.merge(this.constructor(),a);return b.prevObject=this,b.context=this.context,b},each:function(a){return n.each(this,a)},map:function(a){return this.pushStack(n.map(this,function(b,c){return a.call(b,c,b)}))},slice:function(){return this.pushStack(e.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(a){var b=this.length,c=+a+(0>a?b:0);return this.pushStack(c>=0&&b>c?[this[c]]:[])},end:function(){return this.prevObject||this.constructor()},push:g,sort:c.sort,splice:c.splice},n.extend=n.fn.extend=function(){var a,b,c,d,e,f,g=arguments[0]||{},h=1,i=arguments.length,j=!1;for("boolean"==typeof g&&(j=g,g=arguments[h]||{},h++),"object"==typeof g||n.isFunction(g)||(g={}),h===i&&(g=this,h--);i>h;h++)if(null!=(e=arguments[h]))for(d in e)a=g[d],c=e[d],g!==c&&(j&&c&&(n.isPlainObject(c)||(b=n.isArray(c)))?(b?(b=!1,f=a&&n.isArray(a)?a:[]):f=a&&n.isPlainObject(a)?a:{},g[d]=n.extend(j,f,c)):void 0!==c&&(g[d]=c));return g},n.extend({expando:"jQuery"+(m+Math.random()).replace(/\D/g,""),isReady:!0,error:function(a){throw new Error(a)},noop:function(){},isFunction:function(a){return"function"===n.type(a)},isArray:Array.isArray||function(a){return"array"===n.type(a)},isWindow:function(a){return null!=a&&a==a.window},isNumeric:function(a){var b=a&&a.toString();return!n.isArray(a)&&b-parseFloat(b)+1>=0},isEmptyObject:function(a){var b;for(b in a)return!1;return!0},isPlainObject:function(a){var b;if(!a||"object"!==n.type(a)||a.nodeType||n.isWindow(a))return!1;try{if(a.constructor&&!k.call(a,"constructor")&&!k.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(c){return!1}if(!l.ownFirst)for(b in a)return k.call(a,b);for(b in a);return void 0===b||k.call(a,b)},type:function(a){return null==a?a+"":"object"==typeof a||"function"==typeof a?i[j.call(a)]||"object":typeof a},globalEval:function(b){b&&n.trim(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(p,"ms-").replace(q,r)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toLowerCase()===b.toLowerCase()},each:function(a,b){var c,d=0;if(s(a)){for(c=a.length;c>d;d++)if(b.call(a[d],d,a[d])===!1)break}else for(d in a)if(b.call(a[d],d,a[d])===!1)break;return a},trim:function(a){return null==a?"":(a+"").replace(o,"")},makeArray:function(a,b){var c=b||[];return null!=a&&(s(Object(a))?n.merge(c,"string"==typeof a?[a]:a):g.call(c,a)),c},inArray:function(a,b,c){var d;if(b){if(h)return h.call(b,a,c);for(d=b.length,c=c?0>c?Math.max(0,d+c):c:0;d>c;c++)if(c in b&&b[c]===a)return c}return-1},merge:function(a,b){var c=+b.length,d=0,e=a.length;while(c>d)a[e++]=b[d++];if(c!==c)while(void 0!==b[d])a[e++]=b[d++];return a.length=e,a},grep:function(a,b,c){for(var d,e=[],f=0,g=a.length,h=!c;g>f;f++)d=!b(a[f],f),d!==h&&e.push(a[f]);return e},map:function(a,b,c){var d,e,g=0,h=[];if(s(a))for(d=a.length;d>g;g++)e=b(a[g],g,c),null!=e&&h.push(e);else for(g in a)e=b(a[g],g,c),null!=e&&h.push(e);return f.apply([],h)},guid:1,proxy:function(a,b){var c,d,f;return"string"==typeof b&&(f=a[b],b=a,a=f),n.isFunction(a)?(c=e.call(arguments,2),d=function(){return a.apply(b||this,c.concat(e.call(arguments)))},d.guid=a.guid=a.guid||n.guid++,d):void 0},now:function(){return+new Date},support:l}),"function"==typeof Symbol&&(n.fn[Symbol.iterator]=c[Symbol.iterator]),n.each("Boolean Number String Function Array Date RegExp Object Error Symbol".split(" "),function(a,b){i["[object "+b+"]"]=b.toLowerCase()});function s(a){var b=!!a&&"length"in a&&a.length,c=n.type(a);return"function"===c||n.isWindow(a)?!1:"array"===c||0===b||"number"==typeof b&&b>0&&b-1 in a}var t=function(a){var b,c,d,e,f,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u="sizzle"+1*new Date,v=a.document,w=0,x=0,y=ga(),z=ga(),A=ga(),B=function(a,b){return a===b&&(l=!0),0},C=1<<31,D={}.hasOwnProperty,E=[],F=E.pop,G=E.push,H=E.push,I=E.slice,J=function(a,b){for(var c=0,d=a.length;d>c;c++)if(a[c]===b)return c;return-1},K="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",L="[\\x20\\t\\r\\n\\f]",M="(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",N="\\["+L+"*("+M+")(?:"+L+"*([*^$|!~]?=)"+L+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+M+"))|)"+L+"*\\]",O=":("+M+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+N+")*)|.*)\\)|)",P=new RegExp(L+"+","g"),Q=new RegExp("^"+L+"+|((?:^|[^\\\\])(?:\\\\.)*)"+L+"+$","g"),R=new RegExp("^"+L+"*,"+L+"*"),S=new RegExp("^"+L+"*([>+~]|"+L+")"+L+"*"),T=new RegExp("="+L+"*([^\\]'\"]*?)"+L+"*\\]","g"),U=new RegExp(O),V=new RegExp("^"+M+"$"),W={ID:new RegExp("^#("+M+")"),CLASS:new RegExp("^\\.("+M+")"),TAG:new RegExp("^("+M+"|[*])"),ATTR:new RegExp("^"+N),PSEUDO:new RegExp("^"+O),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+L+"*(even|odd|(([+-]|)(\\d*)n|)"+L+"*(?:([+-]|)"+L+"*(\\d+)|))"+L+"*\\)|)","i"),bool:new RegExp("^(?:"+K+")$","i"),needsContext:new RegExp("^"+L+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+L+"*((?:-\\d)?\\d*)"+L+"*\\)|)(?=[^-]|$)","i")},X=/^(?:input|select|textarea|button)$/i,Y=/^h\d$/i,Z=/^[^{]+\{\s*\[native \w/,$=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,_=/[+~]/,aa=/'|\\/g,ba=new RegExp("\\\\([\\da-f]{1,6}"+L+"?|("+L+")|.)","ig"),ca=function(a,b,c){var d="0x"+b-65536;return d!==d||c?b:0>d?String.fromCharCode(d+65536):String.fromCharCode(d>>10|55296,1023&d|56320)},da=function(){m()};try{H.apply(E=I.call(v.childNodes),v.childNodes),E[v.childNodes.length].nodeType}catch(ea){H={apply:E.length?function(a,b){G.apply(a,I.call(b))}:function(a,b){var c=a.length,d=0;while(a[c++]=b[d++]);a.length=c-1}}}function fa(a,b,d,e){var f,h,j,k,l,o,r,s,w=b&&b.ownerDocument,x=b?b.nodeType:9;if(d=d||[],"string"!=typeof a||!a||1!==x&&9!==x&&11!==x)return d;if(!e&&((b?b.ownerDocument||b:v)!==n&&m(b),b=b||n,p)){if(11!==x&&(o=$.exec(a)))if(f=o[1]){if(9===x){if(!(j=b.getElementById(f)))return d;if(j.id===f)return d.push(j),d}else if(w&&(j=w.getElementById(f))&&t(b,j)&&j.id===f)return d.push(j),d}else{if(o[2])return H.apply(d,b.getElementsByTagName(a)),d;if((f=o[3])&&c.getElementsByClassName&&b.getElementsByClassName)return H.apply(d,b.getElementsByClassName(f)),d}if(c.qsa&&!A[a+" "]&&(!q||!q.test(a))){if(1!==x)w=b,s=a;else if("object"!==b.nodeName.toLowerCase()){(k=b.getAttribute("id"))?k=k.replace(aa,"\\$&"):b.setAttribute("id",k=u),r=g(a),h=r.length,l=V.test(k)?"#"+k:"[id='"+k+"']";while(h--)r[h]=l+" "+qa(r[h]);s=r.join(","),w=_.test(a)&&oa(b.parentNode)||b}if(s)try{return H.apply(d,w.querySelectorAll(s)),d}catch(y){}finally{k===u&&b.removeAttribute("id")}}}return i(a.replace(Q,"$1"),b,d,e)}function ga(){var a=[];function b(c,e){return a.push(c+" ")>d.cacheLength&&delete b[a.shift()],b[c+" "]=e}return b}function ha(a){return a[u]=!0,a}function ia(a){var b=n.createElement("div");try{return!!a(b)}catch(c){return!1}finally{b.parentNode&&b.parentNode.removeChild(b),b=null}}function ja(a,b){var c=a.split("|"),e=c.length;while(e--)d.attrHandle[c[e]]=b}function ka(a,b){var c=b&&a,d=c&&1===a.nodeType&&1===b.nodeType&&(~b.sourceIndex||C)-(~a.sourceIndex||C);if(d)return d;if(c)while(c=c.nextSibling)if(c===b)return-1;return a?1:-1}function la(a){return function(b){var c=b.nodeName.toLowerCase();return"input"===c&&b.type===a}}function ma(a){return function(b){var c=b.nodeName.toLowerCase();return("input"===c||"button"===c)&&b.type===a}}function na(a){return ha(function(b){return b=+b,ha(function(c,d){var e,f=a([],c.length,b),g=f.length;while(g--)c[e=f[g]]&&(c[e]=!(d[e]=c[e]))})})}function oa(a){return a&&"undefined"!=typeof a.getElementsByTagName&&a}c=fa.support={},f=fa.isXML=function(a){var b=a&&(a.ownerDocument||a).documentElement;return b?"HTML"!==b.nodeName:!1},m=fa.setDocument=function(a){var b,e,g=a?a.ownerDocument||a:v;return g!==n&&9===g.nodeType&&g.documentElement?(n=g,o=n.documentElement,p=!f(n),(e=n.defaultView)&&e.top!==e&&(e.addEventListener?e.addEventListener("unload",da,!1):e.attachEvent&&e.attachEvent("onunload",da)),c.attributes=ia(function(a){return a.className="i",!a.getAttribute("className")}),c.getElementsByTagName=ia(function(a){return a.appendChild(n.createComment("")),!a.getElementsByTagName("*").length}),c.getElementsByClassName=Z.test(n.getElementsByClassName),c.getById=ia(function(a){return o.appendChild(a).id=u,!n.getElementsByName||!n.getElementsByName(u).length}),c.getById?(d.find.ID=function(a,b){if("undefined"!=typeof b.getElementById&&p){var c=b.getElementById(a);return c?[c]:[]}},d.filter.ID=function(a){var b=a.replace(ba,ca);return function(a){return a.getAttribute("id")===b}}):(delete d.find.ID,d.filter.ID=function(a){var b=a.replace(ba,ca);return function(a){var c="undefined"!=typeof a.getAttributeNode&&a.getAttributeNode("id");return c&&c.value===b}}),d.find.TAG=c.getElementsByTagName?function(a,b){return"undefined"!=typeof b.getElementsByTagName?b.getElementsByTagName(a):c.qsa?b.querySelectorAll(a):void 0}:function(a,b){var c,d=[],e=0,f=b.getElementsByTagName(a);if("*"===a){while(c=f[e++])1===c.nodeType&&d.push(c);return d}return f},d.find.CLASS=c.getElementsByClassName&&function(a,b){return"undefined"!=typeof b.getElementsByClassName&&p?b.getElementsByClassName(a):void 0},r=[],q=[],(c.qsa=Z.test(n.querySelectorAll))&&(ia(function(a){o.appendChild(a).innerHTML="",a.querySelectorAll("[msallowcapture^='']").length&&q.push("[*^$]="+L+"*(?:''|\"\")"),a.querySelectorAll("[selected]").length||q.push("\\["+L+"*(?:value|"+K+")"),a.querySelectorAll("[id~="+u+"-]").length||q.push("~="),a.querySelectorAll(":checked").length||q.push(":checked"),a.querySelectorAll("a#"+u+"+*").length||q.push(".#.+[+~]")}),ia(function(a){var b=n.createElement("input");b.setAttribute("type","hidden"),a.appendChild(b).setAttribute("name","D"),a.querySelectorAll("[name=d]").length&&q.push("name"+L+"*[*^$|!~]?="),a.querySelectorAll(":enabled").length||q.push(":enabled",":disabled"),a.querySelectorAll("*,:x"),q.push(",.*:")})),(c.matchesSelector=Z.test(s=o.matches||o.webkitMatchesSelector||o.mozMatchesSelector||o.oMatchesSelector||o.msMatchesSelector))&&ia(function(a){c.disconnectedMatch=s.call(a,"div"),s.call(a,"[s!='']:x"),r.push("!=",O)}),q=q.length&&new RegExp(q.join("|")),r=r.length&&new RegExp(r.join("|")),b=Z.test(o.compareDocumentPosition),t=b||Z.test(o.contains)?function(a,b){var c=9===a.nodeType?a.documentElement:a,d=b&&b.parentNode;return a===d||!(!d||1!==d.nodeType||!(c.contains?c.contains(d):a.compareDocumentPosition&&16&a.compareDocumentPosition(d)))}:function(a,b){if(b)while(b=b.parentNode)if(b===a)return!0;return!1},B=b?function(a,b){if(a===b)return l=!0,0;var d=!a.compareDocumentPosition-!b.compareDocumentPosition;return d?d:(d=(a.ownerDocument||a)===(b.ownerDocument||b)?a.compareDocumentPosition(b):1,1&d||!c.sortDetached&&b.compareDocumentPosition(a)===d?a===n||a.ownerDocument===v&&t(v,a)?-1:b===n||b.ownerDocument===v&&t(v,b)?1:k?J(k,a)-J(k,b):0:4&d?-1:1)}:function(a,b){if(a===b)return l=!0,0;var c,d=0,e=a.parentNode,f=b.parentNode,g=[a],h=[b];if(!e||!f)return a===n?-1:b===n?1:e?-1:f?1:k?J(k,a)-J(k,b):0;if(e===f)return ka(a,b);c=a;while(c=c.parentNode)g.unshift(c);c=b;while(c=c.parentNode)h.unshift(c);while(g[d]===h[d])d++;return d?ka(g[d],h[d]):g[d]===v?-1:h[d]===v?1:0},n):n},fa.matches=function(a,b){return fa(a,null,null,b)},fa.matchesSelector=function(a,b){if((a.ownerDocument||a)!==n&&m(a),b=b.replace(T,"='$1']"),c.matchesSelector&&p&&!A[b+" "]&&(!r||!r.test(b))&&(!q||!q.test(b)))try{var d=s.call(a,b);if(d||c.disconnectedMatch||a.document&&11!==a.document.nodeType)return d}catch(e){}return fa(b,n,null,[a]).length>0},fa.contains=function(a,b){return(a.ownerDocument||a)!==n&&m(a),t(a,b)},fa.attr=function(a,b){(a.ownerDocument||a)!==n&&m(a);var e=d.attrHandle[b.toLowerCase()],f=e&&D.call(d.attrHandle,b.toLowerCase())?e(a,b,!p):void 0;return void 0!==f?f:c.attributes||!p?a.getAttribute(b):(f=a.getAttributeNode(b))&&f.specified?f.value:null},fa.error=function(a){throw new Error("Syntax error, unrecognized expression: "+a)},fa.uniqueSort=function(a){var b,d=[],e=0,f=0;if(l=!c.detectDuplicates,k=!c.sortStable&&a.slice(0),a.sort(B),l){while(b=a[f++])b===a[f]&&(e=d.push(f));while(e--)a.splice(d[e],1)}return k=null,a},e=fa.getText=function(a){var b,c="",d=0,f=a.nodeType;if(f){if(1===f||9===f||11===f){if("string"==typeof a.textContent)return a.textContent;for(a=a.firstChild;a;a=a.nextSibling)c+=e(a)}else if(3===f||4===f)return a.nodeValue}else while(b=a[d++])c+=e(b);return c},d=fa.selectors={cacheLength:50,createPseudo:ha,match:W,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(a){return a[1]=a[1].replace(ba,ca),a[3]=(a[3]||a[4]||a[5]||"").replace(ba,ca),"~="===a[2]&&(a[3]=" "+a[3]+" "),a.slice(0,4)},CHILD:function(a){return a[1]=a[1].toLowerCase(),"nth"===a[1].slice(0,3)?(a[3]||fa.error(a[0]),a[4]=+(a[4]?a[5]+(a[6]||1):2*("even"===a[3]||"odd"===a[3])),a[5]=+(a[7]+a[8]||"odd"===a[3])):a[3]&&fa.error(a[0]),a},PSEUDO:function(a){var b,c=!a[6]&&a[2];return W.CHILD.test(a[0])?null:(a[3]?a[2]=a[4]||a[5]||"":c&&U.test(c)&&(b=g(c,!0))&&(b=c.indexOf(")",c.length-b)-c.length)&&(a[0]=a[0].slice(0,b),a[2]=c.slice(0,b)),a.slice(0,3))}},filter:{TAG:function(a){var b=a.replace(ba,ca).toLowerCase();return"*"===a?function(){return!0}:function(a){return a.nodeName&&a.nodeName.toLowerCase()===b}},CLASS:function(a){var b=y[a+" "];return b||(b=new RegExp("(^|"+L+")"+a+"("+L+"|$)"))&&y(a,function(a){return b.test("string"==typeof a.className&&a.className||"undefined"!=typeof a.getAttribute&&a.getAttribute("class")||"")})},ATTR:function(a,b,c){return function(d){var e=fa.attr(d,a);return null==e?"!="===b:b?(e+="","="===b?e===c:"!="===b?e!==c:"^="===b?c&&0===e.indexOf(c):"*="===b?c&&e.indexOf(c)>-1:"$="===b?c&&e.slice(-c.length)===c:"~="===b?(" "+e.replace(P," ")+" ").indexOf(c)>-1:"|="===b?e===c||e.slice(0,c.length+1)===c+"-":!1):!0}},CHILD:function(a,b,c,d,e){var f="nth"!==a.slice(0,3),g="last"!==a.slice(-4),h="of-type"===b;return 1===d&&0===e?function(a){return!!a.parentNode}:function(b,c,i){var j,k,l,m,n,o,p=f!==g?"nextSibling":"previousSibling",q=b.parentNode,r=h&&b.nodeName.toLowerCase(),s=!i&&!h,t=!1;if(q){if(f){while(p){m=b;while(m=m[p])if(h?m.nodeName.toLowerCase()===r:1===m.nodeType)return!1;o=p="only"===a&&!o&&"nextSibling"}return!0}if(o=[g?q.firstChild:q.lastChild],g&&s){m=q,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n&&j[2],m=n&&q.childNodes[n];while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if(1===m.nodeType&&++t&&m===b){k[a]=[w,n,t];break}}else if(s&&(m=b,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n),t===!1)while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if((h?m.nodeName.toLowerCase()===r:1===m.nodeType)&&++t&&(s&&(l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),k[a]=[w,t]),m===b))break;return t-=e,t===d||t%d===0&&t/d>=0}}},PSEUDO:function(a,b){var c,e=d.pseudos[a]||d.setFilters[a.toLowerCase()]||fa.error("unsupported pseudo: "+a);return e[u]?e(b):e.length>1?(c=[a,a,"",b],d.setFilters.hasOwnProperty(a.toLowerCase())?ha(function(a,c){var d,f=e(a,b),g=f.length;while(g--)d=J(a,f[g]),a[d]=!(c[d]=f[g])}):function(a){return e(a,0,c)}):e}},pseudos:{not:ha(function(a){var b=[],c=[],d=h(a.replace(Q,"$1"));return d[u]?ha(function(a,b,c,e){var f,g=d(a,null,e,[]),h=a.length;while(h--)(f=g[h])&&(a[h]=!(b[h]=f))}):function(a,e,f){return b[0]=a,d(b,null,f,c),b[0]=null,!c.pop()}}),has:ha(function(a){return function(b){return fa(a,b).length>0}}),contains:ha(function(a){return a=a.replace(ba,ca),function(b){return(b.textContent||b.innerText||e(b)).indexOf(a)>-1}}),lang:ha(function(a){return V.test(a||"")||fa.error("unsupported lang: "+a),a=a.replace(ba,ca).toLowerCase(),function(b){var c;do if(c=p?b.lang:b.getAttribute("xml:lang")||b.getAttribute("lang"))return c=c.toLowerCase(),c===a||0===c.indexOf(a+"-");while((b=b.parentNode)&&1===b.nodeType);return!1}}),target:function(b){var c=a.location&&a.location.hash;return c&&c.slice(1)===b.id},root:function(a){return a===o},focus:function(a){return a===n.activeElement&&(!n.hasFocus||n.hasFocus())&&!!(a.type||a.href||~a.tabIndex)},enabled:function(a){return a.disabled===!1},disabled:function(a){return a.disabled===!0},checked:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&!!a.checked||"option"===b&&!!a.selected},selected:function(a){return a.parentNode&&a.parentNode.selectedIndex,a.selected===!0},empty:function(a){for(a=a.firstChild;a;a=a.nextSibling)if(a.nodeType<6)return!1;return!0},parent:function(a){return!d.pseudos.empty(a)},header:function(a){return Y.test(a.nodeName)},input:function(a){return X.test(a.nodeName)},button:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&"button"===a.type||"button"===b},text:function(a){var b;return"input"===a.nodeName.toLowerCase()&&"text"===a.type&&(null==(b=a.getAttribute("type"))||"text"===b.toLowerCase())},first:na(function(){return[0]}),last:na(function(a,b){return[b-1]}),eq:na(function(a,b,c){return[0>c?c+b:c]}),even:na(function(a,b){for(var c=0;b>c;c+=2)a.push(c);return a}),odd:na(function(a,b){for(var c=1;b>c;c+=2)a.push(c);return a}),lt:na(function(a,b,c){for(var d=0>c?c+b:c;--d>=0;)a.push(d);return a}),gt:na(function(a,b,c){for(var d=0>c?c+b:c;++db;b++)d+=a[b].value;return d}function ra(a,b,c){var d=b.dir,e=c&&"parentNode"===d,f=x++;return b.first?function(b,c,f){while(b=b[d])if(1===b.nodeType||e)return a(b,c,f)}:function(b,c,g){var h,i,j,k=[w,f];if(g){while(b=b[d])if((1===b.nodeType||e)&&a(b,c,g))return!0}else while(b=b[d])if(1===b.nodeType||e){if(j=b[u]||(b[u]={}),i=j[b.uniqueID]||(j[b.uniqueID]={}),(h=i[d])&&h[0]===w&&h[1]===f)return k[2]=h[2];if(i[d]=k,k[2]=a(b,c,g))return!0}}}function sa(a){return a.length>1?function(b,c,d){var e=a.length;while(e--)if(!a[e](b,c,d))return!1;return!0}:a[0]}function ta(a,b,c){for(var d=0,e=b.length;e>d;d++)fa(a,b[d],c);return c}function ua(a,b,c,d,e){for(var f,g=[],h=0,i=a.length,j=null!=b;i>h;h++)(f=a[h])&&(c&&!c(f,d,e)||(g.push(f),j&&b.push(h)));return g}function va(a,b,c,d,e,f){return d&&!d[u]&&(d=va(d)),e&&!e[u]&&(e=va(e,f)),ha(function(f,g,h,i){var j,k,l,m=[],n=[],o=g.length,p=f||ta(b||"*",h.nodeType?[h]:h,[]),q=!a||!f&&b?p:ua(p,m,a,h,i),r=c?e||(f?a:o||d)?[]:g:q;if(c&&c(q,r,h,i),d){j=ua(r,n),d(j,[],h,i),k=j.length;while(k--)(l=j[k])&&(r[n[k]]=!(q[n[k]]=l))}if(f){if(e||a){if(e){j=[],k=r.length;while(k--)(l=r[k])&&j.push(q[k]=l);e(null,r=[],j,i)}k=r.length;while(k--)(l=r[k])&&(j=e?J(f,l):m[k])>-1&&(f[j]=!(g[j]=l))}}else r=ua(r===g?r.splice(o,r.length):r),e?e(null,g,r,i):H.apply(g,r)})}function wa(a){for(var b,c,e,f=a.length,g=d.relative[a[0].type],h=g||d.relative[" "],i=g?1:0,k=ra(function(a){return a===b},h,!0),l=ra(function(a){return J(b,a)>-1},h,!0),m=[function(a,c,d){var e=!g&&(d||c!==j)||((b=c).nodeType?k(a,c,d):l(a,c,d));return b=null,e}];f>i;i++)if(c=d.relative[a[i].type])m=[ra(sa(m),c)];else{if(c=d.filter[a[i].type].apply(null,a[i].matches),c[u]){for(e=++i;f>e;e++)if(d.relative[a[e].type])break;return va(i>1&&sa(m),i>1&&qa(a.slice(0,i-1).concat({value:" "===a[i-2].type?"*":""})).replace(Q,"$1"),c,e>i&&wa(a.slice(i,e)),f>e&&wa(a=a.slice(e)),f>e&&qa(a))}m.push(c)}return sa(m)}function xa(a,b){var c=b.length>0,e=a.length>0,f=function(f,g,h,i,k){var l,o,q,r=0,s="0",t=f&&[],u=[],v=j,x=f||e&&d.find.TAG("*",k),y=w+=null==v?1:Math.random()||.1,z=x.length;for(k&&(j=g===n||g||k);s!==z&&null!=(l=x[s]);s++){if(e&&l){o=0,g||l.ownerDocument===n||(m(l),h=!p);while(q=a[o++])if(q(l,g||n,h)){i.push(l);break}k&&(w=y)}c&&((l=!q&&l)&&r--,f&&t.push(l))}if(r+=s,c&&s!==r){o=0;while(q=b[o++])q(t,u,g,h);if(f){if(r>0)while(s--)t[s]||u[s]||(u[s]=F.call(i));u=ua(u)}H.apply(i,u),k&&!f&&u.length>0&&r+b.length>1&&fa.uniqueSort(i)}return k&&(w=y,j=v),t};return c?ha(f):f}return h=fa.compile=function(a,b){var c,d=[],e=[],f=A[a+" "];if(!f){b||(b=g(a)),c=b.length;while(c--)f=wa(b[c]),f[u]?d.push(f):e.push(f);f=A(a,xa(e,d)),f.selector=a}return f},i=fa.select=function(a,b,e,f){var i,j,k,l,m,n="function"==typeof a&&a,o=!f&&g(a=n.selector||a);if(e=e||[],1===o.length){if(j=o[0]=o[0].slice(0),j.length>2&&"ID"===(k=j[0]).type&&c.getById&&9===b.nodeType&&p&&d.relative[j[1].type]){if(b=(d.find.ID(k.matches[0].replace(ba,ca),b)||[])[0],!b)return e;n&&(b=b.parentNode),a=a.slice(j.shift().value.length)}i=W.needsContext.test(a)?0:j.length;while(i--){if(k=j[i],d.relative[l=k.type])break;if((m=d.find[l])&&(f=m(k.matches[0].replace(ba,ca),_.test(j[0].type)&&oa(b.parentNode)||b))){if(j.splice(i,1),a=f.length&&qa(j),!a)return H.apply(e,f),e;break}}}return(n||h(a,o))(f,b,!p,e,!b||_.test(a)&&oa(b.parentNode)||b),e},c.sortStable=u.split("").sort(B).join("")===u,c.detectDuplicates=!!l,m(),c.sortDetached=ia(function(a){return 1&a.compareDocumentPosition(n.createElement("div"))}),ia(function(a){return a.innerHTML="","#"===a.firstChild.getAttribute("href")})||ja("type|href|height|width",function(a,b,c){return c?void 0:a.getAttribute(b,"type"===b.toLowerCase()?1:2)}),c.attributes&&ia(function(a){return a.innerHTML="",a.firstChild.setAttribute("value",""),""===a.firstChild.getAttribute("value")})||ja("value",function(a,b,c){return c||"input"!==a.nodeName.toLowerCase()?void 0:a.defaultValue}),ia(function(a){return null==a.getAttribute("disabled")})||ja(K,function(a,b,c){var d;return c?void 0:a[b]===!0?b.toLowerCase():(d=a.getAttributeNode(b))&&d.specified?d.value:null}),fa}(a);n.find=t,n.expr=t.selectors,n.expr[":"]=n.expr.pseudos,n.uniqueSort=n.unique=t.uniqueSort,n.text=t.getText,n.isXMLDoc=t.isXML,n.contains=t.contains;var u=function(a,b,c){var d=[],e=void 0!==c;while((a=a[b])&&9!==a.nodeType)if(1===a.nodeType){if(e&&n(a).is(c))break;d.push(a)}return d},v=function(a,b){for(var c=[];a;a=a.nextSibling)1===a.nodeType&&a!==b&&c.push(a);return c},w=n.expr.match.needsContext,x=/^<([\w-]+)\s*\/?>(?:<\/\1>|)$/,y=/^.[^:#\[\.,]*$/;function z(a,b,c){if(n.isFunction(b))return n.grep(a,function(a,d){return!!b.call(a,d,a)!==c});if(b.nodeType)return n.grep(a,function(a){return a===b!==c});if("string"==typeof b){if(y.test(b))return n.filter(b,a,c);b=n.filter(b,a)}return n.grep(a,function(a){return n.inArray(a,b)>-1!==c})}n.filter=function(a,b,c){var d=b[0];return c&&(a=":not("+a+")"),1===b.length&&1===d.nodeType?n.find.matchesSelector(d,a)?[d]:[]:n.find.matches(a,n.grep(b,function(a){return 1===a.nodeType}))},n.fn.extend({find:function(a){var b,c=[],d=this,e=d.length;if("string"!=typeof a)return this.pushStack(n(a).filter(function(){for(b=0;e>b;b++)if(n.contains(d[b],this))return!0}));for(b=0;e>b;b++)n.find(a,d[b],c);return c=this.pushStack(e>1?n.unique(c):c),c.selector=this.selector?this.selector+" "+a:a,c},filter:function(a){return this.pushStack(z(this,a||[],!1))},not:function(a){return this.pushStack(z(this,a||[],!0))},is:function(a){return!!z(this,"string"==typeof a&&w.test(a)?n(a):a||[],!1).length}});var A,B=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/,C=n.fn.init=function(a,b,c){var e,f;if(!a)return this;if(c=c||A,"string"==typeof a){if(e="<"===a.charAt(0)&&">"===a.charAt(a.length-1)&&a.length>=3?[null,a,null]:B.exec(a),!e||!e[1]&&b)return!b||b.jquery?(b||c).find(a):this.constructor(b).find(a);if(e[1]){if(b=b instanceof n?b[0]:b,n.merge(this,n.parseHTML(e[1],b&&b.nodeType?b.ownerDocument||b:d,!0)),x.test(e[1])&&n.isPlainObject(b))for(e in b)n.isFunction(this[e])?this[e](b[e]):this.attr(e,b[e]);return this}if(f=d.getElementById(e[2]),f&&f.parentNode){if(f.id!==e[2])return A.find(a);this.length=1,this[0]=f}return this.context=d,this.selector=a,this}return a.nodeType?(this.context=this[0]=a,this.length=1,this):n.isFunction(a)?"undefined"!=typeof c.ready?c.ready(a):a(n):(void 0!==a.selector&&(this.selector=a.selector,this.context=a.context),n.makeArray(a,this))};C.prototype=n.fn,A=n(d);var D=/^(?:parents|prev(?:Until|All))/,E={children:!0,contents:!0,next:!0,prev:!0};n.fn.extend({has:function(a){var b,c=n(a,this),d=c.length;return this.filter(function(){for(b=0;d>b;b++)if(n.contains(this,c[b]))return!0})},closest:function(a,b){for(var c,d=0,e=this.length,f=[],g=w.test(a)||"string"!=typeof a?n(a,b||this.context):0;e>d;d++)for(c=this[d];c&&c!==b;c=c.parentNode)if(c.nodeType<11&&(g?g.index(c)>-1:1===c.nodeType&&n.find.matchesSelector(c,a))){f.push(c);break}return this.pushStack(f.length>1?n.uniqueSort(f):f)},index:function(a){return a?"string"==typeof a?n.inArray(this[0],n(a)):n.inArray(a.jquery?a[0]:a,this):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(a,b){return this.pushStack(n.uniqueSort(n.merge(this.get(),n(a,b))))},addBack:function(a){return this.add(null==a?this.prevObject:this.prevObject.filter(a))}});function F(a,b){do a=a[b];while(a&&1!==a.nodeType);return a}n.each({parent:function(a){var b=a.parentNode;return b&&11!==b.nodeType?b:null},parents:function(a){return u(a,"parentNode")},parentsUntil:function(a,b,c){return u(a,"parentNode",c)},next:function(a){return F(a,"nextSibling")},prev:function(a){return F(a,"previousSibling")},nextAll:function(a){return u(a,"nextSibling")},prevAll:function(a){return u(a,"previousSibling")},nextUntil:function(a,b,c){return u(a,"nextSibling",c)},prevUntil:function(a,b,c){return u(a,"previousSibling",c)},siblings:function(a){return v((a.parentNode||{}).firstChild,a)},children:function(a){return v(a.firstChild)},contents:function(a){return n.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:n.merge([],a.childNodes)}},function(a,b){n.fn[a]=function(c,d){var e=n.map(this,b,c);return"Until"!==a.slice(-5)&&(d=c),d&&"string"==typeof d&&(e=n.filter(d,e)),this.length>1&&(E[a]||(e=n.uniqueSort(e)),D.test(a)&&(e=e.reverse())),this.pushStack(e)}});var G=/\S+/g;function H(a){var b={};return n.each(a.match(G)||[],function(a,c){b[c]=!0}),b}n.Callbacks=function(a){a="string"==typeof a?H(a):n.extend({},a);var b,c,d,e,f=[],g=[],h=-1,i=function(){for(e=a.once,d=b=!0;g.length;h=-1){c=g.shift();while(++h-1)f.splice(c,1),h>=c&&h--}),this},has:function(a){return a?n.inArray(a,f)>-1:f.length>0},empty:function(){return f&&(f=[]),this},disable:function(){return e=g=[],f=c="",this},disabled:function(){return!f},lock:function(){return e=!0,c||j.disable(),this},locked:function(){return!!e},fireWith:function(a,c){return e||(c=c||[],c=[a,c.slice?c.slice():c],g.push(c),b||i()),this},fire:function(){return j.fireWith(this,arguments),this},fired:function(){return!!d}};return j},n.extend({Deferred:function(a){var b=[["resolve","done",n.Callbacks("once memory"),"resolved"],["reject","fail",n.Callbacks("once memory"),"rejected"],["notify","progress",n.Callbacks("memory")]],c="pending",d={state:function(){return c},always:function(){return e.done(arguments).fail(arguments),this},then:function(){var a=arguments;return n.Deferred(function(c){n.each(b,function(b,f){var g=n.isFunction(a[b])&&a[b];e[f[1]](function(){var a=g&&g.apply(this,arguments);a&&n.isFunction(a.promise)?a.promise().progress(c.notify).done(c.resolve).fail(c.reject):c[f[0]+"With"](this===d?c.promise():this,g?[a]:arguments)})}),a=null}).promise()},promise:function(a){return null!=a?n.extend(a,d):d}},e={};return d.pipe=d.then,n.each(b,function(a,f){var g=f[2],h=f[3];d[f[1]]=g.add,h&&g.add(function(){c=h},b[1^a][2].disable,b[2][2].lock),e[f[0]]=function(){return e[f[0]+"With"](this===e?d:this,arguments),this},e[f[0]+"With"]=g.fireWith}),d.promise(e),a&&a.call(e,e),e},when:function(a){var b=0,c=e.call(arguments),d=c.length,f=1!==d||a&&n.isFunction(a.promise)?d:0,g=1===f?a:n.Deferred(),h=function(a,b,c){return function(d){b[a]=this,c[a]=arguments.length>1?e.call(arguments):d,c===i?g.notifyWith(b,c):--f||g.resolveWith(b,c)}},i,j,k;if(d>1)for(i=new Array(d),j=new Array(d),k=new Array(d);d>b;b++)c[b]&&n.isFunction(c[b].promise)?c[b].promise().progress(h(b,j,i)).done(h(b,k,c)).fail(g.reject):--f;return f||g.resolveWith(k,c),g.promise()}});var I;n.fn.ready=function(a){return n.ready.promise().done(a),this},n.extend({isReady:!1,readyWait:1,holdReady:function(a){a?n.readyWait++:n.ready(!0)},ready:function(a){(a===!0?--n.readyWait:n.isReady)||(n.isReady=!0,a!==!0&&--n.readyWait>0||(I.resolveWith(d,[n]),n.fn.triggerHandler&&(n(d).triggerHandler("ready"),n(d).off("ready"))))}});function J(){d.addEventListener?(d.removeEventListener("DOMContentLoaded",K),a.removeEventListener("load",K)):(d.detachEvent("onreadystatechange",K),a.detachEvent("onload",K))}function K(){(d.addEventListener||"load"===a.event.type||"complete"===d.readyState)&&(J(),n.ready())}n.ready.promise=function(b){if(!I)if(I=n.Deferred(),"complete"===d.readyState||"loading"!==d.readyState&&!d.documentElement.doScroll)a.setTimeout(n.ready);else if(d.addEventListener)d.addEventListener("DOMContentLoaded",K),a.addEventListener("load",K);else{d.attachEvent("onreadystatechange",K),a.attachEvent("onload",K);var c=!1;try{c=null==a.frameElement&&d.documentElement}catch(e){}c&&c.doScroll&&!function f(){if(!n.isReady){try{c.doScroll("left")}catch(b){return a.setTimeout(f,50)}J(),n.ready()}}()}return I.promise(b)},n.ready.promise();var L;for(L in n(l))break;l.ownFirst="0"===L,l.inlineBlockNeedsLayout=!1,n(function(){var a,b,c,e;c=d.getElementsByTagName("body")[0],c&&c.style&&(b=d.createElement("div"),e=d.createElement("div"),e.style.cssText="position:absolute;border:0;width:0;height:0;top:0;left:-9999px",c.appendChild(e).appendChild(b),"undefined"!=typeof b.style.zoom&&(b.style.cssText="display:inline;margin:0;border:0;padding:1px;width:1px;zoom:1",l.inlineBlockNeedsLayout=a=3===b.offsetWidth,a&&(c.style.zoom=1)),c.removeChild(e))}),function(){var a=d.createElement("div");l.deleteExpando=!0;try{delete a.test}catch(b){l.deleteExpando=!1}a=null}();var M=function(a){var b=n.noData[(a.nodeName+" ").toLowerCase()],c=+a.nodeType||1;return 1!==c&&9!==c?!1:!b||b!==!0&&a.getAttribute("classid")===b},N=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,O=/([A-Z])/g;function P(a,b,c){if(void 0===c&&1===a.nodeType){var d="data-"+b.replace(O,"-$1").toLowerCase();if(c=a.getAttribute(d),"string"==typeof c){try{c="true"===c?!0:"false"===c?!1:"null"===c?null:+c+""===c?+c:N.test(c)?n.parseJSON(c):c}catch(e){}n.data(a,b,c)}else c=void 0; 3 | }return c}function Q(a){var b;for(b in a)if(("data"!==b||!n.isEmptyObject(a[b]))&&"toJSON"!==b)return!1;return!0}function R(a,b,d,e){if(M(a)){var f,g,h=n.expando,i=a.nodeType,j=i?n.cache:a,k=i?a[h]:a[h]&&h;if(k&&j[k]&&(e||j[k].data)||void 0!==d||"string"!=typeof b)return k||(k=i?a[h]=c.pop()||n.guid++:h),j[k]||(j[k]=i?{}:{toJSON:n.noop}),"object"!=typeof b&&"function"!=typeof b||(e?j[k]=n.extend(j[k],b):j[k].data=n.extend(j[k].data,b)),g=j[k],e||(g.data||(g.data={}),g=g.data),void 0!==d&&(g[n.camelCase(b)]=d),"string"==typeof b?(f=g[b],null==f&&(f=g[n.camelCase(b)])):f=g,f}}function S(a,b,c){if(M(a)){var d,e,f=a.nodeType,g=f?n.cache:a,h=f?a[n.expando]:n.expando;if(g[h]){if(b&&(d=c?g[h]:g[h].data)){n.isArray(b)?b=b.concat(n.map(b,n.camelCase)):b in d?b=[b]:(b=n.camelCase(b),b=b in d?[b]:b.split(" ")),e=b.length;while(e--)delete d[b[e]];if(c?!Q(d):!n.isEmptyObject(d))return}(c||(delete g[h].data,Q(g[h])))&&(f?n.cleanData([a],!0):l.deleteExpando||g!=g.window?delete g[h]:g[h]=void 0)}}}n.extend({cache:{},noData:{"applet ":!0,"embed ":!0,"object ":"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"},hasData:function(a){return a=a.nodeType?n.cache[a[n.expando]]:a[n.expando],!!a&&!Q(a)},data:function(a,b,c){return R(a,b,c)},removeData:function(a,b){return S(a,b)},_data:function(a,b,c){return R(a,b,c,!0)},_removeData:function(a,b){return S(a,b,!0)}}),n.fn.extend({data:function(a,b){var c,d,e,f=this[0],g=f&&f.attributes;if(void 0===a){if(this.length&&(e=n.data(f),1===f.nodeType&&!n._data(f,"parsedAttrs"))){c=g.length;while(c--)g[c]&&(d=g[c].name,0===d.indexOf("data-")&&(d=n.camelCase(d.slice(5)),P(f,d,e[d])));n._data(f,"parsedAttrs",!0)}return e}return"object"==typeof a?this.each(function(){n.data(this,a)}):arguments.length>1?this.each(function(){n.data(this,a,b)}):f?P(f,a,n.data(f,a)):void 0},removeData:function(a){return this.each(function(){n.removeData(this,a)})}}),n.extend({queue:function(a,b,c){var d;return a?(b=(b||"fx")+"queue",d=n._data(a,b),c&&(!d||n.isArray(c)?d=n._data(a,b,n.makeArray(c)):d.push(c)),d||[]):void 0},dequeue:function(a,b){b=b||"fx";var c=n.queue(a,b),d=c.length,e=c.shift(),f=n._queueHooks(a,b),g=function(){n.dequeue(a,b)};"inprogress"===e&&(e=c.shift(),d--),e&&("fx"===b&&c.unshift("inprogress"),delete f.stop,e.call(a,g,f)),!d&&f&&f.empty.fire()},_queueHooks:function(a,b){var c=b+"queueHooks";return n._data(a,c)||n._data(a,c,{empty:n.Callbacks("once memory").add(function(){n._removeData(a,b+"queue"),n._removeData(a,c)})})}}),n.fn.extend({queue:function(a,b){var c=2;return"string"!=typeof a&&(b=a,a="fx",c--),arguments.lengthh;h++)b(a[h],c,g?d:d.call(a[h],h,b(a[h],c)));return e?a:j?b.call(a):i?b(a[0],c):f},Z=/^(?:checkbox|radio)$/i,$=/<([\w:-]+)/,_=/^$|\/(?:java|ecma)script/i,aa=/^\s+/,ba="abbr|article|aside|audio|bdi|canvas|data|datalist|details|dialog|figcaption|figure|footer|header|hgroup|main|mark|meter|nav|output|picture|progress|section|summary|template|time|video";function ca(a){var b=ba.split("|"),c=a.createDocumentFragment();if(c.createElement)while(b.length)c.createElement(b.pop());return c}!function(){var a=d.createElement("div"),b=d.createDocumentFragment(),c=d.createElement("input");a.innerHTML="
a",l.leadingWhitespace=3===a.firstChild.nodeType,l.tbody=!a.getElementsByTagName("tbody").length,l.htmlSerialize=!!a.getElementsByTagName("link").length,l.html5Clone="<:nav>"!==d.createElement("nav").cloneNode(!0).outerHTML,c.type="checkbox",c.checked=!0,b.appendChild(c),l.appendChecked=c.checked,a.innerHTML="",l.noCloneChecked=!!a.cloneNode(!0).lastChild.defaultValue,b.appendChild(a),c=d.createElement("input"),c.setAttribute("type","radio"),c.setAttribute("checked","checked"),c.setAttribute("name","t"),a.appendChild(c),l.checkClone=a.cloneNode(!0).cloneNode(!0).lastChild.checked,l.noCloneEvent=!!a.addEventListener,a[n.expando]=1,l.attributes=!a.getAttribute(n.expando)}();var da={option:[1,""],legend:[1,"
","
"],area:[1,"",""],param:[1,"",""],thead:[1,"","
"],tr:[2,"","
"],col:[2,"","
"],td:[3,"","
"],_default:l.htmlSerialize?[0,"",""]:[1,"X
","
"]};da.optgroup=da.option,da.tbody=da.tfoot=da.colgroup=da.caption=da.thead,da.th=da.td;function ea(a,b){var c,d,e=0,f="undefined"!=typeof a.getElementsByTagName?a.getElementsByTagName(b||"*"):"undefined"!=typeof a.querySelectorAll?a.querySelectorAll(b||"*"):void 0;if(!f)for(f=[],c=a.childNodes||a;null!=(d=c[e]);e++)!b||n.nodeName(d,b)?f.push(d):n.merge(f,ea(d,b));return void 0===b||b&&n.nodeName(a,b)?n.merge([a],f):f}function fa(a,b){for(var c,d=0;null!=(c=a[d]);d++)n._data(c,"globalEval",!b||n._data(b[d],"globalEval"))}var ga=/<|&#?\w+;/,ha=/r;r++)if(g=a[r],g||0===g)if("object"===n.type(g))n.merge(q,g.nodeType?[g]:g);else if(ga.test(g)){i=i||p.appendChild(b.createElement("div")),j=($.exec(g)||["",""])[1].toLowerCase(),m=da[j]||da._default,i.innerHTML=m[1]+n.htmlPrefilter(g)+m[2],f=m[0];while(f--)i=i.lastChild;if(!l.leadingWhitespace&&aa.test(g)&&q.push(b.createTextNode(aa.exec(g)[0])),!l.tbody){g="table"!==j||ha.test(g)?""!==m[1]||ha.test(g)?0:i:i.firstChild,f=g&&g.childNodes.length;while(f--)n.nodeName(k=g.childNodes[f],"tbody")&&!k.childNodes.length&&g.removeChild(k)}n.merge(q,i.childNodes),i.textContent="";while(i.firstChild)i.removeChild(i.firstChild);i=p.lastChild}else q.push(b.createTextNode(g));i&&p.removeChild(i),l.appendChecked||n.grep(ea(q,"input"),ia),r=0;while(g=q[r++])if(d&&n.inArray(g,d)>-1)e&&e.push(g);else if(h=n.contains(g.ownerDocument,g),i=ea(p.appendChild(g),"script"),h&&fa(i),c){f=0;while(g=i[f++])_.test(g.type||"")&&c.push(g)}return i=null,p}!function(){var b,c,e=d.createElement("div");for(b in{submit:!0,change:!0,focusin:!0})c="on"+b,(l[b]=c in a)||(e.setAttribute(c,"t"),l[b]=e.attributes[c].expando===!1);e=null}();var ka=/^(?:input|select|textarea)$/i,la=/^key/,ma=/^(?:mouse|pointer|contextmenu|drag|drop)|click/,na=/^(?:focusinfocus|focusoutblur)$/,oa=/^([^.]*)(?:\.(.+)|)/;function pa(){return!0}function qa(){return!1}function ra(){try{return d.activeElement}catch(a){}}function sa(a,b,c,d,e,f){var g,h;if("object"==typeof b){"string"!=typeof c&&(d=d||c,c=void 0);for(h in b)sa(a,h,c,d,b[h],f);return a}if(null==d&&null==e?(e=c,d=c=void 0):null==e&&("string"==typeof c?(e=d,d=void 0):(e=d,d=c,c=void 0)),e===!1)e=qa;else if(!e)return a;return 1===f&&(g=e,e=function(a){return n().off(a),g.apply(this,arguments)},e.guid=g.guid||(g.guid=n.guid++)),a.each(function(){n.event.add(this,b,e,d,c)})}n.event={global:{},add:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=n._data(a);if(r){c.handler&&(i=c,c=i.handler,e=i.selector),c.guid||(c.guid=n.guid++),(g=r.events)||(g=r.events={}),(k=r.handle)||(k=r.handle=function(a){return"undefined"==typeof n||a&&n.event.triggered===a.type?void 0:n.event.dispatch.apply(k.elem,arguments)},k.elem=a),b=(b||"").match(G)||[""],h=b.length;while(h--)f=oa.exec(b[h])||[],o=q=f[1],p=(f[2]||"").split(".").sort(),o&&(j=n.event.special[o]||{},o=(e?j.delegateType:j.bindType)||o,j=n.event.special[o]||{},l=n.extend({type:o,origType:q,data:d,handler:c,guid:c.guid,selector:e,needsContext:e&&n.expr.match.needsContext.test(e),namespace:p.join(".")},i),(m=g[o])||(m=g[o]=[],m.delegateCount=0,j.setup&&j.setup.call(a,d,p,k)!==!1||(a.addEventListener?a.addEventListener(o,k,!1):a.attachEvent&&a.attachEvent("on"+o,k))),j.add&&(j.add.call(a,l),l.handler.guid||(l.handler.guid=c.guid)),e?m.splice(m.delegateCount++,0,l):m.push(l),n.event.global[o]=!0);a=null}},remove:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=n.hasData(a)&&n._data(a);if(r&&(k=r.events)){b=(b||"").match(G)||[""],j=b.length;while(j--)if(h=oa.exec(b[j])||[],o=q=h[1],p=(h[2]||"").split(".").sort(),o){l=n.event.special[o]||{},o=(d?l.delegateType:l.bindType)||o,m=k[o]||[],h=h[2]&&new RegExp("(^|\\.)"+p.join("\\.(?:.*\\.|)")+"(\\.|$)"),i=f=m.length;while(f--)g=m[f],!e&&q!==g.origType||c&&c.guid!==g.guid||h&&!h.test(g.namespace)||d&&d!==g.selector&&("**"!==d||!g.selector)||(m.splice(f,1),g.selector&&m.delegateCount--,l.remove&&l.remove.call(a,g));i&&!m.length&&(l.teardown&&l.teardown.call(a,p,r.handle)!==!1||n.removeEvent(a,o,r.handle),delete k[o])}else for(o in k)n.event.remove(a,o+b[j],c,d,!0);n.isEmptyObject(k)&&(delete r.handle,n._removeData(a,"events"))}},trigger:function(b,c,e,f){var g,h,i,j,l,m,o,p=[e||d],q=k.call(b,"type")?b.type:b,r=k.call(b,"namespace")?b.namespace.split("."):[];if(i=m=e=e||d,3!==e.nodeType&&8!==e.nodeType&&!na.test(q+n.event.triggered)&&(q.indexOf(".")>-1&&(r=q.split("."),q=r.shift(),r.sort()),h=q.indexOf(":")<0&&"on"+q,b=b[n.expando]?b:new n.Event(q,"object"==typeof b&&b),b.isTrigger=f?2:3,b.namespace=r.join("."),b.rnamespace=b.namespace?new RegExp("(^|\\.)"+r.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,b.result=void 0,b.target||(b.target=e),c=null==c?[b]:n.makeArray(c,[b]),l=n.event.special[q]||{},f||!l.trigger||l.trigger.apply(e,c)!==!1)){if(!f&&!l.noBubble&&!n.isWindow(e)){for(j=l.delegateType||q,na.test(j+q)||(i=i.parentNode);i;i=i.parentNode)p.push(i),m=i;m===(e.ownerDocument||d)&&p.push(m.defaultView||m.parentWindow||a)}o=0;while((i=p[o++])&&!b.isPropagationStopped())b.type=o>1?j:l.bindType||q,g=(n._data(i,"events")||{})[b.type]&&n._data(i,"handle"),g&&g.apply(i,c),g=h&&i[h],g&&g.apply&&M(i)&&(b.result=g.apply(i,c),b.result===!1&&b.preventDefault());if(b.type=q,!f&&!b.isDefaultPrevented()&&(!l._default||l._default.apply(p.pop(),c)===!1)&&M(e)&&h&&e[q]&&!n.isWindow(e)){m=e[h],m&&(e[h]=null),n.event.triggered=q;try{e[q]()}catch(s){}n.event.triggered=void 0,m&&(e[h]=m)}return b.result}},dispatch:function(a){a=n.event.fix(a);var b,c,d,f,g,h=[],i=e.call(arguments),j=(n._data(this,"events")||{})[a.type]||[],k=n.event.special[a.type]||{};if(i[0]=a,a.delegateTarget=this,!k.preDispatch||k.preDispatch.call(this,a)!==!1){h=n.event.handlers.call(this,a,j),b=0;while((f=h[b++])&&!a.isPropagationStopped()){a.currentTarget=f.elem,c=0;while((g=f.handlers[c++])&&!a.isImmediatePropagationStopped())a.rnamespace&&!a.rnamespace.test(g.namespace)||(a.handleObj=g,a.data=g.data,d=((n.event.special[g.origType]||{}).handle||g.handler).apply(f.elem,i),void 0!==d&&(a.result=d)===!1&&(a.preventDefault(),a.stopPropagation()))}return k.postDispatch&&k.postDispatch.call(this,a),a.result}},handlers:function(a,b){var c,d,e,f,g=[],h=b.delegateCount,i=a.target;if(h&&i.nodeType&&("click"!==a.type||isNaN(a.button)||a.button<1))for(;i!=this;i=i.parentNode||this)if(1===i.nodeType&&(i.disabled!==!0||"click"!==a.type)){for(d=[],c=0;h>c;c++)f=b[c],e=f.selector+" ",void 0===d[e]&&(d[e]=f.needsContext?n(e,this).index(i)>-1:n.find(e,this,null,[i]).length),d[e]&&d.push(f);d.length&&g.push({elem:i,handlers:d})}return h]","i"),va=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:-]+)[^>]*)\/>/gi,wa=/\s*$/g,Aa=ca(d),Ba=Aa.appendChild(d.createElement("div"));function Ca(a,b){return n.nodeName(a,"table")&&n.nodeName(11!==b.nodeType?b:b.firstChild,"tr")?a.getElementsByTagName("tbody")[0]||a.appendChild(a.ownerDocument.createElement("tbody")):a}function Da(a){return a.type=(null!==n.find.attr(a,"type"))+"/"+a.type,a}function Ea(a){var b=ya.exec(a.type);return b?a.type=b[1]:a.removeAttribute("type"),a}function Fa(a,b){if(1===b.nodeType&&n.hasData(a)){var c,d,e,f=n._data(a),g=n._data(b,f),h=f.events;if(h){delete g.handle,g.events={};for(c in h)for(d=0,e=h[c].length;e>d;d++)n.event.add(b,c,h[c][d])}g.data&&(g.data=n.extend({},g.data))}}function Ga(a,b){var c,d,e;if(1===b.nodeType){if(c=b.nodeName.toLowerCase(),!l.noCloneEvent&&b[n.expando]){e=n._data(b);for(d in e.events)n.removeEvent(b,d,e.handle);b.removeAttribute(n.expando)}"script"===c&&b.text!==a.text?(Da(b).text=a.text,Ea(b)):"object"===c?(b.parentNode&&(b.outerHTML=a.outerHTML),l.html5Clone&&a.innerHTML&&!n.trim(b.innerHTML)&&(b.innerHTML=a.innerHTML)):"input"===c&&Z.test(a.type)?(b.defaultChecked=b.checked=a.checked,b.value!==a.value&&(b.value=a.value)):"option"===c?b.defaultSelected=b.selected=a.defaultSelected:"input"!==c&&"textarea"!==c||(b.defaultValue=a.defaultValue)}}function Ha(a,b,c,d){b=f.apply([],b);var e,g,h,i,j,k,m=0,o=a.length,p=o-1,q=b[0],r=n.isFunction(q);if(r||o>1&&"string"==typeof q&&!l.checkClone&&xa.test(q))return a.each(function(e){var f=a.eq(e);r&&(b[0]=q.call(this,e,f.html())),Ha(f,b,c,d)});if(o&&(k=ja(b,a[0].ownerDocument,!1,a,d),e=k.firstChild,1===k.childNodes.length&&(k=e),e||d)){for(i=n.map(ea(k,"script"),Da),h=i.length;o>m;m++)g=k,m!==p&&(g=n.clone(g,!0,!0),h&&n.merge(i,ea(g,"script"))),c.call(a[m],g,m);if(h)for(j=i[i.length-1].ownerDocument,n.map(i,Ea),m=0;h>m;m++)g=i[m],_.test(g.type||"")&&!n._data(g,"globalEval")&&n.contains(j,g)&&(g.src?n._evalUrl&&n._evalUrl(g.src):n.globalEval((g.text||g.textContent||g.innerHTML||"").replace(za,"")));k=e=null}return a}function Ia(a,b,c){for(var d,e=b?n.filter(b,a):a,f=0;null!=(d=e[f]);f++)c||1!==d.nodeType||n.cleanData(ea(d)),d.parentNode&&(c&&n.contains(d.ownerDocument,d)&&fa(ea(d,"script")),d.parentNode.removeChild(d));return a}n.extend({htmlPrefilter:function(a){return a.replace(va,"<$1>")},clone:function(a,b,c){var d,e,f,g,h,i=n.contains(a.ownerDocument,a);if(l.html5Clone||n.isXMLDoc(a)||!ua.test("<"+a.nodeName+">")?f=a.cloneNode(!0):(Ba.innerHTML=a.outerHTML,Ba.removeChild(f=Ba.firstChild)),!(l.noCloneEvent&&l.noCloneChecked||1!==a.nodeType&&11!==a.nodeType||n.isXMLDoc(a)))for(d=ea(f),h=ea(a),g=0;null!=(e=h[g]);++g)d[g]&&Ga(e,d[g]);if(b)if(c)for(h=h||ea(a),d=d||ea(f),g=0;null!=(e=h[g]);g++)Fa(e,d[g]);else Fa(a,f);return d=ea(f,"script"),d.length>0&&fa(d,!i&&ea(a,"script")),d=h=e=null,f},cleanData:function(a,b){for(var d,e,f,g,h=0,i=n.expando,j=n.cache,k=l.attributes,m=n.event.special;null!=(d=a[h]);h++)if((b||M(d))&&(f=d[i],g=f&&j[f])){if(g.events)for(e in g.events)m[e]?n.event.remove(d,e):n.removeEvent(d,e,g.handle);j[f]&&(delete j[f],k||"undefined"==typeof d.removeAttribute?d[i]=void 0:d.removeAttribute(i),c.push(f))}}}),n.fn.extend({domManip:Ha,detach:function(a){return Ia(this,a,!0)},remove:function(a){return Ia(this,a)},text:function(a){return Y(this,function(a){return void 0===a?n.text(this):this.empty().append((this[0]&&this[0].ownerDocument||d).createTextNode(a))},null,a,arguments.length)},append:function(){return Ha(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=Ca(this,a);b.appendChild(a)}})},prepend:function(){return Ha(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=Ca(this,a);b.insertBefore(a,b.firstChild)}})},before:function(){return Ha(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this)})},after:function(){return Ha(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this.nextSibling)})},empty:function(){for(var a,b=0;null!=(a=this[b]);b++){1===a.nodeType&&n.cleanData(ea(a,!1));while(a.firstChild)a.removeChild(a.firstChild);a.options&&n.nodeName(a,"select")&&(a.options.length=0)}return this},clone:function(a,b){return a=null==a?!1:a,b=null==b?a:b,this.map(function(){return n.clone(this,a,b)})},html:function(a){return Y(this,function(a){var b=this[0]||{},c=0,d=this.length;if(void 0===a)return 1===b.nodeType?b.innerHTML.replace(ta,""):void 0;if("string"==typeof a&&!wa.test(a)&&(l.htmlSerialize||!ua.test(a))&&(l.leadingWhitespace||!aa.test(a))&&!da[($.exec(a)||["",""])[1].toLowerCase()]){a=n.htmlPrefilter(a);try{for(;d>c;c++)b=this[c]||{},1===b.nodeType&&(n.cleanData(ea(b,!1)),b.innerHTML=a);b=0}catch(e){}}b&&this.empty().append(a)},null,a,arguments.length)},replaceWith:function(){var a=[];return Ha(this,arguments,function(b){var c=this.parentNode;n.inArray(this,a)<0&&(n.cleanData(ea(this)),c&&c.replaceChild(b,this))},a)}}),n.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){n.fn[a]=function(a){for(var c,d=0,e=[],f=n(a),h=f.length-1;h>=d;d++)c=d===h?this:this.clone(!0),n(f[d])[b](c),g.apply(e,c.get());return this.pushStack(e)}});var Ja,Ka={HTML:"block",BODY:"block"};function La(a,b){var c=n(b.createElement(a)).appendTo(b.body),d=n.css(c[0],"display");return c.detach(),d}function Ma(a){var b=d,c=Ka[a];return c||(c=La(a,b),"none"!==c&&c||(Ja=(Ja||n(" 67 | 68 | 69 | 70 | 71 | 72 |
73 |
74 |
75 |
76 |
78 |
乐视TV视频
79 |
80 |
81 |
83 |
腾讯视频
84 |
85 |
86 |
88 |
爱奇艺视频
89 |
90 |
91 |
93 |
优酷视频
94 |
95 |
96 |
98 |
土豆视频
99 |
100 |
101 |
103 |
芒果TV视频
104 |
105 |
106 |
108 |
搜狐视频
109 |
110 |
111 |
112 |
113 |
优酷云C
114 |
115 |
116 |
118 |
Ac弹幕网
119 |
120 |
121 |
123 |
124 |
哔哩哔哩
125 |
126 |
127 |
129 |
130 |
风行网
131 |
132 |
133 |
135 |
WASU华数视频
136 |
137 |
138 |
140 |
56
141 |
142 |
143 |
145 |
音悦台MV
146 |
147 |
148 |
149 |
150 |
151 | 154 | 155 |
156 |
157 | 158 |
159 | 166 |
167 |
168 | 169 | 170 | 171 | -------------------------------------------------------------------------------- /app/templates/log_list.html: -------------------------------------------------------------------------------- 1 | 2 | Download 3 |

Directory listing

4 |
5 |
    6 | {% for key,value in files.items():%} 7 |
  • {{key}}
  • 8 | {% endfor %} 9 |
-------------------------------------------------------------------------------- /app/third_party/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/app/third_party/__init__.py -------------------------------------------------------------------------------- /app/third_party/movie.py: -------------------------------------------------------------------------------- 1 | import json 2 | import re 3 | from urllib.parse import urljoin 4 | from lxml import etree 5 | from ..log import Log 6 | from ..util.requests_helper import RequestsHelper 7 | from ..models.website_navigation import WebsiteNavigation 8 | 9 | 10 | class Movie(object): 11 | def __init__(self, thumbnail, url, title, director, stars, type, introduction): 12 | self.thumbnail = thumbnail 13 | self.url = url 14 | self.title = title 15 | self.director = director 16 | self.stars = stars 17 | self.type = type 18 | self.introduction = introduction 19 | 20 | # 告诉 Python 如何打印这个类的对象。我们将用它来调试 21 | def __repr__(self): 22 | return 'thumbnail:{}\n url:{}\n title:{}\n director:{}\n stars:{}\n type:{}\n introduction:{}\n'.format(self.thumbnail, 23 | self.url, self.title, self.director, self.stars, self.type, self.introduction) 24 | 25 | @classmethod 26 | def search(cls, key): 27 | try: 28 | websites = WebsiteNavigation.get_rand_online_movie_website(5) 29 | search_results = [] 30 | for website in websites: 31 | movies = cls.search_from_json(key, website.search_json) 32 | # 相对路径转绝对路径 33 | for movie in movies: 34 | if not (movie.url.startswith('http:') or movie.url.startswith('https:')): 35 | movie.url = urljoin(website.url, movie.url) 36 | print('search movie:', movie.url) 37 | 38 | search_results += movies 39 | 40 | # 大于5个结果就不再继续找了 41 | if len(search_results) > 5: 42 | break 43 | 44 | return search_results 45 | except Exception as e: 46 | Log.logger().exception(e) 47 | return [] 48 | 49 | @classmethod 50 | def parse_movie_item(cls, result, search_json_obj, key): 51 | if key + '_xpath' not in search_json_obj['movie_item'].keys(): 52 | return '' 53 | 54 | xpath = search_json_obj['movie_item'][key + '_xpath'] 55 | if len(xpath) == 0: 56 | return '' 57 | item = result.xpath(xpath) 58 | if isinstance(item, list): 59 | if len(item) > 0: 60 | item = item[0] 61 | else: 62 | item = "" 63 | item = ''.join(item).strip() 64 | 65 | if key + '_re' in search_json_obj['movie_item'].keys(): 66 | xpath = search_json_obj['movie_item'][key + '_re'] 67 | if len(xpath) != 0: 68 | pattern = re.compile(xpath) 69 | find = pattern.findall(item) 70 | item = find[0] if len(find) != 0 else '' 71 | 72 | return item 73 | 74 | @classmethod 75 | def search_from_json(cls, key, search_json): 76 | print('search_json:\n', search_json) 77 | search_json_obj = json.loads(search_json) 78 | html = RequestsHelper.download_html(search_json_obj['url'].format(key)) 79 | if len(html) == 0: 80 | Log.logger().info('download_html failed:{}'.format( 81 | search_json_obj['url'].format(key))) 82 | return [] 83 | 84 | et_html = etree.HTML(html) 85 | items = et_html.xpath(search_json_obj['movie_item_xpath']) 86 | 87 | ret_results = [] 88 | print('movie item len:', len(items)) 89 | for result in items: 90 | thumbnail = cls.parse_movie_item( 91 | result, search_json_obj, 'thumbnail') 92 | url = cls.parse_movie_item(result, search_json_obj, 'url') 93 | title = cls.parse_movie_item(result, search_json_obj, 'title') 94 | director = cls.parse_movie_item( 95 | result, search_json_obj, 'director') 96 | stars = cls.parse_movie_item(result, search_json_obj, 'stars') 97 | type = cls.parse_movie_item(result, search_json_obj, 'type') 98 | introduction = cls.parse_movie_item( 99 | result, search_json_obj, 'introduction') 100 | 101 | ret_results.append(Movie(thumbnail, url, title, 102 | director, stars, type, introduction)) 103 | 104 | return ret_results 105 | 106 | 107 | if __name__ == "__main__": 108 | if True: 109 | search_json_string = ('{"url": "https://www.ahrmgg.com/ppyssearch.html?wd={}",' 110 | '"movie_item_xpath": "//ul[@class=\'stui-vodlist__media col-pd clearfix\']/li",' 111 | '"movie_item": {' 112 | '"thumbnail_xpath": "./div[@class=\'thumb\']/a/@data-original", ' 113 | '"url_xpath":"./div[@class=\'detail\']/h3/a/@href", ' 114 | '"title_xpath":"./div[@class=\'detail\']/h3/a/text()",' 115 | '"director_xpath":"./div[@class=\'detail\']/p[1]/text()",' 116 | '"stars_xpath":"./div[@class=\'detail\']/p[2]/text()",' 117 | '"type_xpath":"./div[@class=\'detail\']/p[3]/text()"' 118 | '}' 119 | '}') 120 | 121 | print(Movie.search_from_json('恶人', search_json_string)) 122 | -------------------------------------------------------------------------------- /app/third_party/ocr.py: -------------------------------------------------------------------------------- 1 | # -*- coding: utf-8 -* 2 | 3 | from aip import AipOcr 4 | from flask import current_app 5 | 6 | 7 | def ocr_aip_url(url): 8 | try: 9 | client = AipOcr( 10 | current_app.config['OCR_APP_ID'], current_app.config['OCR_API_KEY'], current_app.config['OCR_SECRET_KEY']) 11 | 12 | # https://ai.baidu.com/ai-doc/OCR/7kibizyfm 13 | options = {} 14 | options["language_type"] = "CHN_ENG" 15 | #options["detect_direction"] = "true" 16 | #options["detect_language"] = "true" 17 | #options["probability"] = "true" 18 | 19 | resp = client.basicGeneralUrl(url, options) 20 | 21 | if 'error_msg' in resp.keys(): 22 | return u'图片识别失败: {}'.format(resp["error_msg"]) 23 | 24 | result = '' 25 | for item in resp['words_result']: 26 | if result != '': 27 | result = result + '\n' 28 | result = result + item['words'] 29 | 30 | return result 31 | except Exception as e: 32 | return u'图片识别异常: {}'.format(str(e)) 33 | 34 | 35 | def get_file_content(filePath): 36 | with open(filePath, 'rb') as fp: 37 | return fp.read() 38 | 39 | 40 | def ocr_aip_path(path): 41 | try: 42 | client = AipOcr(APP_ID, API_KEY, SECRET_KEY) 43 | 44 | # https://ai.baidu.com/ai-doc/OCR/7kibizyfm 45 | options = {} 46 | options["language_type"] = "CHN_ENG" 47 | #options["detect_direction"] = "true" 48 | #options["detect_language"] = "true" 49 | #options["probability"] = "true" 50 | 51 | image = get_file_content(path) 52 | resp = client.basicGeneral(image, options) 53 | 54 | if 'error_msg' in resp.keys(): 55 | return u'图片识别失败: {}'.format(resp["error_msg"]) 56 | 57 | result = '' 58 | for item in resp['words_result']: 59 | if result != '': 60 | result = result + '\n' 61 | result = result + item['words'] 62 | 63 | return result 64 | except Exception as e: 65 | return u'图片识别异常: {}'.format(str(e)) 66 | 67 | # print ocr_aip_url('https://cdn.learnku.com/uploads/images/202008/20/51094/DiV8QB39oJ.jpeg!large') 68 | -------------------------------------------------------------------------------- /app/util/__init__.py: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /app/util/requests_helper.py: -------------------------------------------------------------------------------- 1 | import requests 2 | 3 | class RequestsHelper(object): 4 | 5 | @classmethod 6 | def download_html(cls, url): 7 | try: 8 | header = { 9 | 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/67.0.3396.99 Safari/537.36' 10 | } 11 | r = requests.get(url, headers=header) 12 | return r.text 13 | except Exception as e: 14 | print('Failed downloading', url) 15 | print(e) 16 | return '' 17 | 18 | if __name__ == "__main__": 19 | print(RequestsHelper.download_html('https://neihandianying.com/movie/search?q=蜜桃成熟时')) -------------------------------------------------------------------------------- /docker-compose.yaml: -------------------------------------------------------------------------------- 1 | version: "3.8" 2 | services: 3 | werobot: 4 | image: werobot:latest 5 | build: . 6 | container_name: werobot 7 | restart: always 8 | ports: 9 | - "80:80" 10 | volumes: 11 | - ./app:/deploy/app -------------------------------------------------------------------------------- /docs/image/coderbox.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/barry-ran/werobot/7bfb5690449ef730daece75445e8a4aa236b3052/docs/image/coderbox.jpg -------------------------------------------------------------------------------- /gunicorn.conf.py: -------------------------------------------------------------------------------- 1 | import multiprocessing 2 | debug = False 3 | bind = "0.0.0.0:5000" 4 | pidfile = "gunicorn.pid" 5 | workers = multiprocessing.cpu_count()*2 + 1 6 | worker_class = "gevent" 7 | # daemon=True 在docker中不需要daemon运行,反而会导致看不到gunicorn输出而增加排查问题的难度 -------------------------------------------------------------------------------- /nginx_flask.conf: -------------------------------------------------------------------------------- 1 | server { 2 | listen 80; 3 | server_name localhost; 4 | 5 | location / { 6 | proxy_pass http://0.0.0.0:5000; # 这里是指向 gunicorn host 的服务地址 7 | proxy_set_header Host $host; 8 | proxy_set_header X-Real-IP $remote_addr; 9 | proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; 10 | } 11 | } -------------------------------------------------------------------------------- /requirements.txt: -------------------------------------------------------------------------------- 1 | alembic==1.6.5 2 | APScheduler==3.7.0 3 | autopep8==1.5.7 4 | baidu-aip==2.2.18.0 5 | bottle==0.12.19 6 | certifi==2021.5.30 7 | charset-normalizer==2.0.4 8 | click==8.0.1 9 | colorama==0.4.4 10 | Flask==2.0.1 11 | Flask-APScheduler==1.12.2 12 | Flask-Executor==0.9.4 13 | Flask-Migrate==3.1.0 14 | Flask-SQLAlchemy==2.5.1 15 | greenlet==1.1.0 16 | idna==3.2 17 | importlib-metadata==4.6.3 18 | itsdangerous==2.0.1 19 | Jinja2==3.0.1 20 | lxml==4.6.3 21 | Mako==1.1.4 22 | MarkupSafe==2.0.1 23 | pycodestyle==2.7.0 24 | PyMySQL==1.0.2 25 | python-dateutil==2.8.2 26 | python-dotenv==0.19.0 27 | python-editor==1.0.4 28 | pytz==2021.1 29 | requests==2.26.0 30 | six==1.16.0 31 | SQLAlchemy==1.4.22 32 | toml==0.10.2 33 | typing-extensions==3.10.0.0 34 | tzlocal==2.1 35 | urllib3==1.26.6 36 | Werkzeug==2.0.1 37 | WeRoBot==1.13.1 38 | xmltodict==0.12.0 39 | zipp==3.5.0 40 | -------------------------------------------------------------------------------- /script/compose-deploy.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | docker-compose down 3 | # 不带-d,调试用 4 | # docker-compose up --build 5 | docker-compose up -d --build 6 | -------------------------------------------------------------------------------- /script/docker-deploy.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | docker container stop werobot 3 | docker container rm werobot 4 | docker image rm werobot 5 | 6 | docker build -t werobot:latest . 7 | # -v 命令必须用绝对路径,而compose配置可以用相对路径 8 | # 调试用:前台运行 9 | #docker run -p 80:80 --name=werobot -v $(pwd)/app:/deploy/app werobot 10 | # 调试用:后台运行并且不退出 11 | #docker run -dit -p 80:80 --name=werobot -v $(pwd)/app:/deploy/app werobot 12 | docker run -d -p 80:80 --name=werobot -v $(pwd)/app:/deploy/app werobot 13 | -------------------------------------------------------------------------------- /script/run.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | # 先尝试迁移数据库 3 | flask db init -d ./app/migrations 4 | flask db migrate -d ./app/migrations 5 | flask db upgrade -d ./app/migrations 6 | # 再启动supervisord 7 | /usr/bin/supervisord 8 | -------------------------------------------------------------------------------- /supervisord.conf: -------------------------------------------------------------------------------- 1 | [supervisord] 2 | nodaemon=true 3 | 4 | [program:nginx] 5 | command=/usr/sbin/nginx 6 | 7 | [program:gunicorn] 8 | command=gunicorn "app:create_app('production')" -c /deploy/gunicorn.conf.py 9 | directory=/deploy --------------------------------------------------------------------------------