Compare commits

..

No commits in common. "main" and "v1.0.0" have entirely different histories.
main ... v1.0.0

21 changed files with 54 additions and 793 deletions

View file

@ -1,6 +1,6 @@
.*
/.dockerignore
/.gitignore
/inventory.sqlite3
/README.md
*.pyc
Dockerfile
/contrib/

View file

@ -1,22 +0,0 @@
name: deployment-on-pull-request
on:
pull_request:
pull_request_target:
types:
- opened
- reopened
- synchronize
jobs:
deployment:
runs-on: docker
container:
image: node:bookworm
steps:
- uses: actions/checkout@v4
- name: Build inventory
uses: https://codeberg.org/umglurf/kaniko-action@main
with:
cache: false
push: 'false'
context: /workspace/snoopy/in-c4/

View file

@ -1,24 +0,0 @@
name: deployment
on:
push:
branches:
- 'main'
jobs:
deployment:
runs-on: docker
container:
image: node:bookworm
steps:
- uses: actions/checkout@v4
- name: Build inventory
uses: https://codeberg.org/umglurf/kaniko-action@main
with:
cache: true
cache_repo: git.koeln.ccc.de/${{ github.repository }}/cache
push: 'true'
context: /workspace/snoopy/in-c4/
credentials: |
git.koeln.ccc.de=snoopy:${{ secrets.REGISTRY_WRITE }}
destinations: |
git.koeln.ccc.de/${{ github.repository }}/in-c4:latest

2
.gitignore vendored
View file

