commit f895fe5348e61daf4d4801623e78b6830174b2d3 Author: Kevin F Date: Sun Oct 3 00:56:33 2021 +0000 initial commit diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..abcb1cb --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +venv/* +helloworld/__pycache__/* +testdata/* +.vscode/* diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 0000000..7a79fdb --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,7 @@ +# Changelog + +This project uses Semantic Versioning + +## 0.0.0 + +Initial release diff --git a/Makefile b/Makefile new file mode 100644 index 0000000..110e50a --- /dev/null +++ b/Makefile @@ -0,0 +1,2 @@ +test: + ./run_tests.sh diff --git a/imgmine/__init__.py b/imgmine/__init__.py new file mode 100644 index 0000000..c819c8f --- /dev/null +++ b/imgmine/__init__.py @@ -0,0 +1,66 @@ +from gevent import monkey +from gevent import sleep +monkey.patch_all() +from threading import Thread +from os import remove, mkdir, path +from shutil import rmtree +from glob import glob + +from uuid import uuid4 + +from bottle import route, run +from bottle import static_file +from bottle import SimpleTemplate + +from .get import get +from .config import IMAGE_CACHE, SINGLE_IMAGE_DELETE_AFTER_SECS, ALBUM_DELETE_AFTER_SECS, template_dir + + +def album(id): + req_id = str(uuid4()) + req = IMAGE_CACHE + + get("/a/" + id, req) + + imgs = glob(req + "*") + for c, img in enumerate(imgs): + imgs[c] = img.replace(IMAGE_CACHE, '/') + with open(f'{template_dir}gallery.html', 'r') as img_view: + tpl = SimpleTemplate(img_view) + return tpl.render(imgs=imgs) + +@route('/') +@route('') +def home(): + return static_file("index.html", root=template_dir) + +@route('/static/') +def static(file=''): + return static_file(file, root=template_dir) + + +@route('/gallery/') +def gallery(id=''): + return album(id) + +@route('/a/') +def gallery(id=''): + return album(id) + + +@route('/') +def hello(img=''): + + if not path.exists(IMAGE_CACHE + img): + get(img, IMAGE_CACHE) + return static_file(img, root=IMAGE_CACHE) + + +def start_server(): + try: + rmtree(IMAGE_CACHE) + except FileNotFoundError: + pass + mkdir(IMAGE_CACHE) + + run(server='gevent') \ No newline at end of file diff --git a/imgmine/__pycache__/__init__.cpython-39.pyc b/imgmine/__pycache__/__init__.cpython-39.pyc new file mode 100644 index 0000000..f565fba Binary files /dev/null and b/imgmine/__pycache__/__init__.cpython-39.pyc differ diff --git a/imgmine/__pycache__/config.cpython-39.pyc b/imgmine/__pycache__/config.cpython-39.pyc new file mode 100644 index 0000000..eb6652b Binary files /dev/null and b/imgmine/__pycache__/config.cpython-39.pyc differ diff --git a/imgmine/__pycache__/get.cpython-39.pyc b/imgmine/__pycache__/get.cpython-39.pyc new file mode 100644 index 0000000..2ee801b Binary files /dev/null and b/imgmine/__pycache__/get.cpython-39.pyc differ diff --git a/imgmine/config.py b/imgmine/config.py new file mode 100644 index 0000000..5cc0401 --- /dev/null +++ b/imgmine/config.py @@ -0,0 +1,4 @@ +IMAGE_CACHE = '/tmp/imgmine-imgur-images/' +SINGLE_IMAGE_DELETE_AFTER_SECS = 60 +ALBUM_DELETE_AFTER_SECS = 120 +template_dir = 'imgmine/web/' \ No newline at end of file diff --git a/imgmine/get.py b/imgmine/get.py new file mode 100644 index 0000000..574ad74 --- /dev/null +++ b/imgmine/get.py @@ -0,0 +1,64 @@ +import sys +from shutil import rmtree +from os import remove +from threading import Thread + +import requests +import bs4 +from gevent import sleep + +from .config import SINGLE_IMAGE_DELETE_AFTER_SECS, ALBUM_DELETE_AFTER_SECS + +def delete_file(path): + sleep(SINGLE_IMAGE_DELETE_AFTER_SECS) + print('Erasing', path) + try: + remove(path) + except FileNotFoundError: + pass + + +def error(msg): + sys.stderr.write(msg + "\n") + sys.stderr.flush() + +def get(url: str, write_dir: str, delete=True): + if not url.startswith('https://imgur.com/'): + url = 'https://imgur.com/' + url + found_url = '' + + print('it not album', url) + album = False + if "gallery" in url: + url = url.replace("gallery", "a") + print('it album') + if "/a/" in url: + print('it album') + album = True + if not url.endswith("blog"): + url += "/layout/blog" + + if not album: + print('getting img', url) + url = 'https://i.imgur.com/' + url.rsplit('/', 1)[-1].replace('jpeg', '').replace('jpg', '') + with open(f'{write_dir}/{url[-12:]}', 'wb') as img: + img.write(requests.get(url).content) + if delete: + Thread(target=delete_file, args=[f"{write_dir}/{url[-12:]}"]).start() + else: + print('Detecting album/gallery images', url) + soup = bs4.BeautifulSoup(requests.get(url).text, 'html.parser') + for count, el in enumerate(soup.select('.post-image meta[itemprop="contentUrl"]'), start=1): + try: + found_url = "https:" + el['content'] + except KeyError: + error("Could not obtain url for detected image") + continue + print(f"Downloading image {count}: {found_url}") + + print("Writing image", f"{write_dir}{found_url[-11:]}") + with open(f"{write_dir}{found_url[-11:]}", "wb") as f: + f.write(requests.get(found_url).content) + if delete: + Thread(target=delete_file, args=[f"{write_dir}{found_url[-11:]}"]).start() + diff --git a/imgmine/web/gallery.html b/imgmine/web/gallery.html new file mode 100644 index 0000000..51139b7 --- /dev/null +++ b/imgmine/web/gallery.html @@ -0,0 +1,25 @@ + + + + + + + imgmine - minimal & private imgur proxy + + + + + % for img in imgs: + +
+ % end +
+ + This website does not claim ownership of any media. +
This service simply acts as a proxy to Imgur.com and does not store images aside from a temporary cache. +
Abusive images should be reported to imgur. This website does not create new images/comments. +
+
+ + + \ No newline at end of file diff --git a/imgmine/web/index.html b/imgmine/web/index.html new file mode 100644 index 0000000..19c56fe --- /dev/null +++ b/imgmine/web/index.html @@ -0,0 +1,41 @@ + + + + + + + imgmine - minimal & private imgur proxy + + + + +

+ imgmine +

+
+

+ A minimalist read-only Imgur proxy insipired by software like Invidious, Nitter, and Bibliogram. +

+

+ This project was started because Kevin Froman got fed up with Imgur breaking with Tor because of cow-dung JS. +

+

Feature roadmap

+

Features are in order of priority of most to least

+
    +
  • Image proxying
  • +
  • Gallery/album proxying
  • +
  • Proper order of gallery posts
  • +
  • Image description and author info
  • +
  • GIFs/videos
  • +
  • Comments... maybe
  • +
+ + + + \ No newline at end of file diff --git a/imgmine/web/theme.css b/imgmine/web/theme.css new file mode 100644 index 0000000..a59cdb8 --- /dev/null +++ b/imgmine/web/theme.css @@ -0,0 +1,14 @@ +body{ + background-color: rgba(8, 6, 37, 0.781); + color: white; + font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif; + margin-left: 5%; +} + +a, a:visited { + color: white; +} + +ul{ + list-style-type: none; +} \ No newline at end of file diff --git a/run.py b/run.py new file mode 100644 index 0000000..834f28a --- /dev/null +++ b/run.py @@ -0,0 +1,3 @@ +import imgmine + +imgmine.start_server() \ No newline at end of file diff --git a/run_tests.sh b/run_tests.sh new file mode 100755 index 0000000..ba03150 --- /dev/null +++ b/run_tests.sh @@ -0,0 +1,13 @@ +#!/bin/bash +ran=0 +SECONDS=0 ; +close () { + exit 10; +} + +for f in tests/*.py; do + python3 "$f" || close # if needed + let "ran++" +done +echo "ran $ran test files successfully in $SECONDS seconds" +rm -f *.dat diff --git a/tests/test.py b/tests/test.py new file mode 100644 index 0000000..0d6f79c --- /dev/null +++ b/tests/test.py @@ -0,0 +1,7 @@ +import unittest +class TestBasic(unittest.TestCase): + + def test_basic(self): + self.assertTrue(True) + +unittest.main() \ No newline at end of file