You've already forked amazing-ytdlp-archive
							
							Compare commits
	
		
			34 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 570ac88b99 | ||
|   | c51a72ec2b | ||
|   | fa8f11dad6 | ||
|   | 46e5d8bb02 | ||
|   | 89ce9b1c0a | ||
|   | 729b24debb | ||
|   | 20e5793cd8 | ||
|   | 282b895170 | ||
|   | 38f6f04260 | ||
|   | 43e6c00787 | ||
|   | d42030dcbc | ||
|   | 5530179558 | ||
|   | 1186d236f2 | ||
|   | 5a4726ac10 | ||
|   | 46bde82d32 | ||
|   | 6c681d6b07 | ||
|   | 0d5d233e90 | ||
|   | 548a4860fc | ||
|   | da333ab4f6 | ||
|   | f2b01033ea | ||
|   | 49f0ea7481 | ||
|   | f1287a4212 | ||
|   | 30ea647ca9 | ||
|   | a7c640a8cf | ||
|   | f6da232164 | ||
|   | 1d5934275c | ||
|   | 72af6b6126 | ||
|   | 8bf8e08af3 | ||
|   | 236b56915b | ||
|   | ac0243a783 | ||
|   | bb78c97d52 | ||
|   | 7ccb827a9c | ||
|   | 9c0e4fb63c | ||
|   | 75d42ad3cd | 
| @@ -1,4 +1,4 @@ | |||||||
| name: Generate release | name: Generate docker image | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   release: |   release: | ||||||
| @@ -23,12 +23,3 @@ jobs: | |||||||
|       with: |       with: | ||||||
|         push: true |         push: true | ||||||
|         tags: git.ventilaar.nl/ventilaar/ayta:latest |         tags: git.ventilaar.nl/ventilaar/ayta:latest | ||||||
|          |  | ||||||
|     - name: Update worker server |  | ||||||
|       uses: appleboy/ssh-action@v1.0.3 |  | ||||||
|       with: |  | ||||||
|         host: 192.168.66.109 |  | ||||||
|         username: root |  | ||||||
|         key: ${{ secrets.SERVER_KEY }} |  | ||||||
|         port: 22 |  | ||||||
|         script: /root/update_worker.sh |  | ||||||
							
								
								
									
										18
									
								
								.gitea/workflows/workers-tasks.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.gitea/workflows/workers-tasks.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | name: Update worker server | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   release: | ||||||
|  |     types: [published] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build-and-publish: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - name: Update worker server | ||||||
|  |       uses: appleboy/ssh-action@v1.0.3 | ||||||
|  |       with: | ||||||
|  |         host: 192.168.66.109 | ||||||
|  |         username: root | ||||||
|  |         key: ${{ secrets.SERVER_KEY }} | ||||||
|  |         port: 22 | ||||||
|  |         script: /root/update_worker.sh | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| FROM python:3-alpine | FROM python:3.12-alpine | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| COPY requirements.txt /app | COPY requirements.txt /app | ||||||
| RUN pip install --no-cache-dir -r requirements.txt | RUN pip install --no-cache-dir -r requirements.txt | ||||||
| COPY . /app | COPY . /app | ||||||
| EXPOSE 8000 | EXPOSE 8000 | ||||||
| CMD ["gunicorn", "--bind", "0.0.0.0:8000", "ayta:create_app()"] | CMD ["gunicorn", "--bind", "0.0.0.0:8000", "--workers", "1", "ayta:create_app()"] | ||||||
							
								
								
									
										17
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								README.md
									
									
									
									
									
								
							| @@ -6,7 +6,7 @@ current cronjob yt-dlp archive service. | |||||||