@ -1,2 +1,2 @@
/data/*
/inventory.sqlite3
*.pyc

View file

@ -1,4 +1,4 @@
FROM python:3.12-alpine3.19
FROM python:3.10-alpine
WORKDIR /app

View file

@ -1,47 +0,0 @@
#!/usr/bin/env python3
import sys
import requests
BASE = "https://in.ccc.ac"
COUNT = 0
START_ID = 1
session = requests.Session()
# session.auth = (":)", ":)")
res = session.get(f"{BASE}/api/items")
res.raise_for_status()
ids = res.json().keys()
kiste_ids = {int(i.lstrip("K")) for i in ids if i.startswith("K") and i != "K"}
print("existing ids:", repr(kiste_ids), file=sys.stderr)
current_id = START_ID
new_ids = []
while len(new_ids) < COUNT:
if current_id not in kiste_ids:
new_ids.append(current_id)
current_id += 1
print("created:", file=sys.stderr)
if len(new_ids) == 0:
print("(none)", file=sys.stderr)
for i in new_ids:
j = {
"id": f"K{i}",
"is_in": None,
"coords_bl": None,
"coords_tr": None,
"type": "Kiste",
"name": None,
"content": None,
"note": "Bitte Name, Ort, Inhalt eintragen",
"hidden": False,
}
res = session.put(f"{BASE}/api/items/{j['id']}", json=j)
res.raise_for_status()
print(f"{j['id']},{BASE}/form.html?id={j['id']}")

22
crud.py
View file

@ -1,32 +1,18 @@
from enum import Enum
from sqlalchemy import DECIMAL, cast
from sqlalchemy.orm import Session
from sqlalchemy import DECIMAL, cast
import models
import schemas
import models, schemas
def get_items(db: Session) -> list[models.Item]:
return db.query(models.Item).all()
def put_item(db: Session, id: str, item: schemas.Item):
updated = bool(
db.query(models.Item).filter(models.Item.id == id).update(item.dict())
)
updated = bool(db.query(models.Item).filter(models.Item.id == id).update(item.dict()))
if not updated:
db.add(models.Item(**item.dict()))
db.commit()
if updated:
return PutItemResult.UPDATED
else:
return PutItemResult.ADDED
class PutItemResult(Enum):
ADDED = 1
UPDATED = 2
return updated
def delete_item(db: Session, id: str):

View file

@ -2,10 +2,12 @@ from sqlalchemy import create_engine
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import sessionmaker
SQLALCHEMY_DATABASE_URL = "sqlite:///./data/inventory.sqlite3"
SQLALCHEMY_DATABASE_URL = "sqlite:///./inventory.sqlite3"
engine = create_engine(
SQLALCHEMY_DATABASE_URL, connect_args={"check_same_thread": False}, echo=True
SQLALCHEMY_DATABASE_URL,
connect_args = {"check_same_thread": False},
echo=True
)
SessionLocal = sessionmaker(autocommit=False, autoflush=False, bind=engine)

35
main.py
View file

@ -1,12 +1,9 @@
import re
from calendar import timegm
from datetime import date
from fastapi import Depends, FastAPI, Response
from fastapi.staticfiles import StaticFiles
from sqlalchemy import event
from sqlalchemy.engine import Engine
from sqlalchemy.orm import Session
from sqlalchemy.engine import Engine
import re
import crud
from database import SessionLocal, engine
@ -17,7 +14,6 @@ Base.metadata.create_all(bind=engine)
app = FastAPI()
def get_db():
db = SessionLocal()
try:
@ -25,7 +21,6 @@ def get_db():
finally:
db.close()
@event.listens_for(Engine, "connect")
def _set_sqlite_pragma(conn, _):
cursor = conn.cursor()
@ -36,35 +31,21 @@ def _set_sqlite_pragma(conn, _):
@app.get("/api/items", response_model=dict[str, Item])
async def list_items(db: Session = Depends(get_db)):
# natural sort by id
natsort = lambda item: [
int(t) if t.isdigit() else t.lower() for t in re.split(r"(\d+)", item.id)
]
natsort = lambda item: [int(t) if t.isdigit() else t.lower() for t in re.split('(\d+)', item.id)]
items = crud.get_items(db)
items = sorted(items, key=natsort)
return {i.id: i for i in items}
return {i.id:i for i in items}
@app.put(
"/api/items/{id}",
status_code=201,
responses={201: {"description": "created"}, 204: {"description": "updated"}},
)
@app.put("/api/items/{id}")
async def put_item(id: str, item: Item, db: Session = Depends(get_db)):
if item.last_updated is None:
item.last_updated = month_timestamp()
if crud.put_item(db, id, item) == crud.PutItemResult.UPDATED:
return Response(b"", status_code=204)
return Response(b"", status_code=201)
if crud.put_item(db, id, item):
return Response(b'', status_code=204)
return Response(b'', status_code=201)
@app.delete("/api/items/{id}", status_code=204)
async def delete_item(id: str, db: Session = Depends(get_db)):
crud.delete_item(db, id)
def month_timestamp() -> int:
"""Provides the timestamp of the current month's beginning (for improved privacy)"""
return timegm(date.today().replace(day=1).timetuple())
app.mount("/", StaticFiles(directory="static", html=True), name="static")

View file

@ -1,8 +1,7 @@
from sqlalchemy import Boolean, Column, ForeignKey, Integer, String, Text
from sqlalchemy import Column, ForeignKey, String, Text, Boolean
from database import Base
class Item(Base):
__tablename__ = "items"
@ -15,4 +14,3 @@ class Item(Base):
content = Column(Text)
note = Column(Text)
hidden = Column(Boolean, nullable=False)
last_updated = Column(Integer)

View file

@ -1,3 +1,3 @@
fastapi>=0.109.0
sqlalchemy>=2.0.25
uvicorn>=0.27.0.post1
fastapi>=0.79.0
sqlalchemy>=1.4.39
uvicorn>=0.17.6

View file

@ -1,4 +1,4 @@
from pydantic import BaseModel, Field
from pydantic import BaseModel
class Item(BaseModel):
@ -11,7 +11,6 @@ class Item(BaseModel):
content: str | None
note: str | None
hidden: bool
last_updated: int | None = Field(None)
class Config:
from_attributes = True
orm_mode = True

1
static/80x80.svg Normal file

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 60 KiB

View file

@ -1,122 +0,0 @@
body {
color: #f0f0f0;
background-color: #333;
}
h1, h1 a {
color: #dfaa37; /* C4 Yellow */
}
#search.failed {
color: #200;
background-color: #ec6d5f;
}
#map {
border: 1px solid #202020;
background-color: #4d4d4d;
max-width: 1440px;
max-height: 50vh;
aspect-ratio: 2 / 1;
box-shadow: 0px 6px 12px #000c;
}
#mapnote {
background-color: #ccc;
color: #333;
border-color: #202020;
padding: 0.25em 0.5em;
position: relative;
top: -1px;
box-shadow: 0px 6px 12px #000c;
}
#mapnote:empty {
display: none;
}
#itemcount {
display: inline-block;
margin-right: 1ch;
}
.result {
background-color: #202020;
border-color: #4d4d4d;
}
.result:focus {
background-color: #202020;
border-color: #dfaa37;
outline-color: #dfaa37;
}
.result a {
color: #dfaa37;
}
.note {
background-color: #333;
font-size: smaller;
}
#knowntypes button {
color: #220;
background-color: #dfaa37;
border: 1px solid #220;
border-radius: 4px;
padding: 0.2em 0.4em;
margin: 0.2em;
font-weight: bolder;
cursor: pointer;
}
#knowntypes button:hover, #knowntypes button:focus {
background-color: #f5c251;
}
.btn {
box-shadow: 0 3px 6px -1px rgba(0,0,0,0.3);
}
.btn.primary {
color: #220;
background-color: #dfaa37;
}
.btn.primary:hover,.btn.primary:focus {
background-color: #f5c251;
}
.btn.grn {
color: #020;
background-color: #9f6;
}
.btn.grn:hover ,.btn.grn:focus {
background-color: #3f0;
}
.btn.red {
color: #200;
background-color: #ec6d5f;
}
.btn.red:hover, .btn.red:focus {
background-color: #ff7667;
}
@keyframes blinker {
0% { opacity: 1; }
40% { opacity: 1; }
90% { opacity: 0; }
100% { opacity: 1; }
}
#mapgrid {
outline: 2px solid #333;
background-color: #dfaa37;
border-radius: min(4px, 25%);
animation: blinker 1s infinite;
}

