43 Commits

Author SHA1 Message Date
Sweetbread 7cf4ed3ecd Add own 88x31
Docker Build and Push / build-and-push (push) Successful in 24s
(and rename icon->img, btw...)
2026-04-09 00:32:33 +03:00
Sweetbread 3a47b84a1b Use Pug template generator
Docker Build and Push / build-and-push (push) Successful in 1m15s
2026-04-08 23:03:37 +03:00
Sweetbread 4844cdb7b6 Fix API calls by all workers
Quite a complex commit by GPT. I'll rewrite it somewhen
2026-04-08 22:55:27 +03:00
Sweetbread 6c22740fbd Update info 2026-03-07 18:45:24 +03:00
Sweetbread a0cfa765bd Cover replacing for LB
Docker Build and Push / build-and-push (push) Successful in 20s
2026-02-07 19:25:12 +03:00
Sweetbread 62b8f056fa Fix covers
Docker Build and Push / build-and-push (push) Successful in 20s
2026-02-07 18:39:58 +03:00
Sweetbread 0edf6e35d7 Reactive updating of related time 2026-02-07 18:39:58 +03:00
Sweetbread db11fabe1a Refactoring 2026-02-07 18:39:58 +03:00
Sweetbread 0f5d1b5221 Don't send data if not modified 2026-02-06 18:56:38 +03:00
Sweetbread 9935bc7f1f Add polling
Docker Build and Push / build-and-push (push) Successful in 20s
2026-02-05 23:04:19 +03:00
Sweetbread 42ffd887d2 Update docker image 2026-02-05 23:04:19 +03:00
Sweetbread 96b73e4751 Add OG meta 2026-02-05 23:04:19 +03:00
Sweetbread 0aa2477f35 Add Steam info 2026-02-05 23:04:19 +03:00
Sweetbread c17bf8ca83 Delete otoring >: 2026-02-04 15:31:04 +03:00
Sweetbread ea5156aa6c Rework my page and add info from listenbrainz 2026-02-04 15:31:04 +03:00
Sweetbread e97985b624 Update /us
Docker Build and Push / build-and-push (push) Successful in 32s
2026-01-19 01:02:54 +03:00
Sweetbread 3e2eb74c76 Add 88x31 2026-01-19 00:39:01 +03:00
Sweetbread 8c02395d0d Add otoring
Docker Build and Push / build-and-push (push) Successful in 34s
2026-01-18 19:02:48 +03:00
Sweetbread 60bece505c Update footer 2026-01-18 19:00:32 +03:00
Sweetbread c56f88911c Remove private sites 2026-01-18 19:00:32 +03:00
Sweetbread b28f7886e1 wip: update my page 2026-01-18 19:00:32 +03:00
Sweetbread 335eaa0545 Move locale things into module 2026-01-17 23:51:32 +03:00
Sweetbread 4b2720f8b7 Put personal pages into subdomains 2026-01-17 23:51:32 +03:00
Sweetbread aedd47ba36 Change (temporary?) icon 2026-01-17 23:51:32 +03:00
Sweetbread e1f4021ed5 Add mock email 2026-01-13 16:10:47 +03:00
Sweetbread c806dfff6c Change to new domain 2026-01-13 16:10:47 +03:00
Sweetbread 000465079b Add altfronts block 2026-01-13 16:10:47 +03:00
Sweetbread 97280e2d7c ci: remove per commit images 2026-01-13 16:10:47 +03:00
Sweetbread 25edd56306 Remove typos in host 2026-01-13 16:10:47 +03:00
Sweetbread e776f09b0f Remove whitespaces before sending 2026-01-13 16:10:47 +03:00
Sweetbread c0dc079a71 Add some services 2026-01-13 14:58:10 +03:00
Sweetbread 0a5807e62e Change host info 2026-01-13 14:58:10 +03:00
Sweetbread f60a1940ac Improve style
- Add more depth to .block
- Make footer lighter
- Stylize scrollbar
2026-01-13 14:58:10 +03:00
Sweetbread 7351e28bcd Add mine page 2026-01-13 14:58:10 +03:00
Sweetbread 74afee8f5b style: change sizes to rem 2026-01-13 14:58:10 +03:00
Sweetbread 5f417b433e style: decrease margins 2026-01-13 14:58:10 +03:00
Sweetbread 39d6f469cb feat: add tracker script 2026-01-13 14:58:10 +03:00
Sweetbread 78f658241c update host specs 2026-01-13 14:58:10 +03:00
Sweetbread 4622e730c1 l10n: add en, de, ja and fr 2026-01-13 14:58:10 +03:00
Sweetbread e6ba705862 sec: add production server 2026-01-13 14:58:09 +03:00
Sweetbread cfd81884db ci: add docker-build 2026-01-13 14:58:09 +03:00
Sweetbread 8c75448836 feat: add dockerfile 2026-01-13 14:58:09 +03:00
Sweetbread f5afca83d2 Init commit 2026-01-13 14:58:09 +03:00
72 changed files with 1447 additions and 422 deletions
+9 -4
View File
@@ -11,10 +11,16 @@ jobs:
- name: Checkout code - name: Checkout code
uses: actions/checkout@v4 uses: actions/checkout@v4
- name: 'Docker Tags'
id: tags
uses: cssnr/docker-tags-action@v2
with:
images: 'g.lair.moe/${{ vars.DOCKER_USERNAME }}/lair.moe'
- name: Login to Docker Registry - name: Login to Docker Registry
uses: docker/login-action@v2 uses: docker/login-action@v2
with: with:
registry: g.codrs.ru registry: g.lair.moe
username: ${{ vars.DOCKER_USERNAME }} username: ${{ vars.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }} password: ${{ secrets.DOCKER_PASSWORD }}
@@ -26,8 +32,7 @@ jobs:
with: with:
context: . context: .
push: ${{ github.event_name == 'push' }} push: ${{ github.event_name == 'push' }}
tags: | tags: g.lair.moe/${{ vars.DOCKER_USERNAME }}/lair.moe:latest
g.codrs.ru/${{ vars.DOCKER_USERNAME }}/codrs.ru:latest labels: ${{ steps.tags.outputs.labels }}
g.codrs.ru/${{ vars.DOCKER_USERNAME }}/codrs.ru:${{ github.sha }}
cache-from: type=gha cache-from: type=gha
cache-to: type=gha,mode=max cache-to: type=gha,mode=max
+2 -2
View File
@@ -5,5 +5,5 @@ __pycache__/
*$py.class *$py.class
.python-version .python-version
static/style/*.css *.css
static/style/*.css.map *.css.map
+36 -12
View File
@@ -1,23 +1,47 @@
FROM node:18-alpine as sass FROM node:18-alpine AS sass-builder
RUN NODE_OPTIONS=--dns-result-order=ipv4first npm install -g sass RUN NODE_OPTIONS=--dns-result-order=ipv4first npm install -g sass@latest --omit=dev --no-fund --no-audit
WORKDIR /build WORKDIR /build
COPY ./static/style ./style COPY ./blueprints ./blueprints
RUN sass ./style:./style \
--no-source-map \
--style=compressed
RUN sass ./blueprints:./blueprints \
--no-source-map \
--style=compressed \
--quiet
FROM python:3.11-slim FROM python:3.11-slim
RUN apt-get update && \
apt-get install --no-install-recommends -y \
libmagic1 \
git \
&& rm -rf /var/lib/apt/lists/*
WORKDIR /app WORKDIR /app
COPY . . COPY requirements.txt .
COPY --from=sass /build/style/ ./static/style/
RUN pip install --no-cache-dir -r requirements.txt RUN pip install --no-cache-dir -r requirements.txt
ENV FLASK_ENV=production COPY . .
ENV PYTHONUNBUFFERED=1
CMD ["gunicorn", "app:app", "-b", "0.0.0.0:80", "--workers", "4"] COPY --from=sass-builder /build/blueprints/ ./blueprints/
RUN useradd -m -u 1001 appuser && \
chown -R appuser:appuser /app
USER appuser
ENV FLASK_ENV=production \
PYTHONUNBUFFERED=1 \
PYTHONDONTWRITEBYTECODE=1 \
PIP_NO_CACHE_DIR=1 \
PIP_DISABLE_PIP_VERSION_CHECK=1
CMD ["gunicorn", "app:app", \
"-b", "0.0.0.0:80", \
"--workers", "4", \
"--worker-class", "sync", \
"--worker-tmp-dir", "/dev/shm", \
"--access-logfile", "-", \
"--error-logfile", "-", \
"--log-level", "info"]
+26 -70
View File
@@ -1,78 +1,34 @@
from os import system as console from modules import locale
from configparser import ConfigParser
from flask import ( from blueprints.root import bp as root_bp
Flask, from blueprints.risdeveau import bp as rdv_bp
g,
request, import blueprints.root.modules.style
render_template, import blueprints.risdeveau.modules.style
from flask import Flask
app = Flask(
__name__,
static_folder=None,
subdomain_matching=True,
template_folder="blueprints"
) )
app.jinja_env.add_extension('pypugjs.ext.jinja.PyPugJSExtension')
translations_cache = {} app.before_request(locale.before_request)
app.context_processor(locale.inject_translations)
def load_translations(lang): app.register_blueprint(root_bp)
if lang not in translations_cache: app.register_blueprint(rdv_bp)
translations_cache[lang] = {}
try:
config = ConfigParser()
config.read(f'locale/{lang}.ini')
for section in config.sections():
translations_cache[lang][section] = dict(config.items(section))
except:
pass
return translations_cache[lang]
def get_locale():
return request.accept_languages.best_match(('en', 'ru', 'de', 'fr', 'ja'), 'en')
app = Flask(__name__)
@app.before_request
def before_request():
g.locale = get_locale()
g.translations = load_translations(g.locale)
@app.context_processor
def inject_translations():
def translate(text, **kwargs):
if ":" in text:
section, key = text.split(":", 1)
else:
section, key = "common", text
template = g.translations \
.get(section, {}) \
.get(key, f"${section}: {key}$")
try:
return template.format(**kwargs)
except:
return template
return {'_': translate}
if app.debug: if app.debug:
console("sass static/style/main.scss static/style/main.css") blueprints.root.modules.style.compile_styles()
console("sass static/style/risdeveau.scss static/style/risdeveau.css") blueprints.risdeveau.modules.style.compile_styles()
@app.route("/") app.config['SERVER_NAME'] = "localhost:5000"
def index(): else:
return render_template('index.html') app.config['SERVER_NAME'] = "lair.moe"
@app.route("/host")
def host():
return render_template('host.html')
@app.route("/us")
def us():
return render_template('us.html')
@app.route("/risdeveau")
def risdeveau():
return render_template('personal/risdeveau.html')
+106
View File
@@ -0,0 +1,106 @@
import os
from datetime import datetime, timedelta
from pathlib import Path
from time import time
from flask import (
Blueprint,
abort,
make_response,
render_template,
request,
send_file,
)
import magic
from htmlmin import minify
from musicbrainzngs import get_image_front
from .modules.api.lb import data as lb_data, refresh_cache as lb_refresh
from .modules.api.steam import data as steam_data, refresh_cache as steam_refresh
def tmsmp(sec: int) -> str:
if sec == 0:
return 0
elif sec < 60:
return f"{sec} s"
elif sec < 60*60:
minutes = round(sec / 60, 1)
return f"{minutes:.0f} m" if minutes.is_integer() else f"{minutes:.1f} m"
elif sec < 60*60*24:
hours = round(sec / 3600, 1)
return f"{hours:.0f} h" if hours.is_integer() else f"{hours:.1f} h"
else:
days = round(sec / 86400, 1)
return f"{days:.0f} d" if days.is_integer() else f"{days:.1f} d"
def utmsmp(unix: int) -> str:
return datetime \
.utcfromtimestamp(unix) \
.strftime('%Y-%m-%d %H:%M:%S')
def rtmsmp(unix: int) -> str:
return tmsmp(int(time() - unix))
bp = Blueprint(
"risdeveau",
__name__,
subdomain="risdeveau",
template_folder="..",
static_folder=None
)
def render_tmpl(filename: str, **kwargs) -> str:
template_path = os.path.join("risdeveau/templates", filename)
return minify(
render_template(template_path, **kwargs),
remove_empty_space=True
)
@bp.route("/static/<path:filename>")
def static(filename: str):
static_folders = ("static", "../root/static")
for dir in static_folders:
if os.path.exists(path := os.path.join("blueprints/risdeveau", dir, filename)):
return send_file(path)
return abort(404)
@bp.route("/asset/mb/<mbid>")
def mb_cover(mbid):
r = make_response(image := get_image_front(mbid, "250"))
r.headers['Content-Type'] = magic.from_buffer(image[:2048], mime=True)
r.headers['Cache-Control'] = 'public, max-age=86400'
r.headers['Expires'] = (datetime.now() + timedelta(days=1)) \
.strftime('%a, %d %b %Y %H:%M:%S GMT')
return r
args = {
"lb": lb_data,
"steam": steam_data,
"tmsmp": tmsmp,
"utmsmp": utmsmp,
"rtmsmp": rtmsmp
}
@bp.route("/")
def index():
lb_refresh()
steam_refresh()
return render_tmpl('index.pug', **args)
@bp.route("/m/<module>")
def module(module):
if module == "listenbrainz":
lb_refresh()
elif module == "steam":
steam_refresh()
if none_match := request.headers.get('if-none-match'):
match module:
case "listenbrainz":
if none_match == lb_data['etag']:
return '', 304
case "steam":
if none_match == steam_data['etag']:
return '', 304
return render_tmpl(f'{module}.pug', **args)
+193
View File
@@ -0,0 +1,193 @@
import atexit
import re
from dataclasses import dataclass
from hashlib import md5
from json import dumps
from time import time
from urllib.parse import parse_qs, urlparse
from apscheduler.schedulers.background import BackgroundScheduler
from apscheduler.triggers.interval import IntervalTrigger
import requests
from flask import Flask, jsonify
from .scheduler_guard import should_start_scheduler
from .shared_cache import atomic_write_json, cache_file, load_json_if_newer
@dataclass
class Cache:
data = {}
last_updated = time()
status = None
data = {
"caches": {
"now": Cache(),
"listens": Cache()
},
"last_updated": time(),
"etag": ""
}
_IS_WRITER = should_start_scheduler()
_CACHE_PATH = cache_file("listenbrainz")
_CACHE_MTIME = 0.0
def refresh_cache() -> None:
"""Refresh in-memory cache from the shared JSON file (for non-writer workers)."""
global _CACHE_MTIME
if _IS_WRITER:
return
payload, mtime = load_json_if_newer(_CACHE_PATH, _CACHE_MTIME)
if payload is None:
return
try:
data["etag"] = payload.get("etag", data.get("etag", ""))
data["last_updated"] = payload.get("last_updated", data.get("last_updated", time()))
caches = payload.get("caches", {})
for key, cache in data.get("caches", {}).items():
if isinstance(caches, dict) and key in caches and isinstance(caches[key], dict):
c = caches[key]
cache.data = c.get("data", cache.data)
cache.last_updated = c.get("last_updated", cache.last_updated)
cache.status = c.get("status", cache.status)
finally:
_CACHE_MTIME = mtime
def _persist_cache() -> None:
"""Persist current cache state to the shared JSON file (writer worker only)."""
if not _IS_WRITER:
return
payload = {
"etag": data.get("etag", ""),
"last_updated": data.get("last_updated", time()),
"caches": {
k: {
"data": v.data,
"last_updated": v.last_updated,
"status": v.status,
}
for k, v in data.get("caches", {}).items()
},
}
atomic_write_json(_CACHE_PATH, payload)
def yt_cover(youtube_url):
parsed_url = urlparse(youtube_url)
if parsed_url.netloc in ("youtube.com", "music.youtube.com"):
query_params = parse_qs(parsed_url.query)
video_id = query_params.get('v', [None])[0]
elif parsed_url.netloc == 'youtu.be':
video_id = parsed_url.path[1:]
if not video_id:
return
return f"https://img.youtube.com/vi/{video_id}/2.jpg"
def parse_listens(json: dict) -> dict:
cover_replacing = {
"1e699948-c7c8-4bb2-9f8b-62e14b882a5d": "ca464c1d-5848-45bb-b92d-b1e4b00f9d65",
"0d516a93-061e-4a27-9cf7-f36e3a96f888": "5cc0c0c7-22f9-4a4b-a24c-f1a6732f813b",
"92ea5cc8-80e0-4da0-a10b-1bc2f8e8781e": "e8f3e14a-4794-4bab-b403-d562cdad4c2f",
}
new_json = {
"count": json["count"],
"listens": []
}
for track in json["listens"]:
listened_at = track.get("listened_at", 0)
track = track["track_metadata"]
new_track = {
"artist_name": track["artist_name"],
"track_name": track["track_name"],
"listened_at": listened_at
}
if mb := track.get("mbid_mapping"):
new_track["id"] = \
cover_replacing.get(mb["release_mbid"],
mb.get("caa_release_mbid",
mb["release_mbid"]
))
new_track["artist_name"] = mb["artists"][0]["artist_credit_name"]
new_track["track_name"] = mb["recording_name"]
elif info := track.get("additional_info"):
if info \
.get("music_service_name", "") \
.lower() in ("youtube", "youtube music"):
if cover := yt_cover(track["additional_info"]["origin_url"]):
new_track["cover_url"] = cover
if "cover_url" not in new_track.keys() and "id" in new_track.keys():
new_track["cover_url"] = "/asset/mb/" + new_track["id"]
new_json["listens"].append(new_track)
return new_json
def api_request(url: str, cache: Cache):
changed = False
prev_status = cache.status
try:
response = requests.get(url, timeout=10)
if response.status_code == 200:
json = parse_listens(response.json().get("payload"))
cache.status = 'success'
if cache.data != json:
cache.data = json
cache.last_updated = time()
data['last_updated'] = time()
data['etag'] = md5(''.join(
( dumps(data['caches'][x].data) for x in data['caches'] )
).encode()).hexdigest()
changed = True
else:
cache.status = f'error: {response.status_code}'
except Exception as e:
cache.status = f'error: {str(e)}'
if prev_status != cache.status:
changed = True
if changed:
_persist_cache()
scheduler = BackgroundScheduler()
if _IS_WRITER:
scheduler.add_job(
func=lambda: api_request("https://api.listenbrainz.org/1/user/risdeveau/listens?count=5", data['caches']['listens']),
trigger=IntervalTrigger(minutes=1),
id='risdeveau.listenbrainz.listens',
replace_existing=True
)
scheduler.add_job(
func=lambda: api_request("https://api.listenbrainz.org/1/user/risdeveau/playing-now", data['caches']['now']),
trigger=IntervalTrigger(seconds=15),
id='risdeveau.listenbrainz.playing-now',
replace_existing=True
)
_persist_cache()
scheduler.start()
atexit.register(lambda: scheduler.shutdown())
else:
refresh_cache()
@@ -0,0 +1,35 @@
from __future__ import annotations
import os
import fcntl
from typing import Optional
_LOCK_FD: Optional[int] = None # keep fd open for process lifetime
def should_start_scheduler() -> bool:
"""Return True in exactly one process (the scheduler owner).
Under gunicorn --workers N, code is imported in N processes. If APScheduler
starts at import time, you get N schedulers => N× API calls.
We prevent that by taking a non-blocking exclusive lock on a lockfile.
"""
global _LOCK_FD
if _LOCK_FD is not None:
return True
lock_path = os.environ.get("LAIR_SCHED_LOCK", "/tmp/lair-scheduler.lock")
fd = os.open(lock_path, os.O_CREAT | os.O_RDWR, 0o644)
try:
fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB)
_LOCK_FD = fd
return True
except BlockingIOError:
os.close(fd)
return False
except Exception:
try:
os.close(fd)
finally:
return False
@@ -0,0 +1,53 @@
from __future__ import annotations
import json
import os
import tempfile
from typing import Any, Optional, Tuple
def cache_dir() -> str:
"""Directory where the scheduler owner writes shared cache JSON files."""
d = os.environ.get("LAIR_CACHE_DIR", "/tmp/lair-cache")
os.makedirs(d, exist_ok=True)
return d
def cache_file(name: str) -> str:
return os.path.join(cache_dir(), f"{name}.json")
def atomic_write_json(path: str, payload: dict) -> None:
"""Write JSON atomically so readers never observe partial content."""
parent = os.path.dirname(path) or "."
fd, tmp = tempfile.mkstemp(prefix=".tmp-", dir=parent)
try:
with os.fdopen(fd, "w", encoding="utf-8") as f:
json.dump(payload, f, ensure_ascii=False, separators=(",", ":"))
os.replace(tmp, path)
finally:
try:
if os.path.exists(tmp):
os.unlink(tmp)
except Exception:
pass
def load_json_if_newer(path: str, last_mtime: float) -> Tuple[Optional[dict], float]:
"""Load JSON only if the file exists and has mtime newer than last_mtime."""
try:
st = os.stat(path)
except FileNotFoundError:
return None, last_mtime
except Exception:
return None, last_mtime
mtime = float(st.st_mtime)
if mtime <= float(last_mtime):
return None, last_mtime
try:
with open(path, "r", encoding="utf-8") as f:
return json.load(f), mtime
except Exception:
return None, last_mtime
+173
View File
@@ -0,0 +1,173 @@
import atexit
import re
from dataclasses import dataclass
from hashlib import md5
from json import dumps
from os import environ
from time import time
from urllib.parse import parse_qs, urlparse
from apscheduler.schedulers.background import BackgroundScheduler
from apscheduler.triggers.interval import IntervalTrigger
import requests
from flask import Flask
from .scheduler_guard import should_start_scheduler
from .shared_cache import atomic_write_json, cache_file, load_json_if_newer
TOKEN = environ.get("STEAM_TOKEN")
MY_ID = 76561198826355942
@dataclass
class Cache:
data = {}
last_updated = time()
status = None
data = {
"caches": {
"recent": Cache(),
"owned": Cache()
},
"last_updated": time(),
"etag": ""
}
_IS_WRITER = should_start_scheduler()
_CACHE_PATH = cache_file("steam")
_CACHE_MTIME = 0.0
def refresh_cache() -> None:
"""Refresh in-memory cache from the shared JSON file (for non-writer workers)."""
global _CACHE_MTIME
if _IS_WRITER:
return
payload, mtime = load_json_if_newer(_CACHE_PATH, _CACHE_MTIME)
if payload is None:
return
try:
data["etag"] = payload.get("etag", data.get("etag", ""))
data["last_updated"] = payload.get("last_updated", data.get("last_updated", time()))
caches = payload.get("caches", {})
for key, cache in data.get("caches", {}).items():
if isinstance(caches, dict) and key in caches and isinstance(caches[key], dict):
c = caches[key]
cache.data = c.get("data", cache.data)
cache.last_updated = c.get("last_updated", cache.last_updated)
cache.status = c.get("status", cache.status)
finally:
_CACHE_MTIME = mtime
def _persist_cache() -> None:
"""Persist current cache state to the shared JSON file (writer worker only)."""
if not _IS_WRITER:
return
payload = {
"etag": data.get("etag", ""),
"last_updated": data.get("last_updated", time()),
"caches": {
k: {
"data": v.data,
"last_updated": v.last_updated,
"status": v.status,
}
for k, v in data.get("caches", {}).items()
},
}
atomic_write_json(_CACHE_PATH, payload)
def modify_game_list(json: dict) -> dict:
if 'games' in json.keys():
apps = (3301060, 404790, 1281930, 1920960, 1325960, 431960)
new_games = []
for i, g in enumerate(json['games']):
if g['appid'] not in apps:
json['games'][i]['h_cover'] = f"https://shared.fastly.steamstatic.com/store_item_assets//steam/apps/{g['appid']}/header.jpg"
json['games'][i]['v_cover'] = f"https://shared.fastly.steamstatic.com/store_item_assets//steam/apps/{g['appid']}/library_600x900.jpg"
new_games.append(json['games'][i])
json['games'] = new_games
return json
def steam_request(interface: str, method: str, v: int = 1, **kwargs) -> requests.Response:
return requests.get(
f"https://api.steampowered.com/{interface}/{method}/v{v:04}/",
params=dict({"key": TOKEN}, **kwargs),
timeout=10
)
def api_request(cache, *args, **kwargs):
changed = False
prev_status = cache.status
try:
response = steam_request(*args, **kwargs)
if response.status_code == 200:
json = modify_game_list(response.json().get("response"))
cache.status = 'success'
if cache.data != json:
cache.data = json
cache.last_updated = time()
data['last_updated'] = time()
data['etag'] = md5(''.join(
( dumps(data['caches'][x].data) for x in data['caches'] )
).encode()).hexdigest()
changed = True
else:
cache.status = f'error: {response.status_code}'
except Exception as e:
cache.status = f'error: {str(e)}'
if prev_status != cache.status:
changed = True
if changed:
_persist_cache()
if TOKEN:
scheduler = BackgroundScheduler()
if _IS_WRITER:
scheduler.add_job(
func=lambda: api_request(data['caches']['recent'], "IPlayerService", "GetRecentlyPlayedGames", steamid=MY_ID),
trigger=IntervalTrigger(minutes=15),
id='risdeveau.steam.recent',
replace_existing=True
)
scheduler.add_job(
func=lambda: api_request(data['caches']['owned'], "IPlayerService", "GetOwnedGames", steamid=MY_ID, include_appinfo=1, include_played_free_games=1),
trigger=IntervalTrigger(minutes=60),
id='risdeveau.steam.owned',
replace_existing=True
)
_persist_cache()
scheduler.start()
api_request(data['caches']['recent'], "IPlayerService", "GetRecentlyPlayedGames", steamid=MY_ID)
api_request(data['caches']['owned'], "IPlayerService", "GetOwnedGames", steamid=MY_ID, include_appinfo=1, include_played_free_games=1)
atexit.register(lambda: scheduler.shutdown())
else:
refresh_cache()
else:
msg = "STEAM_TOKEN is not defined"
print(msg)
for c in data["caches"].values():
c.status = msg
if _IS_WRITER:
_persist_cache()
else:
refresh_cache()
+9
View File
@@ -0,0 +1,9 @@
from os import system as console
from os.path import join
def compile_styles():
dir = "blueprints/risdeveau/static/style"
files = ("risdeveau",)
for file in files:
console(f"sass {join(dir, file+'.scss')} {join(dir, file+'.css')}")
Binary file not shown.

After

Width:  |  Height:  |  Size: 594 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 876 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 271 KiB

Before

Width:  |  Height:  |  Size: 3.3 KiB

After

Width:  |  Height:  |  Size: 3.3 KiB

Before

Width:  |  Height:  |  Size: 3.3 KiB

After

Width:  |  Height:  |  Size: 3.3 KiB

Before

Width:  |  Height:  |  Size: 4.0 KiB

After

Width:  |  Height:  |  Size: 4.0 KiB

@@ -0,0 +1,68 @@
document.addEventListener('alpine:init', () => {
Alpine.data('rtime', (unixTimestamp) => ({
targetDate: new Date(unixTimestamp * 1000),
timeString: '',
timer: null,
interval: 1000,
colorClasses: {
green: 't-green',
yellow: 't-yellow',
orange: 't-orange',
red: 't-red'
},
currentColor: 'green',
get textColorClass() {
return this.colorClasses[this.currentColor];
},
init() {
this.updateTime();
this.timer = setInterval(() => this.updateTime(), this.interval);
this.$el.addEventListener('alpine:removing', () => {
if (this.interval) clearInterval(this.interval);
});
},
updateTime() {
const now = new Date();
const diffInSeconds = Math.floor((now - this.targetDate) / 1000);
const diffInMinutes = Math.floor(diffInSeconds / 60);
const diffInHours = Math.floor(diffInSeconds / 3600);
const diffInDays = Math.floor(diffInSeconds / 86400);
let newInterval = this.interval;
if (diffInSeconds < 60) {
this.timeString = 'a moment ago';
} else if (diffInMinutes < 60) {
this.timeString = `${diffInMinutes} m ago`;
newInterval = 10000;
} else if (diffInHours < 24) {
this.timeString = `${diffInHours} h ago`;
newInterval = 60000;
} else {
this.timeString = `${diffInDays} d ago`;
clearInterval(this.timer);
}
if (diffInMinutes <= 15) {
this.currentColor = 'green';
} else if (diffInHours < 1) {
this.currentColor = 'yellow';
} else if (diffInDays < 1) {
this.currentColor = 'orange';
} else {
this.currentColor = 'red';
}
if (this.interval != newInterval) {
clearInterval(this.timer)
this.timer = setInterval(() => this.updateTime(), newInterval);
}
},
}));
});
@@ -0,0 +1,100 @@
@use "sass:color";
@use "../../../root/static/style/catppuccin" as theme;
h3 {
margin-block-end: 0;
}
.qr {
img { width: 100% }
p { text-align: center; }
&.blocks {
flex-wrap: nowrap;
overflow-x: auto;
width: calc(100vw - 1rem);
max-width: 45rem;
scroll: {
behavior: smooth;
snap-type: x mandatory;
}
&::-webkit-scrollbar { display: none; }
&:hover .block.qr:not(:hover) {
filter: blur(5px);
transition: all 0.3s ease;
}
}
&.block {
flex: 0 0 calc(100vw - 2rem);
scroll-snap-align: start;
max-width: 13.666rem;
}
}
.track {
display: flex;
&.active {
box-shadow: theme.$green 0 0 5px 0;
}
img {
width: 5rem;
height: 5rem;
object-fit: cover;
border-radius: .5rem;
}
}
.steam {
.block {
display: flex;
img {
height: 7rem;
margin-right: .5rem;
}
p {
margin: .5rem 0;
}
}
}
table, tbody {
vertical-align: baseline;
border-collapse: collapse;
tr {
border-radius: 10px;
&:hover {
background-color: color.change(theme.$surface1, $alpha:75%);
}
th {
text-align: unset;
}
th + td, td + td {
padding-left: 2rem;
}
}
}
#pie:hover {
animation: rotateY-animation 2s linear infinite;
transform-style: preserve-3d;
@keyframes rotateY-animation {
to {
transform: rotateY(0deg);
}
from {
transform: rotateY(360deg);
}
}
}
+16
View File
@@ -0,0 +1,16 @@
div
.88-31
a.disabled(href="https://chest.lair.moe")
img(src="/static/img/88x31/gf.png")
a(href="//lair.moe")
img(src="/static/img/88x31/lair.gif")
a#pie(href="https://preview.about.akarpov.ru")
img(src="/static/img/88x31/withpie.gif")
.88-31
a(href="https://g.lair.moe/Sweetbread/nixos-config")
img(src="/static/img/88x31/nixos.webp")
img(src="/static/img/88x31/teto.webp")
@@ -0,0 +1,40 @@
.block
h3 Development
.blocks.badges
a.block(href="//g.lair.moe/Sweetbread")
img.img(src="/static/img/service/gitea.webp")
| Gitea
a.block(href="https://github.com/VerySweetBread")
img.img(src="https://github.githubassets.com/assets/GitHub-Mark-ea2971cee799.png")
| GitHub
a.block(href="https://git.kolibrios.org/Sweetbread")
img.img(src="https://git.kolibrios.org/assets/img/logo.svg")
| KolibriOS Git
h3 Contacts
.blocks.badges
a.block(href="https://matrix.to/#/@risdeveau:lair.moe")
img.img(src="https://matrix.org/assets/favimg.ico")
| Matrix
a.block(href="//b.lair.moe/@risdeveau")
img.img(src="/static/img/service/sharkey.webp")
| Fediverse
a.block(href="https://discord.com/users/459823895256498186")
img.img(src="https://cdn.prod.website-files.com/6257adef93867e50d84d30e2/66e3d80db9971f10a9757c99_Symbol.svg")
| Discord
a.block(href="mailto:risdeveau@lair.moe") Mail
h3 Game accounts
.blocks.badges
a.block(href="https://steamcommunity.com/id/risdeveau")
img.img(src="https://store.steampowered.com/favimg.ico")
| Steam
a.block(href="https://gamebanana.com/members/3899828")
img.img(src="https://images.gamebanana.com/static/img/favimg/favimg.ico")
| GameBanana
+14
View File
@@ -0,0 +1,14 @@
div
h3 Wallets
.blocks.qr
.block.qr
p POL, BNB
img(src="/static/img/wallets/evm.webp")
.block.qr
p TON
img(src="/static/img/wallets/ton.webp")
.block.qr
p XMR
img(src="/static/img/wallets/xmr.webp")
+50
View File
@@ -0,0 +1,50 @@
doctype html
html(lang="en")
head
title Sweet Bread
each f in ('tw', 'main', 'risdeveau')
link(rel="stylesheet", href="/static/style/#{f}.css")
link(rel="img" type="image/webp" href="/static/img/us/risdeveau.webp")
script(src="/static/script/rtime.js")
script(
src="https://track.lair.moe/api/script.js"
data-site-id="1"
defer
)
script(
src="https://cdn.jsdelivr.net/npm/htmx.org@2.0.8/dist/htmx.min.js"
integrity="sha384-/TgkGk7p307TH7EXJDuUlgG3Ce1UVolAOFopFekQkkXihi5u/6OCvVKyz1W+idaz"
crossorigin="anonymous"
)
script(defer src="https://cdn.jsdelivr.net/npm/alpinejs@3.x.x/dist/cdn.min.js")
meta(name="viewport" content="width=device-width, initial-scale=1.0")
meta(
name="htmx-config"
content='{
"responseHandling":[
{"code":"204", "swap": false},
{"code":"304", "swap": false},
{"code":"[23]..", "swap": true},
{"code":"422", "swap": true},
{"code":"[45]..", "swap": false, "error":true},
{"code":"...", "swap": true}
]
}'
)
body
header
a(href=url_for('root.index')) Lair
main
include risdeveau/templates/info.pug
include risdeveau/templates/contacts.pug
include risdeveau/templates/listenbrainz.pug
include risdeveau/templates/steam.pug
include risdeveau/templates/donate.pug
include risdeveau/templates/88x31.pug
+35
View File
@@ -0,0 +1,35 @@
.block
table
tr
th DoB
td 2005-01-13
tr
th Languages
td
table
tr
td Russian
td Native
tr
td English
td B2
tr
td French
td A1?
tr
td German
td A2?
tr
td Japanese
td Beginner
tr
th Student
td
table
tr
td Programmer
td 2.5/4yr.
tr
td Translator
td 2.5/3yr.
@@ -0,0 +1,32 @@
mixin track_block(track, is_active=false)
- set act_cls = "active" if is_active else ""
.block.track(class=act_cls)
if track.cover_url
img(src=track.cover_url)
div
p
b= track.artist_name
p= track.track_name
if not is_active
p(
x-data="rtime(#{track.listened_at})",
x-text="`Listened ${timeString}`",
:class="textColorClass"
)
.block(
hx-get="/m/listenbrainz",
hx-trigger="every 15s",
hx-swap="outerHTML",
hx-headers='{"If-None-Match": "#{lb.etag}"}'
)
h2
a(href="https://listenbrainz.org/user/risdeveau/") Listenbrainz
if lb.caches.now.data and lb.caches.now.data.listens[0]
+track_block(lb.caches.now.data.listens[0], true)
if lb.caches.listens.data and lb.caches.listens.data.listens
each track in lb.caches.listens.data.listens
+track_block(track)
+52
View File
@@ -0,0 +1,52 @@
mixin game_block(g, show_last_played=false)
a.block(href="https://store.steampowered.com/app/#{g.appid}", target="_blank")
picture
source(media="(max-width: 45rem)", srcset=g.v_cover)
img(src=g.h_cover)
div
strong= g.name
if g.playtime_2weeks
p Played last 2 weeks: #{tmsmp(g.playtime_2weeks * 60)}
p
- var lin = g.playtime_linux_forever > 0;
- var win = g.playtime_windows_forever > 0;
| Total played:
if lin
= tmsmp(g.playtime_linux_forever * 60)
| (
abbr(title="On Linux") L
| )
if lin and win
= " + "
if win
= tmsmp(g.playtime_windows_forever * 60)
| (
abbr(title="On Windows") W
| )
if lin and win
= " = "
= tmsmp(g.playtime_forever * 60)
if show_last_played and g.rtime_last_played != 0
p Last played: #{utmsmp(g.rtime_last_played)}
.block.steam(
hx-get="/m/steam",
hx-trigger="every 1m",
hx-swap="outerHTML",
hx-headers='{ "If-None-Match": "#{steam.etag}" }'
)
h2
a(href="https://steamcommunity.com/id/risdeveau") Steam
if steam.caches.recent.data.games
h3 Recently played:
each g in steam.caches.recent.data.games
+game_block(g)
p(x-data="rtime(#{steam.caches.recent.last_updated})", x-text="`Last updated: ${timeString}`")
if steam.caches.owned.data.games
h3 Top played games:
- var owned_games = steam.caches.owned.data.games | sort(attribute="playtime_forever", reverse=true)
each g in owned_games[:5]
+game_block(g, true)
p(x-data="rtime(#{steam.caches.owned.last_updated})", x-text="`Last updated: ${timeString}`")
+28
View File
@@ -0,0 +1,28 @@
from htmlmin import minify
from flask import Blueprint, render_template, request, jsonify
bp = Blueprint(
"root",
__name__,
template_folder="templates",
static_folder="static"
)
def render_tmpl(filename: str) -> str:
return minify(
render_template(filename),
remove_empty_space=True
)
@bp.route("/")
def index():
return render_tmpl('index.pug')
@bp.route("/host")
def host():
return render_tmpl('host.pug')
@bp.route("/us")
def us():
return render_tmpl('us.pug')
+9
View File
@@ -0,0 +1,9 @@
from os import system as console
from os.path import join
def compile_styles():
dir = "blueprints/root/static/style"
files = ("main", "tw")
for file in files:
console(f"sass {join(dir, file+'.scss')} {join(dir, file+'.css')}")
Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.5 KiB

Before

Width:  |  Height:  |  Size: 1.9 KiB

After

Width:  |  Height:  |  Size: 1.9 KiB

Before

Width:  |  Height:  |  Size: 9.2 KiB

After

Width:  |  Height:  |  Size: 9.2 KiB

Before

Width:  |  Height:  |  Size: 5.9 KiB

After

Width:  |  Height:  |  Size: 5.9 KiB

Before

Width:  |  Height:  |  Size: 76 KiB

After

Width:  |  Height:  |  Size: 76 KiB

@@ -0,0 +1,25 @@
// Palette: Catppuccin Mocha
// https://catppuccin.com/palette/
$base: #1e1e2e;
$text: #cdd6f4;
$mantle: #181825;
$crust: #11111b;
$overlay0: #6c7086;
$overlay1: #7f849c;
$overlay2: #9399b2;
$surface0: #313244;
$surface1: #45475a;
$surface2: #585b70;
$subtext0: #a6adc8;
$subtext1: #bac2de;
$red: #f38ba8;
$yellow: #f9e2af;
$green: #a6e3a1;
$peach: #fab387;
$blue: #89b4fa;
$mauve: #8839ef;
@@ -1,29 +1,5 @@
@use "sass:color"; @use "sass:color";
@use "catppuccin" as theme;
// Palette: Catppuccin Mocha
// https://catppuccin.com/palette/
$base: #1e1e2e;
$text: #cdd6f4;
$mantle: #181825;
$crust: #11111b;
$overlay0: #6c7086;
$overlay1: #7f849c;
$overlay2: #9399b2;
$surface0: #313244;
$surface1: #45475a;
$surface2: #585b70;
$subtext0: #a6adc8;
$subtext1: #bac2de;
$red: #f38ba8;
$green: #a6e3a1;
$peach: #fab387;
$blue: #89b4fa;
$mauve: #8839ef;
html { html {
@@ -33,9 +9,9 @@ html {
body { body {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
background-color: $base; background-color: theme.$base;
font-family: Pixeloid, PixelMPlus; font-family: Pixeloid, PixelMPlus;
color: $text; color: theme.$text;
width: 100%; width: 100%;
height: 100%; height: 100%;
margin: 0; margin: 0;
@@ -55,7 +31,7 @@ h1 {
a { a {
color: unset; color: unset;
text: { text: {
decoration: underline {color: $blue}; decoration: underline {color: theme.$blue};
underline-offset: 1px; underline-offset: 1px;
} }
transition: 0.3s ease; transition: 0.3s ease;
@@ -68,7 +44,7 @@ a {
transition: none !important; transition: none !important;
display: inline-block; display: inline-block;
transform: scale(.98) !important; transform: scale(.98) !important;
background-color: $mantle !important; background-color: theme.$mantle !important;
} }
&.block { &.block {
@@ -76,7 +52,7 @@ a {
&:hover { &:hover {
transform: scale(1.02) translateY(-.25rem); transform: scale(1.02) translateY(-.25rem);
background-color: $surface1; background-color: theme.$surface1;
} }
} }
} }
@@ -92,16 +68,20 @@ ul {
header { header {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
background-color: $mantle; background-color: theme.$mantle;
padding: .5rem; padding: .5rem;
font-size: larger; font-size: larger;
.header-links * + * {
padding-left: 1ch;
}
} }
footer { footer {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
flex-wrap: wrap; flex-wrap: wrap;
background-color: $mantle; background-color: theme.$mantle;
margin-top: 2rem; margin-top: 2rem;
padding: 1rem; padding: 1rem;
column-gap: 4ch; column-gap: 4ch;
@@ -109,38 +89,48 @@ footer {
.mono { .mono {
font-family: Monocraft, monospace; font-family: Monocraft, monospace;
background-color: $mantle; background-color: theme.$mantle;
border-radius: 2px; border-radius: 2px;
padding: 0 .25rem; padding: 0 .25rem;
color: $subtext0; color: theme.$subtext0;
overflow-wrap: anywhere; overflow-wrap: anywhere;
&:hover { &:hover {
transition: .3s ease; transition: .3s ease;
background-color: $crust; background-color: theme.$crust;
} }
} }
.block { .block {
display: block; display: block;
background-color: $surface0; background-color: theme.$surface0;
border-radius: .5rem; border-radius: .5rem;
padding: .5rem; padding: .5rem;
h2 {
margin: -.5rem -.5rem 1rem;
padding: .5rem;
text-align: center;
}
.block {
background-color: theme.$surface1;
}
& + & { & + & {
margin-top: .5rem; margin-top: .5rem;
} }
&.red { &.red {
background-color: color.mix($surface0, $red, 60%); background-color: color.mix(theme.$surface0, theme.$red, 60%);
} }
&.orange { &.orange {
background-color: color.mix($surface0, $peach, 60%); background-color: color.mix(theme.$surface0, theme.$peach, 60%);
} }
&.green { &.green {
background-color: color.mix($surface0, $green, 60%); background-color: color.mix(theme.$surface0, theme.$green, 60%);
&:hover { background-color: color.mix($surface1, $green, 60%); } &:hover { background-color: color.mix(theme.$surface1, theme.$green, 60%); }
&:active { background-color: color.mix($mantle, $green, 60%) !important; } &:active { background-color: color.mix(theme.$mantle, theme.$green, 60%) !important; }
} }
& .header { & .header {
@@ -148,7 +138,7 @@ footer {
align-items: center; align-items: center;
font-size: x-large; font-size: x-large;
.icon { .img {
margin-right: .5ch; margin-right: .5ch;
} }
} }
@@ -184,12 +174,56 @@ footer {
} }
} }
.icon { .img {
width: 1.5em; width: 1.5em;
vertical-align: middle; vertical-align: middle;
border-radius: .2em; border-radius: .2em;
} }
.webring {
margin-top: 1rem;
display: flex;
justify-content: center;
a {
padding: .5rem 1rem;
margin: .1rem !important;
border-radius: 0;
}
}
.\38 8-31 {
margin-top: 2rem;
display: flex;
justify-content: center;
& + & {
margin-top: 0;
}
a, img {
width: 88px;
height: 31px;
}
img {
transition-timing-function: ease-out;
transition-duration: .2s;
&:hover {
transform: scale(1.5);
}
}
}
.disabled {
opacity: 50%;
cursor: not-allowed;
a {
pointer-events: none;
}
}
::-webkit-scrollbar { ::-webkit-scrollbar {
width: .5rem; width: .5rem;
@@ -199,15 +233,15 @@ footer {
} }
&-track { &-track {
background-color: $base; background-color: theme.$base;
} }
&-thumb { &-thumb {
background-color: $overlay0; background-color: theme.$overlay0;
border-radius: .25rem; border-radius: .25rem;
&:hover { &:hover {
background-color: $overlay1; background-color: theme.$overlay1;
} }
} }
} }
+8
View File
@@ -0,0 +1,8 @@
@use "catppuccin" as theme;
.t {
&-red { color: theme.$red; }
&-orange { color: theme.$peach; }
&-yellow { color: theme.$yellow; }
&-green { color: theme.$green; }
}
+33
View File
@@ -0,0 +1,33 @@
doctype html
html(lang=g.locale)
head
title Lair
link(rel="stylesheet" href="/static/style/main.css")
link(rel="img" type="image/webp" href="/static/img/lair.webp")
script(src="/static/script/copy-mono.js")
script(
src="https://track.lair.moe/api/script.js"
data-site-id="1"
defer
)
meta(name="viewport" content="width=device-width, initial-scale=1.0")
meta(name="mock-email" content="admin@example.com")
meta(property="og:type" value="website")
meta(property="og:url" value="https://lair.moe")
meta(property="og:title" value="Lair.moe")
meta(property="og:image" value="https://lair.moe/static/img/lair.webp")
meta(property="og:description" value=_("description"))
body
include root/templates/header.pug
h1
block title
main
block content
include root/templates/footer.pug
+9
View File
@@ -0,0 +1,9 @@
footer
div
| lair.moe &#127279; 2025 - 2026</div>
div
a(href="https://g.lair.moe/Sweetbread/lair.moe")= _('site source')
div
= _('contact us')
= ": "
a(href="mailto:admin@lair.moe") admin@lair.moe
+17
View File
@@ -0,0 +1,17 @@
header
if request.path != url_for('.index')
a(href=url_for('.index')) Lair
else
div
.header-links
-
var links = (
('.us', _('about us')),
('.host', _('about host')),
)
each l, t in links
if url_for(l) == request.path
strong= t
else
a(href=url_for(l))= t
+20
View File
@@ -0,0 +1,20 @@
extends root/templates/base.pug
block title
= _('about host')
block content
a.block(href="https://play2go.cloud/?ref_id=4baFoOIp5QE" target="_blank")
strong= _("host:hoster")
| : play2go
p= _('host:hoster_descr')
.block
strong= _("host:specifications")
| :
ul
li CPU: Ryzen 9@3.4GHz (4 cores)
li RAM: 8 GB
li SSD: 150 GB
li ETH: 500Mb/s
li Loc: Deutschland, Frankfurt am Main
+70
View File
@@ -0,0 +1,70 @@
extends root/templates/base.pug
block title
img.img(src="/static/img/lair.webp")
| Lair
block content
a.block(href="https://b.lair.moe" target="_blank")
.header
img.img(src="/static/img/service/sharkey.webp")
strong Sharkey
p= _('index.descr:sharkey')
a.block(href="https://g.lair.moe" target="_blank")
.header
img.img(src="/static/img/service/gitea.webp")
strong Gitea
p= _('index.descr:gitea')
.block
p
a(href="https://m.lair.moe" target="_blank")
strong Matrix
| &mdash; {{ _('index.descr:matrix') }}
p
a(href="//c.lair.moe" target="_blank")
strong Copyparty
| &mdash; {{ _('index.descr:copyparty') }}
p
a(href="https://tools.lair.moe" target="_blank")
strong IT-tools
| &mdash; {{ _('index.descr:tools') }}
p
a(href="https://vert.lair.moe" target="_blank")
strong Vert
| &mdash; {{ _('index.descr:vert') }}
.block
strong= _('index:altfronts')
p
a(href="https://s.lair.moe" target="_blank")
strong 4get
| &mdash; {{ _('index.descr:4get') }}
p
a(href="https://tl.lair.moe" target="_blank")
strong TransLite
| &mdash; {{ _('index.descr:tl') }}
p
a(href="https://lyr.lair.moe" target="_blank")
strong Intellectual
| &mdash; {{ _('index.descr:lyr') }}
.block
div
strong DNS
| :
ul
li
span.mono 64.188.64.176
li
| DoT:
span.mono lair.moe:853
li
| DoH:
span.mono dns.lair.moe
p
strong Yggdrasil
| :
span.mono 200:ee1:bad2:1732:4b91:c3e3:2f08:29b3
+17
View File
@@ -0,0 +1,17 @@
extends root/templates/base.pug
block title
| О нас
block content
a.block.green(href=url_for('risdeveau.index'))
.header
img.img(src="/static/img/us/risdeveau.webp")
| Sweetbread
| Главный админ, занимается почти всеми сервисами. Создал этот сайт
.block.orange.disabled
.header
img.img(src="/static/img/us/chest.webp")
| Chest
| Должна была помогать делать этот сайт
+3 -1
View File
@@ -7,7 +7,7 @@ about host = Über Server
[index] [index]
bottom_text = Außerdem bieten wir {glitchtip}, {baikal} und {freshrss} für Mitglieder unserer Gruppe an! altfronts = Altfronts
[index.descr] [index.descr]
sharkey = Föderierter Microblogging-Dienst auf Basis des ActivityPub-Protokolls sharkey = Föderierter Microblogging-Dienst auf Basis des ActivityPub-Protokolls
@@ -17,6 +17,8 @@ copyparty = Cloud-Dateispeicher
4get = Proxy-Suchmaschine 4get = Proxy-Suchmaschine
tools = Satz verschiedener Werkzeuge tools = Satz verschiedener Werkzeuge
vert = Dateiumwandler vert = Dateiumwandler
tl = Altfront für beliebte Suchmaschinen
lyr = Altfront für Genius
[host] [host]
+8 -1
View File
@@ -5,9 +5,14 @@ contact us = Contact us
about us = About us about us = About us
about host = About host about host = About host
contacts = Contacts
donate = Donate
description = Small personal site
[index] [index]
bottom_text = We also have {glitchtip}, {baikal} and {freshrss} for members of our squad! altfronts = Altfronts
[index.descr] [index.descr]
sharkey = ActivityPub-based federated microblogging service sharkey = ActivityPub-based federated microblogging service
@@ -17,6 +22,8 @@ copyparty = cloud file storage
4get = proxy search engine 4get = proxy search engine
tools = set of various tools tools = set of various tools
vert = file converter vert = file converter
tl = altfront for popular search engines
lyr = altfront for Genius
[host] [host]
+3 -1
View File
@@ -7,7 +7,7 @@ about host = À propos de serveur
[index] [index]
bottom_text = On a aussi {glitchtip}, {baikal} et {freshrss} pour les membres du groupe ! altfronts = Altfronts
[index.descr] [index.descr]
sharkey = Service de microblogging fédéré avec ActivityPub sharkey = Service de microblogging fédéré avec ActivityPub
@@ -17,6 +17,8 @@ copyparty = Stockage de fichiers en cloud
4get = Moteur de recherche proxy 4get = Moteur de recherche proxy
tools = ensemble d'outils variés tools = ensemble d'outils variés
vert = convertisseur de fichiers vert = convertisseur de fichiers
tl = altfront pour les moteurs de recherche populaires
lyr = altfront pour Genius
[host] [host]
+3 -1
View File
@@ -7,7 +7,7 @@ about host = サーバーについて
[index] [index]
bottom_text = メンバーには {glitchtip}、{baikal}、{freshrss} も使えるよ! altfronts = 代替フロントエンド
[index.descr] [index.descr]
sharkey = ActivityPubを使った連合型マイクロブログ sharkey = ActivityPubを使った連合型マイクロブログ
@@ -17,6 +17,8 @@ copyparty = クラウドファイルストレージ
4get = プロキシ検索エンジン 4get = プロキシ検索エンジン
tools = 様々なツールのセット tools = 様々なツールのセット
vert = ファイル変換ツール vert = ファイル変換ツール
tl = 人気検索エンジン向けの代替フロントエンド
lyr = Genius向けの代替フロントエンド
[host] [host]
+8 -1
View File
@@ -5,9 +5,14 @@ contact us = Для связи
about us = О нас about us = О нас
about host = О хосте about host = О хосте
contacts = Контакты
donate = Донат
description = Небольшой личный сайт
[index] [index]
bottom_text = Ещё у нас есть {glitchtip}, {baikal} и {freshrss} для участников нашей группы! altfronts = Альтфронты
[index.descr] [index.descr]
sharkey = Федеративная микроблогинговая система поверх протокола ActivityPub sharkey = Федеративная микроблогинговая система поверх протокола ActivityPub
@@ -17,6 +22,8 @@ copyparty = облачное хранилище файлов
4get = прокси-поисковик 4get = прокси-поисковик
tools = набор разнообразных утилит tools = набор разнообразных утилит
vert = конвертация файлов vert = конвертация файлов
tl = альтфронт для популярных поисковиков
lyr = альтфронт для Genius
[host] [host]
+47
View File
@@ -0,0 +1,47 @@
from configparser import ConfigParser
from flask import g, request
translations_cache = {}
def load_translations(lang):
if lang not in translations_cache:
translations_cache[lang] = {}
try:
config = ConfigParser()
config.read(f'locale/{lang}.ini')
for section in config.sections():
translations_cache[lang][section] = dict(config.items(section))
except:
pass
return translations_cache[lang]
def get_locale():
return request.accept_languages.best_match(
('en', 'ru', 'de', 'fr', 'ja'),
) or 'en'
def before_request():
g.locale = get_locale()
g.translations = load_translations(g.locale)
def inject_translations():
def translate(text, **kwargs):
if ":" in text:
section, key = text.split(":", 1)
else:
section, key = "common", text
template = g.translations \
.get(section, {}) \
.get(key, f"${section}: {key}$")
try:
return template.format(**kwargs)
except:
return template
return {'_': translate}
+6
View File
@@ -1,2 +1,8 @@
Flask==3.1.1 Flask==3.1.1
gunicorn gunicorn
htmlmin2
requests
APScheduler
musicbrainzngs
python-magic
git+https://github.com/VerySweetBread/pypugjs
+4 -4
View File
@@ -3,17 +3,17 @@ let
pypkgs = pkgs.python3Packages; pypkgs = pkgs.python3Packages;
in in
pkgs.mkShell { pkgs.mkShell {
name = "codrs.ru"; name = "lair.moe";
buildInputs = with pypkgs; [ buildInputs = with pypkgs; [
python python
python-magic
virtualenv virtualenv
# pkgs.nodejs pkgs.dart-sass
pkgs.nodePackages.sass
]; ];
shellHook = '' shellHook = ''
if [ ! -d "venv" ]; then if [ ! -d ".venv" ]; then
python -m venv .venv python -m venv .venv
fi fi
Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.0 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 59 KiB

-33
View File
@@ -1,33 +0,0 @@
h3 {
margin-block-end: 0;
}
.qr {
img { width: 100% }
p { text-align: center; }
&.blocks {
flex-wrap: nowrap;
overflow-x: auto;
width: calc(100vw - 1rem);
max-width: 45rem;
scroll: {
behavior: smooth;
snap-type: x mandatory;
}
&::-webkit-scrollbar { display: none; }
&:hover .block.qr:not(:hover) {
filter: blur(5px);
transition: all 0.3s ease;
}
}
&.block {
flex: 0 0 calc(100vw - 2rem);
scroll-snap-align: start;
max-width: 13.666rem;
}
}
-27
View File
@@ -1,27 +0,0 @@
<!DOCTYPE html>
<html>
<head>
<title>Coders Squad</title>
<link rel="stylesheet" href="/static/style/main.css">
<link rel="icon" type="image/webp" href="/static/icon/codrs.webp" />
<script src="/static/script/copy-mono.js"> </script>
<script
src="https://track.codrs.ru/api/script.js"
data-site-id="1"
defer
></script>
<meta name="viewport" content="width=device-width, initial-scale=1.0">
</head>
<body>
{% include 'header.tmpl' %}
<h1>{% block title %}{% endblock %}</h1>
<main>
{% block content %}{% endblock %}
</main>
{% include 'footer.tmpl' %}
</body>
</html>
-5
View File
@@ -1,5 +0,0 @@
<footer>
<div>codrs.ru &#169; 2025</div>
<div><a href="https://g.codrs.ru/Sweetbread/codrs.ru">{{ _('site source') }}</a></div>
<div>{{ _('contact us') }}: <a href="mailto:admin@codrs.ru">admin@codrs.ru</a></div>
</footer>
-20
View File
@@ -1,20 +0,0 @@
<header>
{%- if request.path != url_for('index') %}
<a href="{{ url_for('index') }}">Coders Squad</a>
{%- else %}
<div></div>
{%- endif %}
<div class="header-links">
{%- for (l, t) in (
('us', _('about us')),
('host', _('about host'))
) %}
{%- if url_for(l) == request.path %}
<strong>{{ t }}</strong>
{%- else %}
<a href="{{ url_for(l) }}">{{ t }}</a>
{%- endif %}
{%- endfor %}
</div>
</header>
-21
View File
@@ -1,21 +0,0 @@
{% extends 'base.tmpl' %}
{% block title %}{{ _('about host') }}{% endblock %}
{% block content %}
<a href="https://play2go.cloud/?ref_id=4baFoOIp5QE" target="_blank" class="block">
<strong>{{ _("host:hoster") }}</strong>: play2go
<p>{{ _('host:hoster_descr') }}</p>
</a>
<div class="block">
<strong>{{ _("host:specifications") }}</strong>:
<ul>
<li>CPU: Ryzen i9@3.4GHz (4 cores)</li>
<li>RAM: 8 GiB</li>
<li>SSD: 150 GB</li>
<li>ETH: 500Mb/s</li>
<li>Loc: Deutchland, Frankfurt am Mein</li>
</ul>
</div>
{% endblock %}
-56
View File
@@ -1,56 +0,0 @@
{% extends 'base.tmpl' %}
{% block title %}
<img src="/static/icon/codrs.webp" class="icon" />
Coders Squad
{% endblock %}
{% block content %}
<a href="https://b.codrs.ru" target="_blank" class="block">
<div class="header">
<img src="/static/icon/service/sharkey.webp" class="icon"/>
<strong>Sharkey</strong>
</div>
<p>{{ _('index.descr:sharkey') }}</p>
</a>
<a href="https://g.codrs.ru" target="_blank" class="block">
<div class="header">
<img src="/static/icon/service/gitea.webp" class="icon"/>
<strong>Gitea</strong>
</div>
<p>{{ _('index.descr:gitea') }}</p>
</a>
<div class="block">
<p><a href="https://m.codrs.ru" target="_blank"><strong>Matrix</strong></a> &mdash; {{ _('index.descr:matrix') }}</p>
<p><a href="https://c.codrs.ru" target="_blank"><strong>Copyparty</strong></a> &mdash; {{ _('index.descr:copyparty') }}</p>
<p><a href="https://s.codrs.ru" target="_blank"><strong>4get</strong></a> &mdash; {{ _('index.descr:4get') }}</p>
<p><a href="https://tools.codrs.ru" target="_blank"><strong>IT-tools</strong></a> &mdash; {{ _('index.descr:tools') }}</p>
<p><a href="https://vert.codrs.ru" target="_blank"><strong>Vert</strong></a> &mdash; {{ _('index.descr:vert') }}</p>
</div>
<div class="block">
<div>
<strong>DNS</strong>:
<ul>
<li><span class="mono">64.188.64.176</span></li>
<li>DoT: <span class="mono">codrs.ru:853</span></li>
<li>DoH: <span class="mono">dns.codrs.ru</span></li>
</ul>
</div>
<p>
<strong>Yggdrasil</strong>:
<span class="mono">200:ee1:bad2:1732:4b91:c3e3:2f08:29b3</span>
</p>
</div>
<div class="block">
{{
_('index:bottom_text',
glitchtip='<a href="https://bug.codrs.ru" target="_blank"><strong>GlitchTip</strong></a>',
baikal='<a href="https://dav.codrs.ru" target="_blank"><strong>Baikal</strong></a>',
freshrss='<a href="https://rss.codrs.ru" target="_blank"><strong>FreshRSS</strong></a>',
) | safe
}}
</div>
{% endblock %}
-88
View File
@@ -1,88 +0,0 @@
<!DOCTYPE html>
<html>
<head>
<title>Sweet Bread</title>
<link rel="stylesheet" href="/static/style/main.css">
<link rel="stylesheet" href="/static/style/risdeveau.css">
<link rel="icon" type="image/webp" href="/static/icon/us/risdeveau.webp" />
<script
src="https://track.codrs.ru/api/script.js"
data-site-id="1"
defer
></script>
<meta name="viewport" content="width=device-width, initial-scale=1.0">
</head>
<body>
<header>
<a href="{{ url_for('index') }}">Coders Squad</a>
</header>
<main>
<h3>Development</h3>
<div class="blocks badges">
<a class="block" href="//g.codrs.ru/Sweetbread">
<img class="icon" src="/static/icon/service/gitea.webp" />
Gitea
</a>
<a class="block" href="https://github.com/VerySweetBread">
<img class="icon" src="https://github.githubassets.com/assets/GitHub-Mark-ea2971cee799.png" />
GitHub
</a>
<a class="block" href="https://git.kolibrios.org/Sweetbread">
<img class="icon" src="https://git.kolibrios.org/assets/img/logo.svg" />
KolibriOS Git
</a>
</div>
<h3>Contacts</h3>
<div class="blocks badges">
<a class="block" href="https://matrix.to/#/@risdeveau:codrs.ru">
<img class="icon" src="https://matrix.org/assets/favicon.ico" />
Matrix
</a>
<a class="block" href="//b.codrs.ru/@risdeveau">
<img class="icon" src="/static/icon/service/sharkey.webp" />
Fediverse
</a>
<a class="block" href="https://discord.com/users/459823895256498186">
<img class="icon" src="https://cdn.prod.website-files.com/6257adef93867e50d84d30e2/66e3d80db9971f10a9757c99_Symbol.svg" />
Discord
</a>
<a class="block" href="mailto:risdeveau@codrs.ru">
Mail
</a>
</div>
<h3>Game accounts</h3>
<div class="blocks badges">
<a class="block" href="https://steamcommunity.com/id/risdeveau">
<img class="icon" src="https://store.steampowered.com/favicon.ico" />
Steam
</a>
<a class="block" href="https://gamebanana.com/members/3899828">
<img class="icon" src="https://images.gamebanana.com/static/img/favicon/favicon.ico" />
GameBanana
</a>
</div>
<h3>Wallets</h3>
<div class="blocks qr">
<div class="block qr">
<p>POL, BNB</p>
<img src="/static/img/risdeveau/wallets/evm.webp">
</div>
<div class="block qr">
<p>TON</p>
<img src="/static/img/risdeveau/wallets/ton.webp">
</div>
<div class="block qr">
<p>XMR</p>
<img src="/static/img/risdeveau/wallets/xmr.webp">
</div>
</div>
</main>
</body>
</html>
-29
View File
@@ -1,29 +0,0 @@
{% extends 'base.tmpl' %}
{% block title %}О нас{% endblock %}
{% block content %}
<a href="{{ url_for('risdeveau') }}" class="block green">
<div class="header">
<img src="/static/icon/us/risdeveau.webp" class="icon"/>
Sweetbread
</div>
Главный админ, занимается почти всеми сервисами. Создал этот сайт
</a>
<div class="block red">
<div class="header">
<img src="/static/icon/us/zxcqirara.webp" class="icon"/>
zxcqirara
</div>
Второй админ, занимается майном и VPN
</div>
<div class="block orange">
<div class="header">
<img src="/static/icon/us/chest.webp" class="icon"/>
Chest
</div>
Должна была помогать делать этот сайт
</div>
{% endblock %}