| Partially inspired by [hobune](https://github.com/rebane2001/hobune). While that project is amazingby it's own, it's just not scaleable. | Partially inspired by [hobune](https://github.com/rebane2001/hobune). While that project is amazingby it's own, it's just not scaleable. | ||||||
|  |  | ||||||
| ## The idea | ## The idea | ||||||
| Having over 250k videos, scaling the current cronjob yt-dlp archive task is just really hard. Filetypes change, things get partially downloaded and such. | Having over 350k videos, scaling the current cronjob yt-dlp archive task is just really hard. Filetypes change, things get partially downloaded and such. | ||||||
| Partially yt-dlp is to blame because it's a package that needs to change all the time. But with this some changes are not accounted for. | Partially yt-dlp is to blame because it's a package that needs to change all the time. But with this some changes are not accounted for. | ||||||
| yt-dlp will still do the downloads. But a flask frontend will be developed to make all downloaded videos easily indexable. | yt-dlp will still do the downloads. But a flask frontend will be developed to make all downloaded videos easily indexable. | ||||||
| For it to be quick (unlike hobune) a database has to be implemented. This could get solved by a static site generator type of software, but that is not my choice. | For it to be quick (unlike hobune) a database has to be implemented. This could get solved by a static site generator type of software, but that is not my choice. | ||||||
| @@ -52,13 +52,22 @@ Extra functionality for further development of features. | |||||||
|  |  | ||||||
| ### Stage 3 | ### Stage 3 | ||||||
| Mainly focused on retiring the cronjob based scripts and moving it to celery based tasks | Mainly focused on retiring the cronjob based scripts and moving it to celery based tasks | ||||||
| - [ ] manage videos by ID's instead of per channel basis | - [x] manage videos by ID's instead of per channel basis | ||||||
| - [ ] download videos from queue | - [x] download videos from queue | ||||||
| - [ ] Manage websub callbacks | - [x] Manage websub callbacks | ||||||
|  | - [x] Implement yt-dlp proxy servers, as the VPN is blocked | ||||||
|  | - [x] Celery tasks based video downloading | ||||||
|  | - [x] Manage websub callbacks | ||||||
|  | - [x] Celery task queue views | ||||||
|  | - [x] More performant statistics | ||||||
|  | - [ ] Retire cronjobs | ||||||
|  | - [ ] Retire file based configurations | ||||||
|  |  | ||||||
| ### Stage 4 | ### Stage 4 | ||||||
| Mongodb finally has it's limitations. | Mongodb finally has it's limitations. | ||||||
| - [ ] Migrate to postgresql | - [ ] Migrate to postgresql | ||||||
|  | - [ ] Retire time based tasks like channel mirroring | ||||||
|  | - [ ] A more comprehensive statistics page, uploads per day, downloads per day and such | ||||||
|  |  | ||||||
| ### Stage ... | ### Stage ... | ||||||
| Since this is my flagship software which I have developed more features will be added. | Since this is my flagship software which I have developed more features will be added. | ||||||
|   | |||||||
| @@ -1,13 +0,0 @@ | |||||||
| #Import os Library |  | ||||||
| import os |  | ||||||
| import datetime |  | ||||||
| import json |  | ||||||
|  |  | ||||||
| def print_current_time(give=False): |  | ||||||
|     time = datetime.datetime.now().replace(microsecond=0) |  | ||||||
|     print(f'--- It is {time} ---') |  | ||||||
|     return time |  | ||||||
|  |  | ||||||
| with open('lockfile', 'w') as file: |  | ||||||
|         data = {'time': print_current_time(), 'PID': os.getpid()} |  | ||||||
|         file.write(json.dumps(data, default=str)) |  | ||||||
| @@ -9,13 +9,33 @@ def create_app(test_config=None): | |||||||
|     config = {'MONGO_CONNECTION': os.environ.get('AYTA_MONGOCONNECTION', 'mongodb://root:example@192.168.66.140:27017'), |     config = {'MONGO_CONNECTION': os.environ.get('AYTA_MONGOCONNECTION', 'mongodb://root:example@192.168.66.140:27017'), | ||||||
|               'OIDC_PROVIDER': os.environ.get('AYTA_OIDC_PROVIDER', 'https://auth.ventilaar.nl'), |               'OIDC_PROVIDER': os.environ.get('AYTA_OIDC_PROVIDER', 'https://auth.ventilaar.nl'), | ||||||
|               'OIDC_ID': os.environ.get('AYTA_OIDC_ID', 'ayta'), |               'OIDC_ID': os.environ.get('AYTA_OIDC_ID', 'ayta'), | ||||||
|               'CACHE_TYPE': os.environ.get('AYTA_CACHETYPE', 'SimpleCache'), |  | ||||||
|               'CACHE_DEFAULT_TIMEOUT': int(os.environ.get('AYTA_CACHETIMEOUT', 6)), |               'CACHE_DEFAULT_TIMEOUT': int(os.environ.get('AYTA_CACHETIMEOUT', 6)), | ||||||
|               'SECRET_KEY': os.environ.get('AYTA_SECRETKEY', secrets.token_hex(32)), |  | ||||||
|               'DEBUG': bool(os.environ.get('AYTA_DEBUG', False)), |               'DEBUG': bool(os.environ.get('AYTA_DEBUG', False)), | ||||||
|               'DOMAIN': os.environ.get('AYTA_DOMAIN', 'https://testing.mashallah.nl'), |               'DOMAIN': os.environ.get('AYTA_DOMAIN', 'https://testing.mashallah.nl'), | ||||||
|               'CELERY': dict(broker_url=str(os.environ.get('AYTA_CELERYBROKER', 'amqp://guest:guest@192.168.66.140:5672/')),  |               'CELERY': {'broker_url': str(os.environ.get('AYTA_CELERYBROKER', 'amqp://guest:guest@192.168.66.140:5672/'))} | ||||||
|                              task_ignore_result=True,) |              } | ||||||
|  |      | ||||||
|  |     # Static Flask configuration options | ||||||
|  |      | ||||||
|  |     config['CELERY']['task_ignore_result'] = True | ||||||
|  |     config['CACHE_TYPE'] = 'SimpleCache' | ||||||
|  |     config['SECRET_KEY'] = secrets.token_bytes(32) | ||||||
|  |      | ||||||
|  |     # Celery Periodic tasks | ||||||
|  |      | ||||||
|  |     config['CELERY']['beat_schedule'] = {} | ||||||
|  |     config['CELERY']['beat_schedule']['Renew WebSub endpoints around every hour'] = {'task': 'ayta.tasks.websub_renew_expiring', 'schedule': 4000} | ||||||
|  |     config['CELERY']['beat_schedule']['Process WebSub data around every two minutes'] = {'task': 'ayta.tasks.websub_process_data', 'schedule': 100} | ||||||
|  |     config['CELERY']['beat_schedule']['Queue up new videos in static channel playlists about 2 times a day'] = {'task': 'ayta.tasks.playlist_to_queue', 'schedule': 50000} | ||||||
|  |     config['CELERY']['beat_schedule']['Download around 123 videos spread out through the day'] = {'task': 'ayta.tasks.video_queue', 'schedule': 700} | ||||||
|  |     config['CELERY']['beat_schedule']['Generate new statistiscs about every 3 hours'] = {'task': 'ayta.tasks.generate_statistics', 'schedule': 10000} | ||||||
|  |      | ||||||
|  |     # Celery task routing | ||||||
|  |     # Tasks not defined in this configuration will be routed to the default queue "celery" | ||||||
|  |      | ||||||
|  |     config['CELERY']['task_routes'] = { | ||||||
|  |         'ayta.tasks.video_download': {'queue': 'download'}, | ||||||
|  |         'ayta.tasks.video_queue': {'queue': 'download'} | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     app = Flask(__name__) |     app = Flask(__name__) | ||||||
| @@ -32,6 +52,8 @@ def create_app(test_config=None): | |||||||
|     app.jinja_env.filters['pretty_time'] = filters.pretty_time |     app.jinja_env.filters['pretty_time'] = filters.pretty_time | ||||||
|     app.jinja_env.filters['current_time'] = filters.current_time |     app.jinja_env.filters['current_time'] = filters.current_time | ||||||
|     app.jinja_env.filters['epoch_time'] = filters.epoch_time |     app.jinja_env.filters['epoch_time'] = filters.epoch_time | ||||||
|  |     app.jinja_env.filters['epoch_date'] = filters.epoch_date | ||||||
|  |     app.jinja_env.filters['datetime_date'] = filters.datetime_date | ||||||
|      |      | ||||||
|     from .blueprints import watch |     from .blueprints import watch | ||||||
|     from .blueprints import index |     from .blueprints import index | ||||||
|   | |||||||
| @@ -1,8 +1,8 @@ | |||||||
| from flask import Blueprint, render_template, request, redirect, url_for, flash | from flask import Blueprint, render_template, request, redirect, url_for, flash, current_app | ||||||
| from ..nosql import get_nosql | from ..nosql import get_nosql | ||||||
| from ..dlp import checkChannelId, getChannelInfo | from ..dlp import checkChannelId, getChannelInfo | ||||||
| from ..decorators import login_required | from ..decorators import login_required | ||||||
| from ..tasks import subscribe_websub_callback, unsubscribe_websub_callback | from ..tasks import test_sleep, websub_subscribe_callback, websub_unsubscribe_callback, video_download, video_queue, playlist_to_queue | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
| from secrets import token_urlsafe | from secrets import token_urlsafe | ||||||
|  |  | ||||||
| @@ -30,6 +30,9 @@ def channels(): | |||||||
|     generic = {} |     generic = {} | ||||||
|  |  | ||||||
|     if request.method == 'POST': |     if request.method == 'POST': | ||||||
|  |         task = request.form.get('task', None) | ||||||
|  |          | ||||||
|  |         if task == 'add_channel': | ||||||
|             channelId = request.form.get('channel_id', None) |             channelId = request.form.get('channel_id', None) | ||||||
|             originalName = request.form.get('original_name', None) |             originalName = request.form.get('original_name', None) | ||||||
|             addedDate = request.form.get('added_date', None) |             addedDate = request.form.get('added_date', None) | ||||||
| @@ -47,11 +50,15 @@ def channels(): | |||||||
|  |  | ||||||
|             return redirect(url_for('admin.channel', channelId=channelId)) |             return redirect(url_for('admin.channel', channelId=channelId)) | ||||||
|              |              | ||||||
|  |         elif task == 'playlist-queue': | ||||||
|  |             task = playlist_to_queue.delay() | ||||||
|  |             flash(f'Task playlist-queue has been queued: {task.id}') | ||||||
|  |  | ||||||
|     generic['currentDate'] = datetime.utcnow() |     generic['currentDate'] = datetime.utcnow() | ||||||
|     channelIds = get_nosql().list_all_channels() |     channelIds = get_nosql().list_all_channels() | ||||||
|      |      | ||||||
|     for channelId in channelIds: |     for channelId in channelIds: | ||||||
|         channels[channelId] = get_nosql().get_channel_info(channelId) |         channels[channelId] = get_nosql().get_channel_info(channelId, limited=True) | ||||||
|         channels[channelId]['video_count'] = get_nosql().get_channel_videos_count(channelId) |         channels[channelId]['video_count'] = get_nosql().get_channel_videos_count(channelId) | ||||||
|          |          | ||||||
|     return render_template('admin/channels.html', channels=channels, generic=generic) |     return render_template('admin/channels.html', channels=channels, generic=generic) | ||||||
| @@ -71,15 +78,15 @@ def channel(channelId): | |||||||
|         value = request.form.get('value', None) |         value = request.form.get('value', None) | ||||||
|          |          | ||||||
|         if task == 'subscribe-websub': |         if task == 'subscribe-websub': | ||||||
|             task = subscribe_websub_callback.delay(channelId) |             task = websub_subscribe_callback.delay(channelId) | ||||||
|             flash(f"Started task {task.id}") |             flash(f"Started task {task.id}") | ||||||
|             return redirect(url_for('admin.channel', channelId=channelId)) |             return redirect(url_for('admin.channel', channelId=channelId)) | ||||||
|          |          | ||||||
|         if task == 'update-value': |         if task == 'update-value': | ||||||
|             if key == 'active': |             if key in ['active', 'websub']: | ||||||
|                 value = True if value else False |                 value = True if value else False | ||||||
|  |  | ||||||
|             if key == 'added_date': |             if key in ['added_date']: | ||||||
|                 value = datetime.strptime(value, '%Y-%m-%d') |                 value = datetime.strptime(value, '%Y-%m-%d') | ||||||
|                |                | ||||||
|             get_nosql().update_channel_key(channelId, key, value) |             get_nosql().update_channel_key(channelId, key, value) | ||||||
| @@ -109,29 +116,41 @@ def run(runId): | |||||||
| @bp.route('/websub', methods=['GET', 'POST']) | @bp.route('/websub', methods=['GET', 'POST']) | ||||||
| @login_required | @login_required | ||||||
| def websub(): | def websub(): | ||||||
|  |     render = {} | ||||||
|  |      | ||||||
|     if request.method == 'POST': |     if request.method == 'POST': | ||||||
|         task = request.form.get('task', None) |         task = request.form.get('task', None) | ||||||
|         value = request.form.get('value', None) |         value = request.form.get('value', None) | ||||||
|          |          | ||||||
|         if task == 'unsubscribe': |         if task == 'unsubscribe': | ||||||
|             channelId = get_nosql().websub_getCallback(value).get('channel') |             task = websub_unsubscribe_callback.delay(value) | ||||||
|              |  | ||||||
|             task = unsubscribe_websub_callback.delay(value, channelId) |  | ||||||
|              |              | ||||||
|             flash(f"Started task {task.id}") |             flash(f"Started task {task.id}") | ||||||
|             return redirect(url_for('admin.websub')) |             return redirect(url_for('admin.websub')) | ||||||
|              |  | ||||||
|         elif task == 'clean-retired': |         elif task == 'clean-retired': | ||||||
|             get_nosql().websub_cleanRetired() |             get_nosql().websub_cleanRetired() | ||||||
|             return redirect(url_for('admin.websub')) |             return redirect(url_for('admin.websub')) | ||||||
|  |         elif task == 'unsubscribe-callbacks': | ||||||
|  |             for callbackId in get_nosql().websub_getCallbacks(): | ||||||
|  |                 websub_unsubscribe_callback.delay(callbackId) | ||||||
|  |             flash(f"Started unsubscribe tasks for all callbacks") | ||||||
|  |             return redirect(url_for('admin.websub')) | ||||||
|  |         elif task == 'subscribe-channels': | ||||||
|  |             for channelId in get_nosql().list_all_channels(websub=True): | ||||||
|  |                 websub_subscribe_callback.delay(channelId) | ||||||
|  |             flash(f'Started subscribe tasks for activated channels') | ||||||
|  |             return redirect(url_for('admin.websub')) | ||||||
|  |              | ||||||
|      |      | ||||||
|     callbackIds = get_nosql().websub_getCallbacks() |     callbackIds = get_nosql().websub_getCallbacks() | ||||||
|     callbacks = {} |     callbacks = {} | ||||||
|      |      | ||||||
|  |     render['stats'] = get_nosql().websub_statistics() | ||||||
|  |      | ||||||
|     for callbackId in callbackIds: |     for callbackId in callbackIds: | ||||||
|         callbacks[callbackId] = get_nosql().websub_getCallback(callbackId) |         callbacks[callbackId] = get_nosql().websub_getCallback(callbackId) | ||||||
|      |      | ||||||
|     return render_template('admin/websub.html', callbacks=callbacks) |     return render_template('admin/websub.html', callbacks=callbacks, render=render) | ||||||
|  |  | ||||||
| @bp.route('/reports', methods=['GET', 'POST']) | @bp.route('/reports', methods=['GET', 'POST']) | ||||||
| @login_required | @login_required | ||||||
| @@ -144,14 +163,18 @@ def reports(): | |||||||
|             get_nosql().close_report(value) |             get_nosql().close_report(value) | ||||||
|             flash(f'Report closed {value}') |             flash(f'Report closed {value}') | ||||||
|             return redirect(url_for('admin.reports')) |             return redirect(url_for('admin.reports')) | ||||||
|  |         elif task == 'clean-closed': | ||||||
|  |             get_nosql().report_clean() | ||||||
|  |             flash(f'Cleaned closed reports older than 30 days') | ||||||
|  |             return redirect(url_for('admin.reports')) | ||||||
|  |  | ||||||
|     reports = get_nosql().list_reports() |     reports = get_nosql().list_reports() | ||||||
|     |     | ||||||
|     return render_template('admin/reports.html', reports=reports) |     return render_template('admin/reports.html', reports=reports) | ||||||
|  |  | ||||||
| @bp.route('/posters', methods=['GET', 'POST']) | @bp.route('/queue', methods=['GET', 'POST']) | ||||||
| @login_required | @login_required | ||||||
| def posters(): | def queue(): | ||||||
|     if request.method == 'POST': |     if request.method == 'POST': | ||||||
|         task = request.form.get('task', None) |         task = request.form.get('task', None) | ||||||
|         value = request.form.get('value', None) |         value = request.form.get('value', None) | ||||||
| @@ -162,34 +185,54 @@ def posters(): | |||||||
|                 flash('Description must be at least 8 characters long') |                 flash('Description must be at least 8 characters long') | ||||||
|              |              | ||||||
|             if value and len(value) >= 12: |             if value and len(value) >= 12: | ||||||
|                 get_nosql().poster_newEndpoint(value, description) |                 get_nosql().queue_newEndpoint(value, description) | ||||||
|                 flash(f'Created endpoint ID: {value}') |                 flash(f'Created endpoint ID: {value}') | ||||||
|             else: |             else: | ||||||
|                 value = token_urlsafe(16) |                 value = token_urlsafe(16) | ||||||
|                 get_nosql().poster_newEndpoint(value, description) |                 get_nosql().queue_newEndpoint(value, description) | ||||||
|                 flash(f'Created endpoint ID: {value}') |                 flash(f'Created endpoint ID: {value}') | ||||||
|  |                  | ||||||
|         elif task == 'retire': |         elif task == 'retire': | ||||||
|             get_nosql().poster_retireEndpoint(value) |             get_nosql().queue_retireEndpoint(value) | ||||||
|             flash(f'Endpoint retired: {value}') |             flash(f'Endpoint retired: {value}') | ||||||
|              |              | ||||||
|         elif task == 'clean-retired': |         elif task == 'clean-retired': | ||||||
|             get_nosql().poster_cleanRetired() |             get_nosql().queue_cleanRetired() | ||||||
|             flash(f'Cleaned retired endpoints') |             flash(f'Cleaned retired endpoints') | ||||||
|              |              | ||||||
|         elif task == 'manual-queue': |         elif task == 'manual-queue': | ||||||
|             get_nosql().poster_insertQueue('manual', value) |             if not get_nosql().check_exists(value): | ||||||
|  |                 direct = request.form.get('direct', None) | ||||||
|  |                 if direct: | ||||||
|  |                     task = video_download.delay(value) | ||||||
|  |                     flash(f"Started task {task.id}") | ||||||
|  |                 else: | ||||||
|  |                     get_nosql().queue_insertQueue(value, 'webui') | ||||||
|                     flash(f'Added to queue: {value}') |                     flash(f'Added to queue: {value}') | ||||||
|  |             else: | ||||||
|  |                 flash(f'This video ID already exists in the archive: {value}') | ||||||
|              |              | ||||||
|         elif task == 'delete-queue': |         elif task == 'delete-queue': | ||||||
|             get_nosql().poster_deleteQueue(value) |             get_nosql().queue_deleteQueue(value) | ||||||
|             flash(f'Deleted from queue: {value}') |             flash(f'Deleted from queue: {value}') | ||||||
|              |              | ||||||
|         return redirect(url_for('admin.posters')) |         elif task == 'empty-queue': | ||||||
|  |             get_nosql().queue_emptyQueue() | ||||||
|  |             flash(f'Queue has been emptied') | ||||||
|              |              | ||||||
|     endpoints = get_nosql().poster_getEndpoints() |         elif task == 'queue-run-once': | ||||||
|     queue = get_nosql().poster_getQueue() |             value = int(value) if value.isdigit() else 1 | ||||||
|  |             for x in range(value): | ||||||
|  |                 task =  video_queue.delay() | ||||||
|  |                 flash(f'Task has been started on the oldest queued item: {task.id}') | ||||||
|          |          | ||||||
|     return render_template('admin/posters.html', endpoints=endpoints, queue=queue) |         return redirect(url_for('admin.queue')) | ||||||
|  |          | ||||||
|  |     endpoints = get_nosql().queue_getEndpoints() | ||||||
|  |     queue = get_nosql().queue_getQueue() | ||||||
|  |     count = len(list(queue.clone())) | ||||||
|  |     | ||||||
|  |     return render_template('admin/queue.html', endpoints=endpoints, queue=queue, count=count) | ||||||
|  |  | ||||||
| @bp.route('/users', methods=['GET', 'POST']) | @bp.route('/users', methods=['GET', 'POST']) | ||||||
| @login_required | @login_required | ||||||
| @@ -218,3 +261,17 @@ def users(): | |||||||
|     users = get_nosql().list_all_users() |     users = get_nosql().list_all_users() | ||||||
|     |     | ||||||
|     return render_template('admin/users.html', users=users) |     return render_template('admin/users.html', users=users) | ||||||
|  |      | ||||||
|  | @bp.route('/workers', methods=['GET', 'POST']) | ||||||
|  | @login_required | ||||||
|  | def workers(): | ||||||
|  |     if request.method == 'POST': | ||||||
|  |         task = request.form.get('task', None) | ||||||
|  |         if task == 'test-sleep': | ||||||
|  |             test_sleep.delay() | ||||||
|  |  | ||||||
|  |     celery = current_app.extensions.get('celery') | ||||||
|  |      | ||||||
|  |     tasks = celery.control.inspect().active() | ||||||
|  |     reserved = celery.control.inspect().reserved() | ||||||
|  |     return render_template('admin/workers.html', tasks=tasks, reserved=reserved) | ||||||
| @@ -33,16 +33,16 @@ def websub(cap): | |||||||
|         return challenge |         return challenge | ||||||
|      |      | ||||||
|     if get_nosql().websub_existsCallback(cap): |     if get_nosql().websub_existsCallback(cap): | ||||||
|         if not get_nosql().websub_savePost(cap, str(request.data)): |         if not get_nosql().websub_savePost(cap, request.data): | ||||||
|             return abort(500) |             return abort(500) | ||||||
|         return '', 202 |         return '', 202 | ||||||
|      |      | ||||||
|     return abort(404) |     return abort(404) | ||||||
|      |      | ||||||
| @bp.route('/poster/<cap>', methods=['POST']) | @bp.route('/queue/<cap>', methods=['POST']) | ||||||
| def poster(cap): | def queue(cap): | ||||||
|     # if endpoint does not exist |     # if endpoint does not exist | ||||||
|     if not get_nosql().poster_isActive(cap): |     if not get_nosql().queue_isActive(cap): | ||||||
|         return abort(404) |         return abort(404) | ||||||
|      |      | ||||||
|     videoId = request.form.get('v') |     videoId = request.form.get('v') | ||||||
| @@ -60,7 +60,7 @@ def poster(cap): | |||||||
|         return abort(409) |         return abort(409) | ||||||
|      |      | ||||||
|     # try to insert |     # try to insert | ||||||
|     if get_nosql().poster_insertQueue(cap, videoId): |     if get_nosql().queue_insertQueue(videoId, cap): | ||||||
|         return '', 202 |         return '', 202 | ||||||
|     else: |     else: | ||||||
|         return abort(409) |         return abort(409) | ||||||
| @@ -11,7 +11,7 @@ def base(): | |||||||
|     channelIds = get_nosql().list_all_channels() |     channelIds = get_nosql().list_all_channels() | ||||||
|      |      | ||||||
|     for channelId in channelIds: |     for channelId in channelIds: | ||||||
|         channel = get_nosql().get_channel_info(channelId) |         channel = get_nosql().get_channel_info(channelId, limited=True) | ||||||
|         channel['video_count'] = get_nosql().get_channel_videos_count(channelId) |         channel['video_count'] = get_nosql().get_channel_videos_count(channelId) | ||||||
|         channels.append(channel) |         channels.append(channel) | ||||||
|          |          | ||||||
|   | |||||||
| @@ -22,5 +22,4 @@ def base(): | |||||||
|              |              | ||||||
|             return render_template('search/index.html', results=results, query=query) |             return render_template('search/index.html', results=results, query=query) | ||||||
|                  |                  | ||||||
|          |     return render_template('search/index.html', stats=get_nosql().statistics_get()) | ||||||
|     return render_template('search/index.html', stats=get_nosql().gen_stats()) |  | ||||||
| @@ -36,4 +36,9 @@ def base(): | |||||||
|  |  | ||||||
|     render['info'] = get_nosql().get_video_info(vGet) |     render['info'] = get_nosql().get_video_info(vGet) | ||||||
|     render['params'] = request.args.get('v') |     render['params'] = request.args.get('v') | ||||||
|  |      | ||||||
|  |     if render['info'].get('_status') != 'available': | ||||||
|  |         flash(render['info'].get('_status_description', 'Video unavailable because of technical errors. Come back later.')) | ||||||
|  |         return redirect(url_for('index.base')) | ||||||
|  |          | ||||||
|     return render_template('watch/index.html', render=render) |     return render_template('watch/index.html', render=render) | ||||||
|   | |||||||
| @@ -16,9 +16,21 @@ def pretty_time(time): | |||||||
|         except: |         except: | ||||||
|             return time  # return given time |             return time  # return given time | ||||||
|  |  | ||||||
| def epoch_time(time): | def epoch_date(epoch): | ||||||
|     try: |     try: | ||||||
|         return datetime.fromtimestamp(time).strftime('%d %b %Y') |         return datetime.fromtimestamp(epoch).strftime('%d %b %Y') | ||||||
|  |     except: | ||||||
|  |         return None | ||||||
|  |          | ||||||
|  | def epoch_time(epoch): | ||||||
|  |     try: | ||||||
|  |         return datetime.fromtimestamp(epoch).strftime('%d %b %Y %H:%M:%S') | ||||||
|  |     except: | ||||||
|  |         return None | ||||||
|  |          | ||||||
|  | def datetime_date(obj): | ||||||
|  |     try: | ||||||
|  |         return obj.strftime('%d %b %Y %H:%M') | ||||||
|     except: |     except: | ||||||
|         return None |         return None | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										240
									
								
								ayta/nosql.py
									
									
									
									
									
								
							
							
						
						
									
										240
									
								
								ayta/nosql.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										49
									
								
								ayta/oidc.py
									
									
									
									
									
								
							
							
						
						
									
										49
									
								
								ayta/oidc.py
									
									
									
									
									
								
							| @@ -1,4 +1,11 @@ | |||||||
| class OIDC(): | class OIDC(): | ||||||
|  |     """ | ||||||
|  |     This function class is nothing more than a nonce and state store for security in the authentication mechanism. | ||||||
|  |     Additionally this class provides the function to generate redirect url's and check bearer tokens on their validity as well as caching jwt signing keys. | ||||||
|  |     Fairly barebones and should be 100% secure. (famous last words) | ||||||
|  |     This is made for form posted JWT's. While not the most secure it is the most easy way to implement. Moving on to a code based solution might be preferred in the future. | ||||||
|  |     The nonce and state store is in memory, so only one instance can be used at a time until central key caching is implemented. | ||||||
|  |     """ | ||||||
|     def __init__(self, app=None): |     def __init__(self, app=None): | ||||||
|         self.states = {} |         self.states = {} | ||||||
|         self.nonces = {} |         self.nonces = {} | ||||||
| @@ -15,27 +22,34 @@ class OIDC(): | |||||||
|         self.client_id = config['OIDC_ID'] |         self.client_id = config['OIDC_ID'] | ||||||
|         self.provider = config['OIDC_PROVIDER'] |         self.provider = config['OIDC_PROVIDER'] | ||||||
|         self.domain = config['DOMAIN'] |         self.domain = config['DOMAIN'] | ||||||
|  |         self.window = 120  # the time window to allow states and nonces in seconds | ||||||
|          |          | ||||||
|  |         # Authentication provider url must be HTTPS and end on a TLD | ||||||
|         if self.provider[:8] != 'https://' or self.provider[-1] == '/': |         if self.provider[:8] != 'https://' or self.provider[-1] == '/': | ||||||
|             print('Incorrect OIDC provider URI', flush=True) |             print('Incorrect OIDC provider URI', flush=True) | ||||||
|             exit() |             exit() | ||||||
|          |          | ||||||
|  |         # Get the provider configuration endpoints | ||||||
|         configuration = requests.get(f'{self.provider}/.well-known/openid-configuration').json() |         configuration = requests.get(f'{self.provider}/.well-known/openid-configuration').json() | ||||||
|          |          | ||||||
|         jwks_uri = configuration.get('jwks_uri') |         jwks_uri = configuration.get('jwks_uri') | ||||||
|         self.authorize_uri = configuration.get('authorization_endpoint') |         self.authorize_uri = configuration.get('authorization_endpoint') | ||||||
|          |          | ||||||
|  |         # Start the JWKS management client, it will load the keys and maintain them | ||||||
|         self.jwks_manager = jwt.PyJWKClient(jwks_uri) |         self.jwks_manager = jwt.PyJWKClient(jwks_uri) | ||||||
|  |  | ||||||
|     ################################# |     ####################################################### | ||||||
|  |  | ||||||
|     def state_maintenance(self): |     def state_maintenance(self): | ||||||
|         from datetime import datetime |         from datetime import datetime | ||||||
|          |          | ||||||
|         pivot = datetime.now().timestamp() - 120 |         # Current time minus the acceptable window | ||||||
|  |         pivot = datetime.now().timestamp() - self.window | ||||||
|          |          | ||||||
|  |         # List with expired states | ||||||
|         expired_states = [state for state, timestamp in self.states.items() if timestamp <= pivot] |         expired_states = [state for state, timestamp in self.states.items() if timestamp <= pivot] | ||||||
|          |          | ||||||
|  |         # Remove expired states from store | ||||||
|         for state in expired_states: |         for state in expired_states: | ||||||
|             del self.states[state] |             del self.states[state] | ||||||
|  |  | ||||||
| @@ -43,30 +57,40 @@ class OIDC(): | |||||||
|         import secrets |         import secrets | ||||||
|         from datetime import datetime |         from datetime import datetime | ||||||
|          |          | ||||||
|  |         # Clean state store first | ||||||
|         self.state_maintenance() |         self.state_maintenance() | ||||||
|          |          | ||||||
|  |         # Generate token and paired timestamp | ||||||
|         state = secrets.token_urlsafe(8) |         state = secrets.token_urlsafe(8) | ||||||
|         timestamp = datetime.now().timestamp() |         timestamp = datetime.now().timestamp() | ||||||
|          |          | ||||||
|  |         # Add token to the state store | ||||||
|         self.states[state] = timestamp |         self.states[state] = timestamp | ||||||
|          |          | ||||||
|  |         # Return the state | ||||||
|         return state |         return state | ||||||
|          |          | ||||||
|     def state_check(self, state): |     def state_check(self, state): | ||||||
|  |         # Clean state store first | ||||||
|         self.state_maintenance() |         self.state_maintenance() | ||||||
|  |  | ||||||
|  |         # If given state is actively stored | ||||||
|         if state in self.states: |         if state in self.states: | ||||||
|  |             # Delete state and return True | ||||||
|             del self.states[state] |             del self.states[state] | ||||||
|             return True |             return True | ||||||
|          |          | ||||||
|  |         # Given state is not stored | ||||||
|         return False |         return False | ||||||
|          |          | ||||||
|     ################################# |     ####################################################### | ||||||
|  |     # Same code as above but a different store for nonces # | ||||||
|  |     ####################################################### | ||||||
|      |      | ||||||
|     def nonce_maintenance(self): |     def nonce_maintenance(self): | ||||||
|         from datetime import datetime |         from datetime import datetime | ||||||
|          |          | ||||||
|         pivot = datetime.now().timestamp() - 120 |         pivot = datetime.now().timestamp() - self.window | ||||||
|          |          | ||||||
|         expired_nonces = [nonce for nonce, timestamp in self.nonces.items() if timestamp <= pivot] |         expired_nonces = [nonce for nonce, timestamp in self.nonces.items() if timestamp <= pivot] | ||||||
|          |          | ||||||
| @@ -95,7 +119,7 @@ class OIDC(): | |||||||
|              |              | ||||||
|         return False |         return False | ||||||
|      |      | ||||||
|     ################################# |     ####################################################### | ||||||
|      |      | ||||||
|     def generate_redirect(self): |     def generate_redirect(self): | ||||||
|         return str(f'{self.authorize_uri}' |         return str(f'{self.authorize_uri}' | ||||||
| @@ -108,20 +132,31 @@ class OIDC(): | |||||||
|     def check_bearer(self, token): |     def check_bearer(self, token): | ||||||
|         import jwt |         import jwt | ||||||
|          |          | ||||||
|  |         # Test given JWT | ||||||
|         try: |         try: | ||||||
|  |             # Get the signed public key from the token | ||||||
|             signing_key = self.jwks_manager.get_signing_key_from_jwt(token).key |             signing_key = self.jwks_manager.get_signing_key_from_jwt(token).key | ||||||
|  |              | ||||||
|  |             # Try to decode the token, this will also check the validity in these points: | ||||||
|  |             # 1. Token is signed by expected keys | ||||||
|  |             # 2. Token is issued by the expected provider | ||||||
|  |             # 3. Expected parameters are really in the token | ||||||
|  |             # 4. Token is really intended for us | ||||||
|  |             # 5. Token is still valid (with 5 sec margin) | ||||||
|             decoded = jwt.decode(token, signing_key, |             decoded = jwt.decode(token, signing_key, | ||||||
|                                  algorithms=jwt.algorithms.get_default_algorithms(), |                                  algorithms=jwt.algorithms.get_default_algorithms(), | ||||||
|                                  issuer=self.provider, |                                  issuer=self.provider, | ||||||
|                                  require=['aud', 'client_id', 'exp', 'iat', 'iss', 'rat', 'sub'], |                                  require=['aud', 'client_id', 'exp', 'iat', 'iss', 'rat', 'sub'], | ||||||
|                                  audience=self.client_id, |                                  audience=self.client_id, | ||||||
|                                  leeway=5) |                                  leeway=5) | ||||||
|  |                                   | ||||||
|  |         # Any exception (invalid JWT, invalid formatting etc...) must return False | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             print(e, flush=True) |  | ||||||
|             return False |             return False | ||||||
|  |  | ||||||
|         # double check if given token is really requested by us |         # Double check if given token is really requested by us by matching the nonce in the signed key | ||||||
|         if not self.nonce_check(decoded.get('nonce', None)): |         if not self.nonce_check(decoded.get('nonce', None)): | ||||||
|             return False |             return False | ||||||
|          |          | ||||||
|  |         # Return the unique user identifier | ||||||
|         return decoded.get('sub', False) |         return decoded.get('sub', False) | ||||||
|   | |||||||
							
								
								
									
										201
									
								
								ayta/tasks.py
									
									
									
									
									
								
							
							
						
						
									
										201
									
								
								ayta/tasks.py
									
									
									
									
									
								
							| @@ -1,22 +1,74 @@ | |||||||
| from celery import shared_task | from celery import shared_task | ||||||
| from flask import current_app | from flask import current_app | ||||||
|  |  | ||||||
|  | ########################################## | ||||||
|  | #              CELERY TASKS              # | ||||||
|  | ########################################## | ||||||
|  |  | ||||||
| @shared_task() | @shared_task() | ||||||
| def subscribe_websub_callback(channelId): | def test_sleep(time=60): | ||||||
|  |     from time import sleep | ||||||
|  |     sleep(time) | ||||||
|  |     return True | ||||||
|  |      | ||||||
|  | @shared_task() | ||||||
|  | def video_download(videoId): | ||||||
|  |     """  | ||||||
|  |     I do not want to deal with the quirks of native yt-dlp in python, hence the subprocess. | ||||||
|  |     """ | ||||||
|  |     import subprocess | ||||||
|  |      | ||||||
|  |     process = subprocess.run(['/usr/local/bin/yt-dlp', '--config-location', '/var/www/archive.ventilaar.net/goodstuff/config_video.conf', '--', f'https://www.youtube.com/watch?v={videoId}'], stdout=subprocess.PIPE, stderr=subprocess.STDOUT, text=True) | ||||||
|  |      | ||||||
|  |     if process.returncode != 0: | ||||||
|  |         return (False, process.stdout) | ||||||
|  |     return (True, None) | ||||||
|  |      | ||||||
|  | @shared_task() | ||||||
|  | def video_queue(): | ||||||
|  |     """  | ||||||
|  |     Gets the oldest video ID from the queue and runs video_download() on it. | ||||||
|  |     """ | ||||||
|  |     from .nosql import get_nosql | ||||||
|  |      | ||||||
|  |     videoId = get_nosql().queue_getNext() | ||||||
|  |      | ||||||
|  |     if videoId: | ||||||
|  |         videoId = videoId['id'] | ||||||
|  |     else: | ||||||
|  |         return None | ||||||
|  |      | ||||||
|  |     status, reason = video_download(videoId) | ||||||
|  |      | ||||||
|  |     if status: | ||||||
|  |         get_nosql().queue_deleteQueue(videoId) | ||||||
|  |         return True | ||||||
|  |     else: | ||||||
|  |         get_nosql().queue_setFailed(videoId, reason) | ||||||
|  |         return False | ||||||
|  |  | ||||||
|  | @shared_task() | ||||||
|  | def websub_subscribe_callback(channelId): | ||||||
|     import requests |     import requests | ||||||
|     from .nosql import get_nosql |     from .nosql import get_nosql | ||||||
|      |      | ||||||
|  |     # check if a callback already exists for channel | ||||||
|  |     answer = get_nosql().websub_existsCallback(channelId, channel=True) | ||||||
|  |      | ||||||
|  |     if not answer: | ||||||
|         callbackId = get_nosql().websub_newCallback(channelId) |         callbackId = get_nosql().websub_newCallback(channelId) | ||||||
|  |     else: | ||||||
|  |         callbackId = answer | ||||||
|      |      | ||||||
|     url = 'https://pubsubhubbub.appspot.com/subscribe' |     url = 'https://pubsubhubbub.appspot.com/subscribe' | ||||||
|     data = { |     data = { | ||||||
|         'hub.callback': f'https://{current_app.config["DOMAIN"]}/api/websub//{callbackId}', |         'hub.callback': f'{current_app.config["DOMAIN"]}/api/websub/{callbackId}', | ||||||
|         'hub.topic': f'https://www.youtube.com/xml/feeds/videos.xml?channel_id={channelId}', |         'hub.topic': f'https://www.youtube.com/xml/feeds/videos.xml?channel_id={channelId}', | ||||||
|         'hub.verify': 'async', |         'hub.verify': 'async', | ||||||
|         'hub.mode': 'subscribe', |         'hub.mode': 'subscribe', | ||||||
|         'hub.verify_token': '', |         'hub.verify_token': '', | ||||||
|         'hub.secret': '', |         'hub.secret': '', | ||||||
|         'hub.lease_numbers': '86400', |         'hub.lease_numbers': '432000', | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     get_nosql().websub_requestingCallback(callbackId) |     get_nosql().websub_requestingCallback(callbackId) | ||||||
| @@ -24,15 +76,24 @@ def subscribe_websub_callback(channelId): | |||||||
|     if response.status_code == 202: |     if response.status_code == 202: | ||||||
|         return True |         return True | ||||||
|      |      | ||||||
|  |     # maybe handle errors? | ||||||
|  |      | ||||||
|     return False |     return False | ||||||
|      |      | ||||||
| @shared_task() | @shared_task() | ||||||
| def unsubscribe_websub_callback(callbackId, channelId): | def websub_unsubscribe_callback(callbackId): | ||||||
|     import requests |     import requests | ||||||
|     from .nosql import get_nosql |     from .nosql import get_nosql | ||||||
|      |      | ||||||
|  |     answer = get_nosql().websub_existsCallback(callbackId) | ||||||
|  |      | ||||||
|  |     if not answer: | ||||||
|  |         return False | ||||||
|  |          | ||||||
|  |     channelId = get_nosql().websub_getCallback(callbackId).get('channel') | ||||||
|  |      | ||||||
|     url = 'https://pubsubhubbub.appspot.com/subscribe' |     url = 'https://pubsubhubbub.appspot.com/subscribe' | ||||||
|     data = {'hub.callback': f'https://{current_app.config["DOMAIN"]}/api/websub/{callbackId}', |     data = {'hub.callback': f'{current_app.config["DOMAIN"]}/api/websub/{callbackId}', | ||||||
|             'hub.topic': f'https://www.youtube.com/xml/feeds/videos.xml?channel_id={channelId}', |             'hub.topic': f'https://www.youtube.com/xml/feeds/videos.xml?channel_id={channelId}', | ||||||
|             'hub.verify': 'async', |             'hub.verify': 'async', | ||||||
|             'hub.mode': 'unsubscribe' |             'hub.mode': 'unsubscribe' | ||||||
| @@ -44,4 +105,134 @@ def unsubscribe_websub_callback(callbackId, channelId): | |||||||
|     if response.status_code == 202: |     if response.status_code == 202: | ||||||
|         return True |         return True | ||||||
|          |          | ||||||
|  |     # maybe handle errors? | ||||||
|  |          | ||||||
|     return False |     return False | ||||||
|  |      | ||||||
|  | @shared_task() | ||||||
|  | def websub_process_data(): | ||||||
|  |     from .nosql import get_nosql | ||||||
|  |      | ||||||
|  |     while True: | ||||||
|  |         blob = get_nosql().websub_getFirstPostData() | ||||||
|  |         if not blob: | ||||||
|  |             break | ||||||
|  |          | ||||||
|  |         _id, data = blob | ||||||
|  |          | ||||||
|  |         parsed = do_parse_data(data) | ||||||
|  |         if parsed: | ||||||
|  |             state, channelId, videoId = parsed | ||||||
|  |          | ||||||
|  |             if state == 'added': | ||||||
|  |                 if not get_nosql().check_exists(videoId):  # if video not exists | ||||||
|  |                    get_nosql().queue_insertQueue(videoId, 'WebSub') | ||||||
|  |                    # note for future me | ||||||
|  |                    # the websub notifications report ALL videos, including shorts and livestreams | ||||||
|  |                    # so if you are going to work on individual video downloading make sure you filter them! | ||||||
|  |  | ||||||
|  |             elif state == 'removed': | ||||||
|  |                 # we currently do not do anything with removed videos | ||||||
|  |                 # but the idea is to trigger a full channel mirror in case a creator started to mass delete videos | ||||||
|  |                 pass | ||||||
|  |              | ||||||
|  |         get_nosql().websub_deletePostProcessing(_id) | ||||||
|  |  | ||||||
|  | @shared_task() | ||||||
|  | def websub_renew_expiring(hours=6): | ||||||
|  |     from .nosql import get_nosql | ||||||
|  |     from datetime import datetime, timedelta | ||||||
|  |      | ||||||
|  |     count = 0 | ||||||
|  |  | ||||||
|  |     for callbackId in get_nosql().websub_getCallbacks(): | ||||||
|  |         data = get_nosql().websub_getCallback(callbackId) | ||||||
|  |          | ||||||
|  |         if data.get('status') not in ['active']:  # callback not active | ||||||
|  |             continue | ||||||
|  |          | ||||||
|  |         pivot = datetime.utcnow() + timedelta(hours=hours)  # hours past now | ||||||
|  |         expires = data.get('activation_time') + timedelta(seconds=data.get('lease'))  # callback expires at | ||||||
|  |          | ||||||
|  |         if pivot <= expires:  # expiration happens after n hours fron now | ||||||
|  |             continue  # skip callback | ||||||
|  |          | ||||||
|  |         # expiration happens within n hours | ||||||
|  |         websub_subscribe_callback.delay(data.get('channel')) | ||||||
|  |          | ||||||
|  |         # limit amount of subscribe requests to spread out the requests over time | ||||||
|  |         # with an expiration pivot of 6h and a maximum validity of 5 days we can currently handle 3072 channels | ||||||
|  |         count = count + 1 | ||||||
|  |         if count >= 256: | ||||||
|  |             break | ||||||
|  |  | ||||||
|  | @shared_task() | ||||||
|  | def playlist_to_queue(): | ||||||
|  |     """ | ||||||
|  |     As there is still one cronjob based task running daily in the background, we have to make sure that gets hooked as well into the system. | ||||||
|  |     The cronjob task gets the last 50 uploads for all channels and commits the playlist json into the database | ||||||
|  |     This task makes sure we append the ID's that we got from the playlist into the download queue. | ||||||
|  |     Should idealy be run after the cronjob completes, but I don't want to implement an API that does that, so this gets run twice a day. | ||||||
|  |     """ | ||||||
|  |     from .nosql import get_nosql | ||||||
|  |     import random | ||||||
|  |     from datetime import datetime, timedelta | ||||||
|  |      | ||||||
|  |     pivot = datetime.utcnow() - timedelta(days=3)  # calculates 3 days before now | ||||||
|  |     channels = list(get_nosql().list_all_channels(active=True)) | ||||||
|  |     random.shuffle(channels)  # randomize channelId order because otherwise the queue will follow the channel order as well | ||||||
|  |      | ||||||
|  |     for channel in channels: | ||||||
|  |         info = get_nosql().get_channel_info(channel) | ||||||
|  |          | ||||||
|  |         # if last_run not set or last_run is older than the pivot (indicating it has not been updated) | ||||||
|  |         if not info.get('last_run') or info.get('last_run') < pivot: | ||||||
|  |             # skip channel | ||||||
|  |             continue | ||||||
|  |          | ||||||
|  |         for item in info['playlist']['entries']: | ||||||
|  |             videoId = item['id'] | ||||||
|  |             get_nosql().queue_insertQueue(videoId, 'Playlist mirroring') | ||||||
|  |              | ||||||
|  | @shared_task() | ||||||
|  | def generate_statistics(): | ||||||
|  |     from .nosql import get_nosql | ||||||
|  |     get_nosql().statistics_generate() | ||||||
|  |  | ||||||
|  | ########################################## | ||||||
|  | #              TASK MODULES              # | ||||||
|  | ########################################## | ||||||
|  |  | ||||||
|  | def do_parse_data(data): | ||||||
|  |     import xml.etree.ElementTree as ET | ||||||
|  |      | ||||||
|  |     data = data.decode('utf-8') | ||||||
|  |  | ||||||
|  |     try: | ||||||
|  |         root = ET.fromstring(data) | ||||||
|  |     except ET.ParseError: | ||||||
|  |         print('Not XML') | ||||||
|  |         return False | ||||||
|  |  | ||||||
|  |     yt = any(child.tag.startswith('{http://www.youtube.com/xml/schemas/2015}') for child in root.iter()) | ||||||
|  |     at = any(child.tag.startswith('{http://purl.org/atompub/tombstones/1.0}') for child in root.iter()) | ||||||
|  |  | ||||||
|  |     if yt and not at: | ||||||
|  |         # Video published | ||||||
|  |         state = 'added' | ||||||
|  |         ns = {'yt': 'http://www.youtube.com/xml/schemas/2015', '': 'http://www.w3.org/2005/Atom'} | ||||||
|  |         entry = root.find('.//{http://www.w3.org/2005/Atom}entry') | ||||||
|  |         videoId = entry.find('./yt:videoId', ns).text | ||||||
|  |         channelId = entry.find('./yt:channelId', ns).text | ||||||
|  |     elif not yt and at: | ||||||
|  |         # Video hidden | ||||||
|  |         state = 'removed' | ||||||
|  |         ns = {'at': 'http://purl.org/atompub/tombstones/1.0', '': 'http://www.w3.org/2005/Atom'} | ||||||
|  |         deleted_entry = root.find('.//{http://purl.org/atompub/tombstones/1.0}deleted-entry') | ||||||
|  |         videoId = deleted_entry.attrib['ref'].split(':')[-1] | ||||||
|  |         channelId = deleted_entry.find('./at:by/uri', ns).text.split('/')[-1] | ||||||
|  |     else: | ||||||
|  |         print('Unknown xml') | ||||||
|  |         return False | ||||||
|  |          | ||||||
|  |     return (state, channelId, videoId) | ||||||
| @@ -19,7 +19,7 @@ | |||||||
|     {% for item in channelInfo %} |     {% for item in channelInfo %} | ||||||
| 	  <form method="POST"> | 	  <form method="POST"> | ||||||
| 	    <div class="input-field"> | 	    <div class="input-field"> | ||||||
| 		  <span class="supporting-text">{{ item }}</span> | 		  <span class="supporting-text mb-2">{{ item }}</span> | ||||||
| 		  <input class="validate" type="text" value="{{ item }}" name="key" hidden> | 		  <input class="validate" type="text" value="{{ item }}" name="key" hidden> | ||||||
| 		</div> | 		</div> | ||||||
| 		   | 		   | ||||||
|   | |||||||
| @@ -15,6 +15,18 @@ | |||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
| <div class="row"> | <div class="row"> | ||||||
|  |   <div class="col s12 l4 m-4"> | ||||||
|  |     <div class="card"> | ||||||
|  |       <div class="card-content"> | ||||||
|  |         <span class="card-title">Direct actions</span> | ||||||
|  |         <form class="mt-4" method="post"> | ||||||
|  |           <button class="btn mb-2 green" type="submit" name="task" value="playlist-queue">Playlist to Queue</button> | ||||||
|  |           <br> | ||||||
|  |           <span class="supporting-text">Forcerun playlist to queue task</span> | ||||||
|  |         </form> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|   <div class="col s12 l4 m-4"> |   <div class="col s12 l4 m-4"> | ||||||
| 	<div class="card"> | 	<div class="card"> | ||||||
|       <div class="card-content"> |       <div class="card-content"> | ||||||
| @@ -38,7 +50,7 @@ | |||||||
|               }); |               }); | ||||||
|             </script> |             </script> | ||||||
| 		  </div> | 		  </div> | ||||||
|           <button class="btn mt-4" type="submit" name="action" value="add_channel">Add</button> |           <button class="btn mt-4" type="submit" name="task" value="add_channel">Add</button> | ||||||
| 	    </form> | 	    </form> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -66,11 +66,11 @@ | |||||||
|     </a> |     </a> | ||||||
|   </div> |   </div> | ||||||
|   <div class="col s6 l4 m-4"> |   <div class="col s6 l4 m-4"> | ||||||
| 	<a href="{{ url_for('admin.posters') }}"> |     <a href="{{ url_for('admin.queue') }}"> | ||||||
|       <div class="card black-text"> |       <div class="card black-text"> | ||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
|           <span class="card-title">Posters</span> |           <span class="card-title">Queue</span> | ||||||
| 		  <p class="grey-text">Extension posters</p> |           <p class="grey-text">Video download queue and API access</p> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|     </a> |     </a> | ||||||
| @@ -85,5 +85,15 @@ | |||||||
|       </div> |       </div> | ||||||
|     </a> |     </a> | ||||||
|   </div> |   </div> | ||||||
|  |   <div class="col s6 l4 m-4"> | ||||||
|  |     <a href="{{ url_for('admin.workers') }}"> | ||||||
|  |       <div class="card black-text"> | ||||||
|  |         <div class="card-content"> | ||||||
|  |           <span class="card-title">Workers</span> | ||||||
|  |           <p class="grey-text">Worker and task management</p> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </a> | ||||||
|  |   </div> | ||||||
| </div> | </div> | ||||||
| {% endblock %} | {% endblock %} | ||||||
| @@ -1,150 +0,0 @@ | |||||||
| {% extends 'material_base.html' %} |  | ||||||
| {% block title %}Posters administration page{% endblock %} |  | ||||||
| {% block description %}Posters administration page of the AYTA system{% endblock %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="row"> |  | ||||||
|   <div class="col s12 l11"> |  | ||||||
|     <h4>Posters administration page</h4> |  | ||||||
|   </div> |  | ||||||
|   <div class="col s12 l1 m-5"> |  | ||||||
|     <form method="POST"> |  | ||||||
|       <input title="Prunes all deleted endpoints, but keeps last 3 days" type="submit" value="clean-retired" name="task"> |  | ||||||
|     </form> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| <div class="divider"></div> |  | ||||||
| <div class="row"> |  | ||||||
|   <div class="col s12"> |  | ||||||
| 	<h5>Poster options</h5> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| <div class="row"> |  | ||||||
|   <div class="col s12 l4 m-4"> |  | ||||||
| 	<div class="card"> |  | ||||||
|       <div class="card-content"> |  | ||||||
|         <span class="card-title">Create new endpoint</span> |  | ||||||
| 	    <form method="post"> |  | ||||||
| 		  <div class="row"> |  | ||||||
|             <div class="col s12 input-field"> |  | ||||||
|               <input placeholder="Custom endpoint" name="value" type="text" class="validate" minlength="12"> |  | ||||||
| 			  <span class="supporting-text">Leaving this empty will create a random secure string</span> |  | ||||||
| 	        </div> |  | ||||||
| 			<div class="col s12 input-field"> |  | ||||||
|               <input placeholder="Description" name="description" type="text" class="validate" minlength="8" maxlength="64" required> |  | ||||||
| 			  <span class="supporting-text">Description for the endpoint for better administration</span> |  | ||||||
| 	        </div> |  | ||||||
|           <button class="btn mt-4" type="submit" name="task" value="add-endpoint">Create</button> |  | ||||||
| 		  </div> |  | ||||||
| 	    </form> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
|   <div class="col s12 l4 m-4"> |  | ||||||
| 	<div class="card"> |  | ||||||
|       <div class="card-content"> |  | ||||||
|         <span class="card-title">Queue manually</span> |  | ||||||
| 	    <form method="post"> |  | ||||||
| 		  <div class="row"> |  | ||||||
|             <div class="col s12 input-field"> |  | ||||||
|               <input placeholder="Youtube video ID" name="value" type="text" class="validate" minlength="11" maxlength="11" required> |  | ||||||
| 			  <span class="supporting-text">Must be a valid Youtube video ID</span> |  | ||||||
| 	        </div> |  | ||||||
| 			<div class="col s12 mt-5 input-field"> |  | ||||||
|               <div class="switch"> |  | ||||||
| 		        <label>Queue<input type="checkbox" value="direct" name="value" disabled><span class="lever"></span>Direct</label> |  | ||||||
| 				<span class="supporting-text">Queue up or start directly</span> |  | ||||||
| 		      </div> |  | ||||||
| 	        </div> |  | ||||||
|           <button class="btn mt-4" type="submit" name="task" value="manual-queue">Queue</button> |  | ||||||
| 		  </div> |  | ||||||
| 	    </form> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| <div class="divider"></div> |  | ||||||
| <div class="row"> |  | ||||||
|   <div class="col s6 l9"> |  | ||||||
| 	<h5>Registered endpoints</h5> |  | ||||||
|   </div> |  | ||||||
|   <div class="col s6 l3 m-4 input-field"> |  | ||||||
| 	<input id="filter_query" type="text"> |  | ||||||
|     <label for="filter_query">Filter results</label> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| <div class="row"> |  | ||||||
|   <div class="col s12"> |  | ||||||
|     <table class="striped highlight responsive-table"> |  | ||||||
|       <thead> |  | ||||||
|         <tr> |  | ||||||
| 		  <th>Actions</th> |  | ||||||
| 		  <th>id</th> |  | ||||||
| 		  <th>description</th> |  | ||||||
| 		  <th>status</th> |  | ||||||
| 		  <th>created_time</th> |  | ||||||
| 		  <th>retired_time</th> |  | ||||||
|         </tr> |  | ||||||
|       </thead> |  | ||||||
|       <tbody> |  | ||||||
| 	    {% for endpoint in endpoints %} |  | ||||||
|         <tr class="filterable"> |  | ||||||
| 		  <td> |  | ||||||
| 		    <form method="post"> |  | ||||||
| 			  <input type="text" value="{{ endpoint.get('id') }}" name="value" hidden> |  | ||||||
| 		      <button class="btn-small waves-effect waves-light" type="submit" name="task" value="retire" title="Retire endpoint" {% if endpoint.get('status') != 'active' %}disabled{% endif %}>🗑️</button> |  | ||||||
| 			</form> |  | ||||||
| 		  </td> |  | ||||||
| 		  <td>{{ endpoint.get('id') }}</td> |  | ||||||
| 		  <td>{{ endpoint.get('description') }}</td> |  | ||||||
| 		  <td>{{ endpoint.get('status') }}</td> |  | ||||||
| 		  <td>{{ endpoint.get('created_time') }}</td> |  | ||||||
| 		  <td>{{ endpoint.get('retired_time') }}</td> |  | ||||||
|         </tr> |  | ||||||
| 		{% endfor %} |  | ||||||
|       </tbody> |  | ||||||
|     </table> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| <div class="divider"></div> |  | ||||||
| <div class="row"> |  | ||||||
|   <div class="col s6 l9"> |  | ||||||
| 	<h5>Queued ID's</h5> |  | ||||||
|   </div> |  | ||||||
|   <div class="col s6 l3 m-4 input-field"> |  | ||||||
| 	<input id="filter_query" type="text"> |  | ||||||
|     <label for="filter_query">Filter results</label> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| <div class="row"> |  | ||||||
|   <div class="col s12"> |  | ||||||
|     <table class="striped highlight responsive-table"> |  | ||||||
|       <thead> |  | ||||||
|         <tr> |  | ||||||
| 		  <th>Actions</th> |  | ||||||
| 		  <th>id</th> |  | ||||||
| 		  <th>endpoint</th> |  | ||||||
| 		  <th>status</th> |  | ||||||
| 		  <th>created_time</th> |  | ||||||
|         </tr> |  | ||||||
|       </thead> |  | ||||||
|       <tbody> |  | ||||||
| 	    {% for id in queue %} |  | ||||||
|         <tr class="filterable"> |  | ||||||
| 		  <td> |  | ||||||
| 		    <form method="post"> |  | ||||||
| 			  <input type="text" value="{{ id.get('id') }}" name="value" hidden> |  | ||||||
| 		      <button class="btn-small waves-effect waves-light" type="submit" name="task" value="delete-queue" title="Delete from queue" {% if id.get('status') != 'queued' %}disabled{% endif %}>🗑️</button> |  | ||||||
| 			</form> |  | ||||||
| 		  </td> |  | ||||||
| 		  <td>{{ id.get('id') }}</td> |  | ||||||
| 		  <td>{{ id.get('endpoint') }}</td> |  | ||||||
| 		  <td>{{ id.get('status') }}</td> |  | ||||||
| 		  <td>{{ id.get('created_time') }}</td> |  | ||||||
|         </tr> |  | ||||||
| 		{% endfor %} |  | ||||||
|       </tbody> |  | ||||||
|     </table> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
							
								
								
									
										178
									
								
								ayta/templates/admin/queue.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								ayta/templates/admin/queue.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,178 @@ | |||||||
|  | {% extends 'material_base.html' %} | ||||||
|  | {% block title %}Queue administration page{% endblock %} | ||||||
|  | {% block description %}Queue administration page of the AYTA system{% endblock %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s12"> | ||||||
|  |     <h4>Queue administration page</h4> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | <div class="divider"></div> | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s12"> | ||||||
|  |     <h5>Options</h5> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s12 l4 m-4"> | ||||||
|  |     <div class="card"> | ||||||
|  |       <div class="card-content"> | ||||||
|  |         <span class="card-title">Direct actions</span> | ||||||
|  |         <form class="mt-4" method="post" onsubmit="return confirm('Are you sure?');"> | ||||||
|  |           <button class="btn mb-2 red" type="submit" name="task" value="empty-queue">Empty Queue</button> | ||||||
|  |           <br> | ||||||
|  |           <span class="supporting-text">Removes all queued ids</span> | ||||||
|  |         </form> | ||||||
|  |         <form class="mt-4" method="post" onsubmit="return confirm('Are you sure?');"> | ||||||
|  |           <button class="btn mb-2" type="submit" name="task" value="clean-retired">Clean retired</button> | ||||||
|  |           <br> | ||||||
|  |           <span class="supporting-text">Prunes all deactivated endpoints, but keeps last 3 days</span> | ||||||
|  |         </form> | ||||||
|  |         <form class="mt-4 input-field" method="post" onsubmit="return confirm('Are you sure?');"> | ||||||
|  |           <input type="number" style="width: 80px" value="1" name="value" min="1" max="99"> | ||||||
|  |           <button class="btn mb-2 green" type="submit" name="task" value="queue-run-once">Download oldest queued</button> | ||||||
|  |           <br> | ||||||
|  |           <span class="supporting-text">Will download the oldest queued video ID</span> | ||||||
|  |         </form> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  |   <div class="col s12 l4 m-4"> | ||||||
|  |     <div class="card"> | ||||||
|  |       <div class="card-content"> | ||||||
|  |         <span class="card-title">Create new endpoint</span> | ||||||
|  |         <form method="post"> | ||||||
|  |           <div class="row"> | ||||||
|  |             <div class="col s12 input-field"> | ||||||
|  |               <input placeholder="Custom endpoint" name="value" type="text" class="validate" minlength="12"> | ||||||
|  |               <span class="supporting-text">Leaving this empty will create a random secure string</span> | ||||||
|  |             </div> | ||||||
|  |             <div class="col s12 input-field"> | ||||||
|  |               <input placeholder="Description" name="description" type="text" class="validate" minlength="8" maxlength="64" required> | ||||||
|  |               <span class="supporting-text">Description for the endpoint for better administration</span> | ||||||
|  |             </div> | ||||||
|  |           <button class="btn mt-4" type="submit" name="task" value="add-endpoint">Create</button> | ||||||
|  |           </div> | ||||||
|  |         </form> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  |   <div class="col s12 l4 m-4"> | ||||||
|  |     <div class="card"> | ||||||
|  |       <div class="card-content"> | ||||||
|  |         <span class="card-title">Queue manually</span> | ||||||
|  |         <form method="post"> | ||||||
|  |           <div class="row"> | ||||||
|  |             <div class="col s12 input-field"> | ||||||
|  |               <input placeholder="Youtube video ID" name="value" type="text" class="validate" minlength="11" maxlength="11" required> | ||||||
|  |               <span class="supporting-text">Must be a valid Youtube video ID</span> | ||||||
|  |             </div> | ||||||
|  |             <div class="col s12 mt-5 input-field"> | ||||||
|  |               <div class="switch"> | ||||||
|  |                 <label>Queue<input type="checkbox" value="direct" name="direct"><span class="lever"></span>Direct</label> | ||||||
|  |                 <span class="supporting-text">Queue up or start directly</span> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           <button class="btn mt-4" type="submit" name="task" value="manual-queue">Queue</button> | ||||||
|  |           </div> | ||||||
|  |         </form> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | <div class="divider"></div> | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s6 l9"> | ||||||
|  |     <h5>Registered endpoints</h5> | ||||||
|  |   </div> | ||||||
|  |   <div class="col s6 l3 m-4 input-field"> | ||||||
|  |     <input id="filter_query" type="text"> | ||||||
|  |     <label for="filter_query">Filter results</label> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s12"> | ||||||
|  |     <table class="striped highlight responsive-table"> | ||||||
|  |       <thead> | ||||||
|  |         <tr> | ||||||
|  |           <th>Actions</th> | ||||||
|  |           <th>id</th> | ||||||
|  |           <th>description</th> | ||||||
|  |           <th>status</th> | ||||||
|  |           <th>created_time</th> | ||||||
|  |           <th>retired_time</th> | ||||||
|  |         </tr> | ||||||
|  |       </thead> | ||||||
|  |       <tbody> | ||||||
|  |         {% for endpoint in endpoints %} | ||||||
|  |         <tr class="filterable"> | ||||||
|  |           <td> | ||||||
|  |             <form method="post"> | ||||||
|  |               <input type="text" value="{{ endpoint.get('id') }}" name="value" hidden> | ||||||
|  |               <button class="btn-small waves-effect waves-light" type="submit" name="task" value="retire" title="Retire endpoint" {% if endpoint.get('status') != 'active' %}disabled{% endif %}>🗑️</button> | ||||||
|  |             </form> | ||||||
|  |           </td> | ||||||
|  |           <td>{{ endpoint.get('id') }}</td> | ||||||
|  |           <td>{{ endpoint.get('description') }}</td> | ||||||
|  |           <td>{{ endpoint.get('status') }}</td> | ||||||
|  |           <td>{{ endpoint.get('created_time') }}</td> | ||||||
|  |           <td>{{ endpoint.get('retired_time') }}</td> | ||||||
|  |         </tr> | ||||||
|  |         {% endfor %} | ||||||
|  |       </tbody> | ||||||
|  |     </table> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | <div class="divider"></div> | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s4 l8"> | ||||||
|  |     <h5>Queued ID's</h5> | ||||||
|  |   </div> | ||||||
|  |   <div class="col s4 l1"> | ||||||
|  |     <p>{{ count }} items</p> | ||||||
|  |   </div> | ||||||
|  |   <div class="col s4 l3 m-4 input-field"> | ||||||
|  |     <input id="filter_query" type="text"> | ||||||
|  |     <label for="filter_query">Filter results</label> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s12"> | ||||||
|  |     <table class="striped highlight responsive-table"> | ||||||
|  |       <thead> | ||||||
|  |         <tr> | ||||||
|  |           <th>Actions</th> | ||||||
|  |           <th>id</th> | ||||||
|  |           <th>endpoint</th> | ||||||
|  |           <th>status</th> | ||||||
|  |           <th>created_time</th> | ||||||
|  |           <th>fail_reason</th> | ||||||
|  |         </tr> | ||||||
|  |       </thead> | ||||||
|  |       <tbody> | ||||||
|  |         {% for id in queue %} | ||||||
|  |         <tr class="filterable"> | ||||||
|  |           <td> | ||||||
|  |             <form method="post"> | ||||||
|  |               <input type="text" value="{{ id.get('id') }}" name="value" hidden> | ||||||
|  |               <button class="btn-small waves-effect waves-light" type="submit" name="task" value="delete-queue" title="Delete from queue" {% if id.get('status') == 'working' %}disabled{% endif %}>🗑️</button> | ||||||
|  |             </form> | ||||||
|  |             <form method="post"> | ||||||
|  |               <input type="text" value="{{ id.get('id') }}" name="value" hidden> | ||||||
|  |               <button class="btn-small waves-effect waves-light" type="submit" name="task" value="run-download" title="Run download task" disabled>⏩</button> | ||||||
|  | 			  <!-- This function fill not work until the download queue and video download process is rewritten --> | ||||||
|  |             </form> | ||||||
|  |           </td> | ||||||
|  |           <td>{{ id.get('id') }}</td> | ||||||
|  |           <td>{{ id.get('endpoint') }}</td> | ||||||
|  |           <td>{{ id.get('status') }}</td> | ||||||
|  |           <td>{{ id.get('created_time') }}</td> | ||||||
|  |           <td><textarea class="info">{{ id.get('fail_reason') }}</textarea></td> | ||||||
|  |         </tr> | ||||||
|  |         {% endfor %} | ||||||
|  |       </tbody> | ||||||
|  |     </table> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | {% endblock %} | ||||||
| @@ -4,14 +4,9 @@ | |||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| <div class="row"> | <div class="row"> | ||||||
|   <div class="col s12 l11"> |   <div class="col s12"> | ||||||
|     <h4>WebSub administration page</h4> |     <h4>WebSub administration page</h4> | ||||||
|   </div> |   </div> | ||||||
|   <div class="col s12 l1 m-5"> |  | ||||||
|     <form method="POST"> |  | ||||||
| 	  <input title="Prunes all retired callbacks, but keeps last 3 days" type="submit" value="clean-retired" name="task"> |  | ||||||
|     </form> |  | ||||||
|   </div> |  | ||||||
| </div> | </div> | ||||||
| <div class="divider"></div> | <div class="divider"></div> | ||||||
| <div class="row"> | <div class="row"> | ||||||
| @@ -19,6 +14,43 @@ | |||||||
| 	<h5>WebSub options</h5> | 	<h5>WebSub options</h5> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s12 l4 m-4"> | ||||||
|  | 	<div class="card"> | ||||||
|  |       <div class="card-content"> | ||||||
|  |         <span class="card-title">Direct actions</span> | ||||||
|  | 	    <form method="post" onsubmit="return confirm('Are you sure?');"> | ||||||
|  |           <button class="btn mb-2 green" type="submit" name="task" value="subscribe-channels">Subscribe channels</button> | ||||||
|  | 		  <br> | ||||||
|  | 		  <span class="supporting-text">Send WebSub subscription request for all activated channels. (This will renew existing ones as well)</span> | ||||||
|  | 	    </form> | ||||||
|  | 		<form class="mt-4" method="post" onsubmit="return confirm('Are you sure?');"> | ||||||
|  |           <button class="btn mb-2 red" type="submit" name="task" value="unsubscribe-callbacks">Unsubscribe channels</button> | ||||||
|  | 		  <br> | ||||||
|  | 		  <span class="supporting-text">Send WebSub unsubscription request for all activated endpoints. (This will only unsubscribe, not disable)</span> | ||||||
|  | 	    </form> | ||||||
|  | 		<form class="mt-4" method="post" onsubmit="return confirm('Are you sure?');"> | ||||||
|  |           <button class="btn mb-2" type="submit" name="task" value="clean-retired">Clean retired</button> | ||||||
|  | 		  <br> | ||||||
|  | 		  <span class="supporting-text">Prunes all retired callbacks, but keeps until last day</span> | ||||||
|  | 	    </form> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  |   <div class="col s12 l4 m-4"> | ||||||
|  | 	<div class="card"> | ||||||
|  |       <div class="card-content"> | ||||||
|  |         <span class="card-title">Statistics</span> | ||||||
|  | 	    <h6>Unprocessed callback datapoints</h6> | ||||||
|  | 		<p>{{ render['stats']['unprocessed_data'] }}</p> | ||||||
|  | 		<h6>Active callbacks</h6> | ||||||
|  | 		<p>{{ render['stats']['active_callbacks'] }}</p> | ||||||
|  | 		<h6>Something</h6> | ||||||
|  | 		<p>Blah</p> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
| <div class="divider"></div> | <div class="divider"></div> | ||||||
| <div class="row"> | <div class="row"> | ||||||
|   <div class="col s6 l9"> |   <div class="col s6 l9"> | ||||||
| @@ -50,6 +82,7 @@ | |||||||
| 	    {% for callback in callbacks %} | 	    {% for callback in callbacks %} | ||||||
|         <tr class="filterable"> |         <tr class="filterable"> | ||||||
| 		  <td> | 		  <td> | ||||||
|  | 			<a target="_blank" rel="noopener noreferrer" href="https://pubsubhubbub.appspot.com/subscription-details?hub.callback={{ config['DOMAIN'] }}/api/websub/{{ callbacks[callback].get('id') }}&hub.topic=https://www.youtube.com/xml/feeds/videos.xml?channel_id={{ callbacks[callback].get('channel') }}"><button class="btn-small waves-effect waves-light" title="Information on Pubsubhubbub (external link)">ℹ️</button></a> | ||||||
| 		    <form method="post"> | 		    <form method="post"> | ||||||
| 			  <input type="text" value="{{ callbacks[callback].get('id') }}" name="value" hidden> | 			  <input type="text" value="{{ callbacks[callback].get('id') }}" name="value" hidden> | ||||||
| 		      <button class="btn-small waves-effect waves-light" type="submit" name="task" value="unsubscribe" title="Send unsubscribe request to hub" {% if callbacks[callback].get('status') != 'active' %}disabled{% endif %}>🗑️</button> | 		      <button class="btn-small waves-effect waves-light" type="submit" name="task" value="unsubscribe" title="Send unsubscribe request to hub" {% if callbacks[callback].get('status') != 'active' %}disabled{% endif %}>🗑️</button> | ||||||
|   | |||||||
							
								
								
									
										80
									
								
								ayta/templates/admin/workers.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								ayta/templates/admin/workers.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | |||||||
|  | {% extends 'material_base.html' %} | ||||||
|  | {% block title %}Workers administration page{% endblock %} | ||||||
|  | {% block description %}Workers administration page of the AYTA system{% endblock %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s12"> | ||||||
|  |     <h4>Workers administration page</h4> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | <div class="divider"></div> | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s12"> | ||||||
|  |     <h5>Options</h5> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | <form method="POST"> | ||||||
|  |       <input title="test-sleep" type="submit" value="test-sleep" name="task"> | ||||||
|  | </form> | ||||||
|  | <div class="divider"></div> | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s12 m-4"> | ||||||
|  |     <h5>Reserved tasks per worker</h5> | ||||||
|  |     <p>Usually 4 tasks per worker</p> | ||||||
|  | 	{% if reserved is none %} | ||||||
|  | 	<h6>No workers with reserved tasks, are there any workers with stuck tasks or are they even online?</h6> | ||||||
|  | 	{% else %} | ||||||
|  |     {% for worker in reserved %} | ||||||
|  |     <span>{{ worker }}</span> | ||||||
|  |     <table class="striped highlight responsive-table" style=" border: 1px solid black;"> | ||||||
|  |       <thead> | ||||||
|  |         <tr> | ||||||
|  |           <th>ID</th> | ||||||
|  |           <th>Task</th> | ||||||
|  |           <th>Arguments</th> | ||||||
|  |         </tr> | ||||||
|  |       </thead> | ||||||
|  |       <tbody> | ||||||
|  |         {% for task in reserved[worker] %} | ||||||
|  |         <tr> | ||||||
|  |           <td>{{ task.get('id') }}</td> | ||||||
|  |           <td>{{ task.get('name') }}</td> | ||||||
|  |           <td>{{ task.get('args') }} {{ task.get('kwargs') }}</td> | ||||||
|  |         </tr> | ||||||
|  |         {% endfor %} | ||||||
|  |       </tbody> | ||||||
|  |     </table> | ||||||
|  |     {% endfor %} | ||||||
|  | 	{% endif %} | ||||||
|  |   </div> | ||||||
|  |   <div class="col s12 m-4"> | ||||||
|  |     <h5>Current workers and processing tasks</h5> | ||||||
|  | 	{% if tasks is none %} | ||||||
|  | 	<h6>No workers with running tasks, are there any workers with stuck tasks or are they even online?</h6> | ||||||
|  | 	{% else %} | ||||||
|  |     {% for worker in tasks %} | ||||||
|  |     <span>{{ worker }}</span> | ||||||
|  |     <table class="striped highlight responsive-table" style=" border: 1px solid black;"> | ||||||
|  |       <thead> | ||||||
|  |         <tr> | ||||||
|  |           <th>ID</th> | ||||||
|  |           <th>Task</th> | ||||||
|  |           <th>Time started</th> | ||||||
|  |         </tr> | ||||||
|  |       </thead> | ||||||
|  |       <tbody> | ||||||
|  |         {% for task in tasks[worker] %} | ||||||
|  |         <tr> | ||||||
|  |           <td>{{ task.get('id') }}</td> | ||||||
|  |           <td>{{ task.get('name') }}</td> | ||||||
|  |           <td>{{ task.get('time_start')|epoch_time }}</td> | ||||||
|  |         </tr> | ||||||
|  |         {% endfor %} | ||||||
|  |       </tbody> | ||||||
|  |     </table> | ||||||
|  |     {% endfor %} | ||||||
|  | 	{% endif %} | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | {% endblock %} | ||||||
| @@ -25,7 +25,7 @@ | |||||||
| 	<div class="card medium black-text"> | 	<div class="card medium black-text"> | ||||||
| 	  <a href="{{ url_for('watch.base') }}?v={{ video.get('id') }}"> | 	  <a href="{{ url_for('watch.base') }}?v={{ video.get('id') }}"> | ||||||
| 	    <div class="card-image"> | 	    <div class="card-image"> | ||||||
| 		  <img loading="lazy" src="https://archive.ventilaar.net/videos/automatic/{{ video.get('channel_id') }}/{{ video.get('id') }}/{{ video.get('title_slug') }}.jpg"> | 		  <img loading="lazy" src="https://archive.ventilaar.net/videos/automatic/{{ video.get('channel_id') }}/{{ video.get('id') }}/{{ video.get('_title_slug') }}.jpg"> | ||||||
|         </div> |         </div> | ||||||
| 	  </a> | 	  </a> | ||||||
|       <div class="card-content activator"> |       <div class="card-content activator"> | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ | |||||||
| 	<div class="card medium black-text"> | 	<div class="card medium black-text"> | ||||||
| 	  <a href="{{ url_for('watch.base') }}?v={{ video.get('id') }}"> | 	  <a href="{{ url_for('watch.base') }}?v={{ video.get('id') }}"> | ||||||
| 	    <div class="card-image"> | 	    <div class="card-image"> | ||||||
| 		  <img loading="lazy" src="https://archive.ventilaar.net/videos/automatic/{{ video.get('channel_id') }}/{{ video.get('id') }}/{{ video.get('title_slug') }}.jpg"> | 		  <img loading="lazy" src="https://archive.ventilaar.net/videos/automatic/{{ video.get('channel_id') }}/{{ video.get('id') }}/{{ video.get('_title_slug') }}.jpg"> | ||||||
|         </div> |         </div> | ||||||
| 	  </a> | 	  </a> | ||||||
|       <div class="card-content activator"> |       <div class="card-content activator"> | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ | |||||||
| 	<div class="card medium black-text"> | 	<div class="card medium black-text"> | ||||||
| 	  <a href="{{ url_for('watch.base') }}?v={{ video.get('id') }}"> | 	  <a href="{{ url_for('watch.base') }}?v={{ video.get('id') }}"> | ||||||
| 	    <div class="card-image"> | 	    <div class="card-image"> | ||||||
| 		  <img loading="lazy" src="https://archive.ventilaar.net/videos/automatic/{{ video.get('channel_id') }}/{{ video.get('id') }}/{{ video.get('title_slug') }}.jpg"> | 		  <img loading="lazy" src="https://archive.ventilaar.net/videos/automatic/{{ video.get('channel_id') }}/{{ video.get('id') }}/{{ video.get('_title_slug') }}.jpg"> | ||||||
|         </div> |         </div> | ||||||
| 	  </a> | 	  </a> | ||||||
|       <div class="card-content activator"> |       <div class="card-content activator"> | ||||||
|   | |||||||
| @@ -25,12 +25,24 @@ | |||||||
| 	  </li> | 	  </li> | ||||||
| 	  <li class="collection-item"> | 	  <li class="collection-item"> | ||||||
| 	    <a href="{{ url_for('channel.channel', channelId='UCIcgBZ9hEJxHv6r_jDYOMqg') }}"><span class="title">Unus Annus</span></a> | 	    <a href="{{ url_for('channel.channel', channelId='UCIcgBZ9hEJxHv6r_jDYOMqg') }}"><span class="title">Unus Annus</span></a> | ||||||
|         <p>Reason: This channel does not exist. (Self removed)</p> |         <p>Reason: This channel does not exist.</p> | ||||||
| 	  </li> | 	  </li> | ||||||
| 	  <li class="collection-item"> | 	  <li class="collection-item"> | ||||||
| 	    <a href="{{ url_for('channel.channel', channelId='UCz1s8aJYSQuaXJCtEi-VWRA') }}"><span class="title">Dutch Legion</span></a> | 	    <a href="{{ url_for('channel.channel', channelId='UCz1s8aJYSQuaXJCtEi-VWRA') }}"><span class="title">Dutch Legion</span></a> | ||||||
|         <p>Reason: This account has been terminated due to multiple or severe violations of YouTube's policy prohibiting hate speech.</p> |         <p>Reason: This account has been terminated due to multiple or severe violations of YouTube's policy prohibiting hate speech.</p> | ||||||
| 	  </li> | 	  </li> | ||||||
|  | 	  <li class="collection-item"> | ||||||
|  | 	    <a href="{{ url_for('channel.channel', channelId='UC91-8aNaRbp71UMEb_34ryg') }}"><span class="title">RBMK5000</span></a> | ||||||
|  |         <p>Reason: This channel does not exist.</p> | ||||||
|  | 	  </li> | ||||||
|  | 	  <li class="collection-item"> | ||||||
|  | 	    <a href="{{ url_for('channel.channel', channelId='UCoPSAT64vfXlulyWd_dPE3Q') }}"><span class="title">Evilfisher2</span></a> | ||||||
|  |         <p>Reason: This channel was removed because it violated our Community Guidelines.</p> | ||||||
|  | 	  </li> | ||||||
|  | 	  <li class="collection-item"> | ||||||
|  | 	    <a href="{{ url_for('channel.channel', channelId='UCZXkvavD2YKnFCzCkZ-bNPw') }}"><span class="title">mrabhy</span></a> | ||||||
|  |         <p>Reason: This channel was removed because it violated our Community Guidelines.</p> | ||||||
|  | 	  </li> | ||||||
|     </ul> |     </ul> | ||||||
|   </div> |   </div> | ||||||
|   <div class="col s12 l6 center-align"> |   <div class="col s12 l6 center-align"> | ||||||
| @@ -43,6 +55,22 @@ | |||||||
| 	    <a href="{{ url_for('channel.channel', channelId='UCzGdxkzULCa9RlD-Q2EZPXQ') }}"><span class="title">Kalashnikov Group</span></a> | 	    <a href="{{ url_for('channel.channel', channelId='UCzGdxkzULCa9RlD-Q2EZPXQ') }}"><span class="title">Kalashnikov Group</span></a> | ||||||
|         <p>Reason: This account has been terminated for a violation of YouTube's Terms of Service.</p> |         <p>Reason: This account has been terminated for a violation of YouTube's Terms of Service.</p> | ||||||
| 	  </li> | 	  </li> | ||||||
|  | 	  <li class="collection-item"> | ||||||
|  | 	    <a href="{{ url_for('channel.channel', channelId='UCtfg1tENiu3SgGMZVduFmTg') }}"><span class="title">FiberNinja</span></a> | ||||||
|  |         <p>Reason: This channel was removed because it violated our Community Guidelines.</p> | ||||||
|  | 	  </li> | ||||||
|  | 	  <li class="collection-item"> | ||||||
|  | 	    <a href="{{ url_for('channel.channel', channelId='UCv4VkfbX8YfqodF-4coEEfQ') }}"><span class="title">James Somerton</span></a> | ||||||
|  |         <p>Reason: This channel does not exist.</p> | ||||||
|  | 	  </li> | ||||||
|  | 	  <li class="collection-item"> | ||||||
|  | 	    <a href="{{ url_for('channel.channel', channelId='UC8XH9kpilkuss4bVeRZD1kw') }}"><span class="title">Plagued Moth</span></a> | ||||||
|  |         <p>Reason: This channel was removed because it violated our Community Guidelines.</p> | ||||||
|  | 	  </li> | ||||||
|  | 	  <li class="collection-item"> | ||||||
|  | 	    <a href="{{ url_for('channel.channel', channelId='UCxZTTWP0QN7-ch2wW1QeFwg') }}"><span class="title">CowOfTheSea</span></a> | ||||||
|  |         <p>Reason: This channel was removed because it violated our Community Guidelines.</p> | ||||||
|  | 	  </li> | ||||||
| 	</ul> | 	</ul> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -9,14 +9,18 @@ | |||||||
|     <p>Searching is currently partially working and will probably not work optimally for a long time until the database and backend is fully reworked.</p> |     <p>Searching is currently partially working and will probably not work optimally for a long time until the database and backend is fully reworked.</p> | ||||||
|     <p>In the meantime if you know the channel name and video title you can use local search on <a href="{{ url_for('channel.base') }}">this</a> page</p> |     <p>In the meantime if you know the channel name and video title you can use local search on <a href="{{ url_for('channel.base') }}">this</a> page</p> | ||||||
|     <img class="responsive-img" src="{{ url_for('static', filename='img/mongo_meme.png') }}"> |     <img class="responsive-img" src="{{ url_for('static', filename='img/mongo_meme.png') }}"> | ||||||
| 	{% if stats is defined %} |     {% if stats is not none and stats is defined %} | ||||||
|     <div class="divider"></div> |     <div class="divider"></div> | ||||||
|     <h5>Stats of the archive</h5> |     <h5>Stats of the archive</h5> | ||||||
|     <ul class="collection"> |     <ul class="collection"> | ||||||
|       {% for stat in stats %} |       {% for stat in stats %} | ||||||
|       <li class="collection-item"> |       <li class="collection-item"> | ||||||
| 	    <span class="title">{{ stat }}</span> |         <!--<span class="title">{{ stat }}</span>--> | ||||||
| 		<p>{{ stats[stat] }}</p> |         {% if stat == 'last_updated' %} | ||||||
|  |         Last updated {{ stats[stat]|datetime_date }} UTC | ||||||
|  |         {% else %} | ||||||
|  |         {{ stats[stat] }} | ||||||
|  |         {% endif %} | ||||||
|       </li> |       </li> | ||||||
|       {% endfor %} |       {% endfor %} | ||||||
|     </ul> |     </ul> | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
| <meta property="og:title" content="{{ render.get('info').get('title') }}" /> | <meta property="og:title" content="{{ render.get('info').get('title') }}" /> | ||||||
| <meta property="og:type" content="website" /> | <meta property="og:type" content="website" /> | ||||||
| <meta property="og:url" content="{{ url_for('watch.base') }}?v={{ render.get('info').get('id') }}" /> | <meta property="og:url" content="{{ url_for('watch.base') }}?v={{ render.get('info').get('id') }}" /> | ||||||
| <meta property="og:image" content="https://archive.ventilaar.net/videos/automatic/{{ render.get('info').get('channel_id') }}/{{ render.get('info').get('id') }}/{{ render.get('info').get('title') }}.jpg" /> | <meta property="og:image" content="https://archive.ventilaar.net/videos/automatic/{{ render.get('info').get('channel_id') }}/{{ render.get('info').get('id') }}/{{ render.get('info').get('_title_slug') }}.jpg" /> | ||||||
| <meta property="og:description" content="{{ render.get('info').get('description', '')|truncate(100) }}" /> | <meta property="og:description" content="{{ render.get('info').get('description', '')|truncate(100) }}" /> | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
| @@ -14,8 +14,8 @@ | |||||||
| <div class="row"> | <div class="row"> | ||||||
|   <div class="col s12 mt-4 center-align"> |   <div class="col s12 mt-4 center-align"> | ||||||
|     <video controls class="responsive-video"> |     <video controls class="responsive-video"> | ||||||
|     <source src="https://archive.ventilaar.net/videos/automatic/{{ render.get('info').get('channel_id') }}/{{ render.get('info').get('id') }}/{{ render.get('info').get('title_slug') }}.mp4"> |     <source src="https://archive.ventilaar.net/videos/automatic/{{ render.get('info').get('channel_id') }}/{{ render.get('info').get('id') }}/{{ render.get('info').get('_title_slug') }}.mp4"> | ||||||
|     <source src="https://archive.ventilaar.net/videos/automatic/{{ render.get('info').get('channel_id') }}/{{ render.get('info').get('id') }}/{{ render.get('info').get('title_slug') }}.webm"> |     <source src="https://archive.ventilaar.net/videos/automatic/{{ render.get('info').get('channel_id') }}/{{ render.get('info').get('id') }}/{{ render.get('info').get('_title_slug') }}.webm"> | ||||||
|     Your browser does not support the video tag. |     Your browser does not support the video tag. | ||||||
|     </video> |     </video> | ||||||
|   </div> |   </div> | ||||||
| @@ -27,7 +27,7 @@ | |||||||
|   <div class="col s12 l3"> |   <div class="col s12 l3"> | ||||||
|     <p><b>Video by:</b> <a href="{{ url_for('channel.channel', channelId=render.get('info').get('channel_id')) }}">{{ render.get('info').get('uploader') }}</a></p> |     <p><b>Video by:</b> <a href="{{ url_for('channel.channel', channelId=render.get('info').get('channel_id')) }}">{{ render.get('info').get('uploader') }}</a></p> | ||||||
|     <p><b>Upload date:</b> {{ render.get('info').get('upload_date')|pretty_time }}</p> |     <p><b>Upload date:</b> {{ render.get('info').get('upload_date')|pretty_time }}</p> | ||||||
|     <p><b>Archive date:</b> {{ render.get('info').get('epoch')|epoch_time }}</p> |     <p><b>Archive date:</b> {{ render.get('info').get('epoch')|epoch_date }}</p> | ||||||
|     <p><b>Video length:</b> {{ render.get('info').get('duration')|pretty_duration }}</p> |     <p><b>Video length:</b> {{ render.get('info').get('duration')|pretty_duration }}</p> | ||||||
|   </div> |   </div> | ||||||
|   <div class="col s4 l3 center-align"> |   <div class="col s4 l3 center-align"> | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								one_offs/add_failed_queue_to_unavailable.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								one_offs/add_failed_queue_to_unavailable.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | from ayta.nosql import Mango | ||||||
|  | #import ayta | ||||||
|  | #app = ayta.create_app() | ||||||
|  | mango = Mango('mongodb://root:example@192.168.66.140:27017') | ||||||
|  |  | ||||||
|  | data = mango.download_queue.find({'status': 'failed'}) | ||||||
|  | for x in data: | ||||||
|  |     vId = x['id'] | ||||||
|  |     lines = x['fail_reason'].splitlines() | ||||||
|  |     error = lines[-1] | ||||||
|  |     check = "This video has been removed for violating YouTube's Terms of Service" | ||||||
|  |      | ||||||
|  |     if check in error: | ||||||
|  |         print(vId) | ||||||
|  |         mango.info_json.insert_one({'id': vId, '_status': 'unavailable',  | ||||||
|  |         '_status_description': f'Video is unavailable because YouTube said: {check}'}) | ||||||
|  |         mango.queue_deleteQueue(vId) | ||||||
|  |     else: | ||||||
|  |         print(error) | ||||||
|  | print('done') | ||||||
							
								
								
									
										18
									
								
								one_offs/archive_size.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								one_offs/archive_size.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | from ayta.nosql import Mango | ||||||
|  | #import ayta | ||||||
|  | #app = ayta.create_app() | ||||||
|  | mango = Mango('mongodb://root:example@192.168.66.140:27017') | ||||||
|  |  | ||||||
|  | data = mango.info_json.find({'_status': 'available'}, {'filesize_approx': 1}) | ||||||
|  |  | ||||||
|  | total = 0 | ||||||
|  |  | ||||||
|  | for x in data: | ||||||
|  |     size = x.get('filesize_approx') | ||||||
|  |     if size: | ||||||
|  |         total = total + int(size) | ||||||
|  |  | ||||||
|  | # the 5000 is the amount of GB of unjust approximation | ||||||
|  | total = int(total / 1000000000 + 5000) | ||||||
|  |  | ||||||
|  | print(f'Approximate size: {total} GB') | ||||||
							
								
								
									
										37
									
								
								one_offs/stats_downloads_per_day.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								one_offs/stats_downloads_per_day.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | from ayta.nosql import Mango | ||||||
|  | import matplotlib.pyplot as plt | ||||||
|  | from datetime import datetime, timedelta | ||||||
|  | #import ayta | ||||||
|  | #app = ayta.create_app() | ||||||
|  | mango = Mango('mongodb://root:example@192.168.66.140:27017') | ||||||
|  |  | ||||||
|  | pivot = datetime.utcnow() - timedelta(days=90) | ||||||
|  | pivot = int(pivot.timestamp()) | ||||||
|  |  | ||||||
|  | data = mango.info_json.find({'_status': 'available', 'timestamp': {'$gt': pivot}}, {'epoch': 1}) | ||||||
|  |  | ||||||
|  | stat = {} | ||||||
|  |  | ||||||
|  | for x in data: | ||||||
|  |     epoch = x['epoch'] | ||||||
|  |     day = datetime.fromtimestamp(epoch).strftime('%Y%m%d') | ||||||
|  |      | ||||||
|  |     if day not in stat: | ||||||
|  |         stat[day] = 1 | ||||||
|  |     else: | ||||||
|  |         stat[day] = stat[day] + 1 | ||||||
|  |  | ||||||
|  | dates = list(stat.keys()) | ||||||
|  | values = list(stat.values()) | ||||||
|  |  | ||||||
|  | plt.figure(figsize=(16, 8))  # Set the figure size | ||||||
|  | plt.bar(dates, values)       # Create the bar chart | ||||||
|  |  | ||||||
|  | # Customize the x-axis labels to be vertical | ||||||
|  | plt.xticks(rotation=45, ha='right')  # Rotate xticklabels by 45 degrees and align them to the right | ||||||
|  | plt.xlabel('Date')           # Label for x-axis | ||||||
|  | plt.ylabel('Counts')         # Label for y-axis | ||||||
|  | plt.title('Bar Graph of Counts by Date')  # Title of the graph | ||||||
|  |  | ||||||
|  | # Display the graph | ||||||
|  | plt.show() | ||||||
							
								
								
									
										35
									
								
								one_offs/stats_uploads_per_day.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								one_offs/stats_uploads_per_day.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | |||||||
|  | from ayta.nosql import Mango | ||||||
|  | import matplotlib.pyplot as plt | ||||||
|  | from datetime import datetime, timedelta | ||||||
|  | #import ayta | ||||||
|  | #app = ayta.create_app() | ||||||
|  | mango = Mango('mongodb://root:example@192.168.66.140:27017') | ||||||
|  |  | ||||||
|  | pivot = '20220101' | ||||||
|  |  | ||||||
|  | data = mango.info_json.find({'_status': 'available', 'upload_date': {'$gt': pivot}}, {'upload_date': 1}) | ||||||
|  |  | ||||||
|  | stat = {} | ||||||
|  |  | ||||||
|  | for x in data: | ||||||
|  |     day = x['upload_date'] | ||||||
|  |      | ||||||
|  |     if day not in stat: | ||||||
|  |         stat[day] = 1 | ||||||
|  |     else: | ||||||
|  |         stat[day] = stat[day] + 1 | ||||||
|  |  | ||||||
|  | dates = list(stat.keys()) | ||||||
|  | values = list(stat.values()) | ||||||
|  |  | ||||||
|  | plt.figure(figsize=(16, 8))  # Set the figure size | ||||||
|  | plt.bar(dates, values)       # Create the bar chart | ||||||
|  |  | ||||||
|  | # Customize the x-axis labels to be vertical | ||||||
|  | plt.xticks(rotation=45, ha='right')  # Rotate xticklabels by 45 degrees and align them to the right | ||||||
|  | plt.xlabel('Date')           # Label for x-axis | ||||||
|  | plt.ylabel('Counts')         # Label for y-axis | ||||||
|  | plt.title('Bar Graph of Counts by Date')  # Title of the graph | ||||||
|  |  | ||||||
|  | # Display the graph | ||||||
|  | plt.show() | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user