auto-archiver/auto_archive.py

189 wiersze
7.4 KiB
Python
Czysty Zwykły widok Historia

import os
2022-02-21 13:19:09 +00:00
import datetime
import argparse
import requests
2022-02-23 15:43:42 +00:00
import shutil
2022-02-21 13:19:09 +00:00
import gspread
from loguru import logger
2022-02-21 13:19:09 +00:00
from dotenv import load_dotenv
2022-02-25 15:09:35 +00:00
from selenium import webdriver
import traceback
2022-02-21 13:19:09 +00:00
import archivers
from storages import S3Storage, S3Config
2022-05-11 13:01:22 +00:00
from storages.gd_storage import GDConfig, GDStorage
2022-02-23 15:32:38 +00:00
from utils import GWorksheet, mkdir_if_not_exists
2022-05-09 10:55:10 +00:00
import sys
2022-05-09 10:55:10 +00:00
logger.add("logs/1trace.log", level="TRACE")
logger.add("logs/2info.log", level="INFO")
logger.add("logs/3success.log", level="SUCCESS")
logger.add("logs/4warning.log", level="WARNING")
logger.add("logs/5error.log", level="ERROR")
2022-05-09 11:02:43 +00:00
load_dotenv()
2021-05-03 12:16:09 +00:00
2022-02-23 08:54:03 +00:00
def update_sheet(gw, row, result: archivers.ArchiveResult):
cell_updates = []
row_values = gw.get_row(row)
2021-05-03 12:16:09 +00:00
2022-02-23 08:54:03 +00:00
def batch_if_valid(col, val, final_value=None):
final_value = final_value or val
if val and gw.col_exists(col) and gw.get_cell(row_values, col) == '':
cell_updates.append((row, col, final_value))
2021-03-18 10:03:13 +00:00
cell_updates.append((row, 'status', result.status))
2021-03-15 09:08:02 +00:00
2022-02-23 08:54:03 +00:00
batch_if_valid('archive', result.cdn_url)
batch_if_valid('date', True, datetime.datetime.utcnow().replace(tzinfo=datetime.timezone.utc).isoformat())
2022-02-25 15:09:35 +00:00
batch_if_valid('thumbnail', result.thumbnail,
f'=IMAGE("{result.thumbnail}")')
2022-02-23 08:54:03 +00:00
batch_if_valid('thumbnail_index', result.thumbnail_index)
batch_if_valid('title', result.title)
batch_if_valid('duration', result.duration, str(result.duration))
2022-02-25 15:09:35 +00:00
batch_if_valid('screenshot', result.screenshot)
batch_if_valid('hash', result.hash)
if result.timestamp is not None:
if type(result.timestamp) == int:
timestamp_string = datetime.datetime.fromtimestamp(result.timestamp).replace(tzinfo=datetime.timezone.utc).isoformat()
elif type(result.timestamp) == str:
timestamp_string = result.timestamp
else:
timestamp_string = result.timestamp.isoformat()
batch_if_valid('timestamp', timestamp_string)
2021-03-15 09:08:02 +00:00
gw.batch_set_cell(cell_updates)
def expand_url(url):
# expand short URL links
if 'https://t.co/' in url:
try:
r = requests.get(url)
url = r.url
except:
logger.error(f'Failed to expand url {url}')
return url
2021-03-15 09:08:02 +00:00
2021-06-01 09:00:40 +00:00
def process_sheet(sheet, header=1, columns=GWorksheet.COLUMN_NAMES):
2021-06-01 09:05:13 +00:00
gc = gspread.service_account(filename='service_account.json')
2021-03-25 12:42:42 +00:00
sh = gc.open(sheet)
s3_config = S3Config(
bucket=os.getenv('DO_BUCKET'),
region=os.getenv('DO_SPACES_REGION'),
key=os.getenv('DO_SPACES_KEY'),
secret=os.getenv('DO_SPACES_SECRET')
)
2022-03-15 17:45:53 +00:00
telegram_config = archivers.TelegramConfig(
api_id=os.getenv('TELEGRAM_API_ID'),
api_hash=os.getenv('TELEGRAM_API_HASH')
)
2022-02-25 15:09:35 +00:00
# loop through worksheets to check
for ii, wks in enumerate(sh.worksheets()):
2022-05-09 14:59:35 +00:00
logger.info(f'Opening worksheet {ii=}: {wks.title=} {header=}')
gw = GWorksheet(wks, header_row=header, columns=columns)
2021-03-15 09:08:02 +00:00
2022-02-23 08:57:04 +00:00
if not gw.col_exists('url'):
2022-02-25 15:09:35 +00:00
logger.warning(
f'No "{columns["url"]}" column found, skipping worksheet {wks.title}')
2021-03-25 12:42:42 +00:00
continue
2022-02-23 08:57:04 +00:00
if not gw.col_exists('status'):
2022-02-25 15:09:35 +00:00
logger.warning(
f'No "{columns["status"]}" column found, skipping worksheet {wks.title}')
2021-03-25 12:42:42 +00:00
continue
# archives will be in a folder 'doc_name/worksheet_name'
2022-02-25 15:09:35 +00:00
s3_config.folder = f'{sheet.replace(" ", "_")}/{wks.title.replace(" ", "_")}/'
s3_client = S3Storage(s3_config)
2022-02-21 13:19:09 +00:00
# order matters, first to succeed excludes remaining
active_archivers = [
2022-03-15 17:45:53 +00:00
archivers.TelethonArchiver(s3_client, driver, telegram_config),
2022-02-25 15:09:35 +00:00
archivers.TelegramArchiver(s3_client, driver),
archivers.TiktokArchiver(s3_client, driver),
archivers.YoutubeDLArchiver(s3_client, driver, os.getenv('FACEBOOK_COOKIE')),
2022-02-25 15:09:35 +00:00
archivers.TwitterArchiver(s3_client, driver),
archivers.WaybackArchiver(s3_client, driver)
]
# loop through rows in worksheet
2022-02-25 15:09:35 +00:00
for row in range(1 + header, gw.count_rows() + 1):
url = gw.get_cell(row, 'url')
original_status = gw.get_cell(row, 'status')
2022-03-14 10:10:51 +00:00
status = gw.get_cell(row, 'status', fresh=original_status in ['', None] and url != '')
2022-02-23 08:54:03 +00:00
if url != '' and status in ['', None]:
2022-03-09 10:46:14 +00:00
gw.set_cell(row, 'status', 'Archive in progress')
url = expand_url(url)
# make a new driver so each spreadsheet row is idempotent
options = webdriver.FirefoxOptions()
options.headless = True
options.set_preference('network.protocol-handler.external.tg', False)
2022-03-09 10:46:14 +00:00
driver = webdriver.Firefox(options=options)
driver.set_window_size(1400, 2000)
# in seconds, telegram screenshots catch which don't come back
driver.set_page_load_timeout(120)
2022-03-09 10:46:14 +00:00
for archiver in active_archivers:
logger.debug(f'Trying {archiver} on row {row}')
try:
result = archiver.download(url, check_if_exists=True)
except Exception as e:
result = False
logger.error(f'Got unexpected error in row {row} with archiver {archiver} for url {url}: {e}\n{traceback.format_exc()}')
if result:
2022-03-09 10:46:14 +00:00
if result.status in ['success', 'already archived']:
result.status = archiver.name + \
": " + str(result.status)
logger.success(
f'{archiver} succeeded on row {row}')
break
logger.warning(
f'{archiver} did not succeed on row {row}, final status: {result.status}')
result.status = archiver.name + \
": " + str(result.status)
# get rid of driver so can reload on next row
driver.quit()
2022-03-09 10:46:14 +00:00
if result:
update_sheet(gw, row, result)
else:
gw.set_cell(row, 'status', 'failed: no archiver')
logger.success(f'Finshed worksheet {wks.title}')
2021-03-15 09:08:02 +00:00
2022-05-09 10:55:10 +00:00
@logger.catch
2021-03-25 12:42:42 +00:00
def main():
2022-05-09 10:55:10 +00:00
logger.debug(f'Passed args:{sys.argv}')
2021-03-25 12:42:42 +00:00
parser = argparse.ArgumentParser(
2022-02-23 08:57:04 +00:00
description='Automatically archive social media videos from a Google Sheets document')
parser.add_argument('--sheet', action='store', dest='sheet', help='the name of the google sheets document', required=True)
parser.add_argument('--header', action='store', dest='header', default=1, type=int, help='1-based index for the header row')
2022-03-18 08:53:21 +00:00
parser.add_argument('--private', action='store_true', help='Store content without public access permission')
for k, v in GWorksheet.COLUMN_NAMES.items():
parser.add_argument(f'--col-{k}', action='store', dest=k, default=v, help=f'the name of the column to fill with {k} (defaults={v})')
2021-03-25 12:42:42 +00:00
args = parser.parse_args()
2022-03-12 19:14:16 +00:00
config_columns = {k: getattr(args, k).lower() for k in GWorksheet.COLUMN_NAMES.keys()}
2021-03-25 12:42:42 +00:00
2022-03-12 18:54:10 +00:00
logger.info(f'Opening document {args.sheet} for header {args.header}')
2021-03-25 12:42:42 +00:00
2022-02-23 15:32:38 +00:00
mkdir_if_not_exists('tmp')
process_sheet(args.sheet, header=args.header, columns=config_columns)
2022-02-23 15:43:42 +00:00
shutil.rmtree('tmp')
2021-06-01 09:00:40 +00:00
2022-02-25 15:09:35 +00:00
2022-02-23 08:57:04 +00:00
if __name__ == '__main__':
main()