View file

@ -3,7 +3,6 @@
<head>
<title>in?</title>
<link rel="stylesheet" href="style.css">
<link rel="stylesheet" href="c4.css">
<meta name="viewport" content="width=device-width, initial-scale=1">
<script src="form.js"></script>
<script src="shared.js"></script>
@ -22,7 +21,7 @@
<b>Coordinates</b> (click on map)
<div id="map" title="Map for reference only. Not to scale!">
<img src="map.svg" onclick="mapClick(event)">
<img src="80x80.svg" onclick="mapClick(event)">
<div id="mapgrid"></div>
</div>
@ -44,12 +43,9 @@
<label for="hidden">Hide by default</label>
<input type="checkbox" id="hidden">
<label for="last_updated">Last updated</label>
<input id="last_updated" type="text" disabled>
</form>
<button id="delete" class="btn red" autocomplete="off">Delete</button>
<button id="save" class="btn primary" autocomplete="off">Save</button>
<button id="save" class="btn green" autocomplete="off">Save</button>
</body>
</html>

View file

@ -39,7 +39,6 @@ function fillForm() {
document.getElementById('content').value = item.content;
document.getElementById('note').value = item.note;
document.getElementById('hidden').checked = item.hidden;
document.getElementById('last_updated').value = formatTimestamp(item.last_updated);
formCoordsToMap();
}
@ -65,22 +64,8 @@ function knownTypes() {
map[i.type] = true;
}
}
// Get location in document.
const known_types = document.getElementById('knowntypes');
known_types.textContent = `Known types: `;
// Create a button for every known type.
for (let label of Object.keys(map).sort()) {
btn = document.createElement('button');
btn.setAttribute('type', 'button');
btn.textContent = label;
btn.addEventListener('click', function() {
document.getElementById('type').value = label;
});
known_types.appendChild(btn);
}
let types = Object.keys(map).sort().join(', ')
document.getElementById('knowntypes').textContent = `Known types: ${types}.`;
}
function save() {
@ -150,14 +135,14 @@ function del() {
let clicks = {x: [], y: []};
function mapClick(e) {
let x = Math.floor(36 / e.target.width * e.layerX);
let y = Math.floor(18 / e.target.height * e.layerY);
let x = Math.floor(31 / e.target.width * e.layerX);
let y = Math.floor(10 / e.target.height * e.layerY);
let humanPos = (x, y) => {
return `${String.fromCharCode(65 + 16 - y)}${x}`;
return `${String.fromCharCode(65 + 8 - y)}${x}`;
};
if (x > 0 && x < 36 && y > 0 && y < 18) {
if (x > 0 && x < 31 && y > 0 && y < 9) {
if (clicks.x.length > 1) {
clicks.x = [x];
clicks.y = [y];
@ -187,9 +172,3 @@ function formCoordsToMap() {
coordsToMap(coords_bl, coords_tr);
}
}
function formatTimestamp(ts) {
const date = new Date(ts * 1000);
// using Swedish format as a hack to get an iso formatted date
return date.toLocaleDateString("sv", {timeZone: "UTC"}).replace(/\-\d+$/,'')
}

View file

@ -3,7 +3,6 @@
<head>
<title>in?</title>
<link rel="stylesheet" href="style.css">
<link rel="stylesheet" href="c4.css">
<meta name="viewport" content="width=device-width, initial-scale=1">
<script src="index.js"></script>
<script src="shared.js"></script>
@ -12,13 +11,12 @@
<div id="grid">
<div id="searchcontainer">
<input type="text" id="search" value="" autocomplete="off" placeholder="RegEx search" tabindex="1" autofocus>
<p id="itemcount"></p>
<label for="showhidden">Show hidden items</label>
<input type="checkbox" id="showhidden">
</div>
<div id="mapcontainer">
<div id="map">
<img src="map.svg" title="Map for reference only. Not to scale!">
<img src="80x80.svg" title="Map for reference only. Not to scale!">
<div id="mapgrid"></div>
</div>
<p id="mapnote"></p>
@ -39,7 +37,7 @@
</div>
</template>
</div>
<a href="form.html" class="btn primary">+</a>
<a href="form.html" class="btn green">+</a>
</div>
</body>
</html>

View file

@ -17,7 +17,6 @@ function renderItems() {
const container = document.getElementById('results');
const template = document.getElementById('item');
const loading = document.getElementById('loading');
let count = 0;
for (const [id, item] of Object.entries(items)) {
const clone = template.content.cloneNode(true);
@ -32,14 +31,11 @@ function renderItems() {
clone.querySelector(".result").id = `item-${id}`;
clone.querySelector("a").href = `form.html?id=${encodeURIComponent(id)}`;
if (item.hidden) {
clone.querySelector(".result").classList.add('hidden');
} else {
count++;
}
clone.querySelector(".result").classList.add('hidden')
};
container.appendChild(clone);
}
};
loading.remove();
updateCounter(count);
}
function getLocString(items, item) {
@ -71,7 +67,6 @@ function search(e) {
const searchAttrs = ['id', 'name', 'type', 'note', 'content'];
const query = e.target.value;
const regex = new RegExp(query, 'i')
let count = 0;
for (const elem of document.getElementsByClassName('result')) {
const item = items[elem.id.slice(5)];
@ -90,36 +85,10 @@ function search(e) {
if (found) {
elem.classList.remove('filtered');
count++;
} else {
elem.classList.add('filtered');
}
}
// Indicate failed search.
if (count) {
e.target.classList.remove('failed');
} else {
e.target.classList.add('failed');
}
updateCounter(count);
}
function updateCounter(count) {
const itemcount = document.getElementById('itemcount');
switch(count) {
case 0:
itemcount.textContent = 'No items found.';
break;
case 1:
itemcount.textContent = '1 item found.';
break;
default:
itemcount.textContent = `${count} items found.`;
break;
}
}
function showhidden(e){
@ -149,4 +118,4 @@ function showItem(e) {
function hideItem(e) {
clearMap();
}
}

File diff suppressed because one or more lines are too long

Before

Width:  |  Height:  |  Size: 96 KiB

View file

@ -1,12 +1,12 @@
function renderMap(xx, yy) {
if (xx[0] > 0 && xx[0] < 36 && yy[0] > 0 && yy[0] < 18 &&
xx[1] > 0 && xx[1] < 36 && yy[1] > 0 && yy[1] < 18 &&
if (xx[0] > 0 && xx[0] < 31 && yy[0] > 0 && yy[0] < 9 &&
xx[1] > 0 && xx[1] < 31 && yy[1] > 0 && yy[1] < 9 &&
xx[1] >= xx[0] && yy[1] >= yy[0]) {
let grid = document.getElementById('mapgrid');
grid.style.top = `${yy[0]*100/18}%`;
grid.style.left = `${100/36*xx[0]}%`;
grid.style.height = `${(yy[1] - yy[0] + 1) * 100/18}%`;
grid.style.width = `${100/36*(xx[1] - xx[0] + 1)}%`;
grid.style.top = `${yy[0]*10}%`;
grid.style.left = `${100/31*xx[0]}%`;
grid.style.height = `${(yy[1] - yy[0] + 1) * 10}%`;
grid.style.width = `${100/31*(xx[1] - xx[0] + 1)}%`;
} else {
alert(`invalid coordinates x=${xx} y=${yy}`);
}
@ -26,8 +26,8 @@ function coordsToMap(coords_bl, coords_tr) {
let bl_x = parseInt(coords_bl.slice(1));
let tr_x = parseInt(coords_tr.slice(1));
bl_y = 16 - (bl_y.charCodeAt(0) - 65);
tr_y = 16 - (tr_y.charCodeAt(0) - 65);
bl_y = 8 - (bl_y.charCodeAt(0) - 65);
tr_y = 8 - (tr_y.charCodeAt(0) - 65);
renderMap([bl_x, tr_x], [tr_y, bl_y]);
}
}

View file

@ -171,10 +171,10 @@ textarea {
#map {
position: relative;
border: 1px solid black;
background: #4d4d4d;
max-width: 1440px;
background: #fff;
max-width: 800px;
max-height: 50vh;
aspect-ratio: 2 / 1;
aspect-ratio: 31 / 10;
margin: 0 auto;
}
@ -238,4 +238,4 @@ form #map {
max-width: 100%;
text-align: center;
margin: auto;
}
}