Compare commits
No commits in common. "master" and "main" have entirely different histories.
171
.gitignore
vendored
171
.gitignore
vendored
@ -1,55 +1,170 @@
|
|||||||
# Byte-compiled files
|
# ---> Python
|
||||||
|
# Byte-compiled / optimized / DLL files
|
||||||
**pycache**/
|
__pycache__/
|
||||||
*.py[cod]
|
*.py[cod]
|
||||||
*$py.class
|
*$py.class
|
||||||
|
|
||||||
# Virtual environments
|
# C extensions
|
||||||
|
*.so
|
||||||
.env/
|
|
||||||
.venv/
|
|
||||||
env/
|
|
||||||
venv/
|
|
||||||
|
|
||||||
# Distribution
|
|
||||||
|
|
||||||
|
# Distribution / packaging
|
||||||
|
.Python
|
||||||
build/
|
build/
|
||||||
|
develop-eggs/
|
||||||
dist/
|
dist/
|
||||||
|
downloads/
|
||||||
|
eggs/
|
||||||
|
.eggs/
|
||||||
|
lib/
|
||||||
|
lib64/
|
||||||
|
parts/
|
||||||
|
sdist/
|
||||||
|
var/
|
||||||
|
wheels/
|
||||||
|
share/python-wheels/
|
||||||
*.egg-info/
|
*.egg-info/
|
||||||
|
.installed.cfg
|
||||||
*.egg
|
*.egg
|
||||||
|
MANIFEST
|
||||||
|
|
||||||
|
# PyInstaller
|
||||||
|
# Usually these files are written by a python script from a template
|
||||||
|
# before PyInstaller builds the exe, so as to inject date/other infos into it.
|
||||||
|
*.manifest
|
||||||
|
*.spec
|
||||||
|
|
||||||
# Installer logs
|
# Installer logs
|
||||||
|
|
||||||
pip-log.txt
|
pip-log.txt
|
||||||
pip-delete-this-directory.txt
|
pip-delete-this-directory.txt
|
||||||
|
|
||||||
# Python coverage
|
# Unit test / coverage reports
|
||||||
|
|
||||||
htmlcov/
|
htmlcov/
|
||||||
.tox/
|
.tox/
|
||||||
.coverage
|
|
||||||
.pytest_cache/
|
|
||||||
.nox/
|
.nox/
|
||||||
|
.coverage
|
||||||
|
.coverage.*
|
||||||
|
.cache
|
||||||
|
nosetests.xml
|
||||||
|
coverage.xml
|
||||||
|
*.cover
|
||||||
|
*.py,cover
|
||||||
|
.hypothesis/
|
||||||
|
.pytest_cache/
|
||||||
|
cover/
|
||||||
|
|
||||||
# Logs
|
# Translations
|
||||||
|
*.mo
|
||||||
|
*.pot
|
||||||
|
|
||||||
|
# Django stuff:
|
||||||
*.log
|
*.log
|
||||||
|
local_settings.py
|
||||||
|
db.sqlite3
|
||||||
|
db.sqlite3-journal
|
||||||
|
|
||||||
# IDEs
|
# Flask stuff:
|
||||||
|
instance/
|
||||||
|
.webassets-cache
|
||||||
|
|
||||||
.vscode/
|
# Scrapy stuff:
|
||||||
.idea/
|
.scrapy
|
||||||
|
|
||||||
# macOS
|
# Sphinx documentation
|
||||||
|
docs/_build/
|
||||||
|
|
||||||
.DS_Store
|
# PyBuilder
|
||||||
|
.pybuilder/
|
||||||
|
target/
|
||||||
|
|
||||||
# Linux
|
# Jupyter Notebook
|
||||||
|
.ipynb_checkpoints
|
||||||
|
|
||||||
*~
|
# IPython
|
||||||
|
profile_default/
|
||||||
|
ipython_config.py
|
||||||
|
|
||||||
# Data files and locks
|
# pyenv
|
||||||
|
# For a library or package, you might want to ignore these files since the code is
|
||||||
|
# intended to run in multiple environments; otherwise, check them in:
|
||||||
|
# .python-version
|
||||||
|
|
||||||
|
# pipenv
|
||||||
|
# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
|
||||||
|
# However, in case of collaboration, if having platform-specific dependencies or dependencies
|
||||||
|
# having no cross-platform support, pipenv may install dependencies that don't work, or not
|
||||||
|
# install all needed dependencies.
|
||||||
|
#Pipfile.lock
|
||||||
|
|
||||||
|
# UV
|
||||||
|
# Similar to Pipfile.lock, it is generally recommended to include uv.lock in version control.
|
||||||
|
# This is especially recommended for binary packages to ensure reproducibility, and is more
|
||||||
|
# commonly ignored for libraries.
|
||||||
|
#uv.lock
|
||||||
|
|
||||||
|
# poetry
|
||||||
|
# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control.
|
||||||
|
# This is especially recommended for binary packages to ensure reproducibility, and is more
|
||||||
|
# commonly ignored for libraries.
|
||||||
|
# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control
|
||||||
|
#poetry.lock
|
||||||
|
|
||||||
|
# pdm
|
||||||
|
# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control.
|
||||||
|
#pdm.lock
|
||||||
|
# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it
|
||||||
|
# in version control.
|
||||||
|
# https://pdm.fming.dev/latest/usage/project/#working-with-version-control
|
||||||
|
.pdm.toml
|
||||||
|
.pdm-python
|
||||||
|
.pdm-build/
|
||||||
|
|
||||||
|
# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm
|
||||||
|
__pypackages__/
|
||||||
|
|
||||||
|
# Celery stuff
|
||||||
|
celerybeat-schedule
|
||||||
|
celerybeat.pid
|
||||||
|
|
||||||
|
# SageMath parsed files
|
||||||
|
*.sage.py
|
||||||
|
|
||||||
|
# Environments
|
||||||
|
.env
|
||||||
|
.venv
|
||||||
|
env/
|
||||||
|
venv/
|
||||||
|
ENV/
|
||||||
|
env.bak/
|
||||||
|
venv.bak/
|
||||||
|
|
||||||
|
# Spyder project settings
|
||||||
|
.spyderproject
|
||||||
|
.spyproject
|
||||||
|
|
||||||
|
# Rope project settings
|
||||||
|
.ropeproject
|
||||||
|
|
||||||
|
# mkdocs documentation
|
||||||
|
/site
|
||||||
|
|
||||||
|
# mypy
|
||||||
|
.mypy_cache/
|
||||||
|
.dmypy.json
|
||||||
|
dmypy.json
|
||||||
|
|
||||||
|
# Pyre type checker
|
||||||
|
.pyre/
|
||||||
|
|
||||||
|
# pytype static type analyzer
|
||||||
|
.pytype/
|
||||||
|
|
||||||
|
# Cython debug symbols
|
||||||
|
cython_debug/
|
||||||
|
|
||||||
|
# PyCharm
|
||||||
|
# JetBrains specific template is maintained in a separate JetBrains.gitignore that can
|
||||||
|
# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
|
||||||
|
# and can be added to the global gitignore or merged into this file. For a more nuclear
|
||||||
|
# option (not recommended) you can uncomment the following to ignore the entire idea folder.
|
||||||
|
#.idea/
|
||||||
|
|
||||||
data/
|
|
||||||
*.json
|
|
||||||
*.lock
|
|
||||||
|
|||||||
343
README.md
343
README.md
@ -1,342 +1,3 @@
|
|||||||
# Tour Voting App
|
# TourPlanner
|
||||||
|
|
||||||
Tour Voting App is a full-stack web application for planning tours, proposing ideas, and voting on those ideas. It consists of:
|
Tour Planner is a lightweight FastAPI-based service that lets you create “tours,” propose ideas for each tour, and vote on those ideas. It stores each tour as a JSON file in a configurable data directory (with file‐level locking for safe concurrent access), and ships with a minimal static HTML/JavaScript frontend along with helper scripts for local development or deployment under Supervisor.
|
||||||
|
|
||||||
* **Backend API** built with FastAPI
|
|
||||||
* **Data persistence** using per-tour JSON files in a configurable directory, with concurrency safety via file locks
|
|
||||||
* **Frontend**: Single-page HTML/JavaScript application that consumes the API
|
|
||||||
* **Helper scripts** for local development or Supervisor-based deployment
|
|
||||||
|
|
||||||
## Table of Contents
|
|
||||||
|
|
||||||
1. Features
|
|
||||||
2. Tech Stack
|
|
||||||
3. Prerequisites
|
|
||||||
4. Installation
|
|
||||||
5. Configuration
|
|
||||||
6. Development
|
|
||||||
7. Running in Production
|
|
||||||
8. Frontend Usage
|
|
||||||
9. API Reference
|
|
||||||
* Models
|
|
||||||
* Endpoints
|
|
||||||
10. Directory Structure
|
|
||||||
11. Contributing
|
|
||||||
12. License
|
|
||||||
|
|
||||||
## Features
|
|
||||||
|
|
||||||
* Create and manage tours with metadata (name, start/end dates, description)
|
|
||||||
* Add multiple ideas per tour, each with its own name, description, and optional time window
|
|
||||||
* Vote on ideas by recording unique voter names
|
|
||||||
* No external database: JSON file–based persistence
|
|
||||||
* Concurrency-safe via file locking
|
|
||||||
* Minimal static HTML/JavaScript frontend for user interaction
|
|
||||||
* Helper scripts for local development or Supervisor deployment
|
|
||||||
|
|
||||||
## Tech Stack
|
|
||||||
|
|
||||||
* **Backend**: Python 3.10+, FastAPI, Uvicorn, Pydantic, filelock
|
|
||||||
* **Frontend**: HTML, CSS, JavaScript (Fetch API)
|
|
||||||
* **Process Control (optional)**: Supervisor
|
|
||||||
* **Data storage**: JSON files, one per tour
|
|
||||||
|
|
||||||
## Prerequisites
|
|
||||||
|
|
||||||
* Python 3.10 or higher
|
|
||||||
* pip (Python package installer)
|
|
||||||
* (Optional) Supervisor for process management
|
|
||||||
|
|
||||||
## Installation
|
|
||||||
|
|
||||||
1. **Clone the repository**
|
|
||||||
git clone [https://github.com/your-organization/tour-voting-app.git](https://github.com/your-organization/tour-voting-app.git)
|
|
||||||
cd tour-voting-app
|
|
||||||
|
|
||||||
2. **Create virtual environment and install dependencies**
|
|
||||||
python3 -m venv .venv
|
|
||||||
source .venv/bin/activate
|
|
||||||
pip install -r requirements.txt
|
|
||||||
|
|
||||||
If `requirements.txt` is not provided, install directly:
|
|
||||||
pip install fastapi uvicorn pydantic filelock
|
|
||||||
|
|
||||||
## Configuration
|
|
||||||
|
|
||||||
Configure via environment variables or command-line flags:
|
|
||||||
|
|
||||||
* **DATA\_DIR**: directory to store tour JSON files (default: `./data`)
|
|
||||||
* **HOST**: address to bind the API (default: `0.0.0.0`)
|
|
||||||
* **PORT**: port for the API (default: `8000`)
|
|
||||||
|
|
||||||
Example using environment variables:
|
|
||||||
|
|
||||||
```
|
|
||||||
export DATA_DIR=./data
|
|
||||||
export HOST=0.0.0.0
|
|
||||||
export PORT=8000
|
|
||||||
```
|
|
||||||
|
|
||||||
Or via flags:
|
|
||||||
|
|
||||||
```
|
|
||||||
python server.py --data-dir ./data --host 0.0.0.0 --port 8000
|
|
||||||
```
|
|
||||||
|
|
||||||
## Development
|
|
||||||
|
|
||||||
1. Activate your virtual environment.
|
|
||||||
2. Start the server in reload mode:
|
|
||||||
|
|
||||||
```
|
|
||||||
uvicorn server:app --reload --host 0.0.0.0 --port 8000
|
|
||||||
```
|
|
||||||
3. Open `index.html` in your browser (or serve it via a local HTTP server to avoid CORS issues).
|
|
||||||
|
|
||||||
## Running in Production
|
|
||||||
|
|
||||||
Use the provided `service.sh` script and Supervisor config:
|
|
||||||
|
|
||||||
**service.sh**
|
|
||||||
|
|
||||||
```bash
|
|
||||||
#!/usr/bin/env bash
|
|
||||||
cd /path/to/tour-voting-app || exit 1
|
|
||||||
exec uvicorn server:app \
|
|
||||||
--host 0.0.0.0 \
|
|
||||||
--port 3002 \
|
|
||||||
--loop uvloop \
|
|
||||||
--workers 2 \
|
|
||||||
--access-log \
|
|
||||||
--log-level info
|
|
||||||
```
|
|
||||||
|
|
||||||
**tour.ini** (Supervisor)
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[program:tour_voting_app]
|
|
||||||
command=/path/to/service.sh
|
|
||||||
directory=/path/to/tour-voting-app
|
|
||||||
autostart=true
|
|
||||||
autorestart=true
|
|
||||||
stderr_logfile=/var/log/tour_voting_app.err.log
|
|
||||||
stdout_logfile=/var/log/tour_voting_app.out.log
|
|
||||||
```
|
|
||||||
|
|
||||||
## Frontend Usage
|
|
||||||
|
|
||||||
The frontend is a single `index.html` file that interacts with the API. By default, `API_URL` is set to `http://localhost:8000/tour/v1`. To point it at a remote server, edit the `API_URL` constant near the top of `index.html`.
|
|
||||||
|
|
||||||
## API Reference
|
|
||||||
|
|
||||||
Base path: `/tour/v1`
|
|
||||||
|
|
||||||
### Models
|
|
||||||
|
|
||||||
**Tour**
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"id": "string",
|
|
||||||
"name": "string",
|
|
||||||
"description": "string",
|
|
||||||
"startDate": "ISO-8601 timestamp",
|
|
||||||
"endDate": "ISO-8601 timestamp",
|
|
||||||
"createdAt": "ISO-8601 timestamp",
|
|
||||||
"ideas": [ Idea ]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
**Idea**
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"id": "string",
|
|
||||||
"name": "string",
|
|
||||||
"description": "string",
|
|
||||||
"startTime": "ISO-8601 timestamp | null",
|
|
||||||
"endTime": "ISO-8601 timestamp | null",
|
|
||||||
"voters": [ "string" ]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Endpoints
|
|
||||||
|
|
||||||
#### 1. List Tours
|
|
||||||
|
|
||||||
* **Method**: GET
|
|
||||||
* **URL**: `/tours`
|
|
||||||
* **Response**: `200 OK`
|
|
||||||
|
|
||||||
```json
|
|
||||||
[ { Tour }, ... ]
|
|
||||||
```
|
|
||||||
|
|
||||||
#### 2. Create a Tour
|
|
||||||
|
|
||||||
* **Method**: POST
|
|
||||||
* **URL**: `/tours`
|
|
||||||
* **Request Body**:
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"name": "Europe Explorer",
|
|
||||||
"description": "A tour across Europe",
|
|
||||||
"startDate": "2025-07-01T00:00:00Z",
|
|
||||||
"endDate": "2025-07-15T00:00:00Z"
|
|
||||||
}
|
|
||||||
```
|
|
||||||
* **Response**: `201 Created`
|
|
||||||
|
|
||||||
```json
|
|
||||||
{ Tour }
|
|
||||||
```
|
|
||||||
|
|
||||||
#### 3. Get Tour Details
|
|
||||||
|
|
||||||
* **Method**: GET
|
|
||||||
* **URL**: `/tours/{tour_id}`
|
|
||||||
* **Path Parameters**:
|
|
||||||
|
|
||||||
* `tour_id` (string): ID of the tour
|
|
||||||
* **Response**: `200 OK`
|
|
||||||
|
|
||||||
```json
|
|
||||||
{ Tour }
|
|
||||||
```
|
|
||||||
* **Error**: `404 Not Found` if tour does not exist.
|
|
||||||
|
|
||||||
#### 4. Add an Idea to a Tour
|
|
||||||
|
|
||||||
* **Method**: POST
|
|
||||||
* **URL**: `/tours/{tour_id}/ideas`
|
|
||||||
* **Path Parameters**:
|
|
||||||
|
|
||||||
* `tour_id` (string): ID of the tour
|
|
||||||
* **Request Body**:
|
|
||||||
|
|
||||||
```json
|
|
||||||
{
|
|
||||||
"name": "Visit the Louvre",
|
|
||||||
"description": "Guided museum tour",
|
|
||||||
"startTime": "2025-07-03T10:00:00Z",
|
|
||||||
"endTime": "2025-07-03T14:00:00Z"
|
|
||||||
}
|
|
||||||
```
|
|
||||||
* **Response**: `201 Created`
|
|
||||||
|
|
||||||
```json
|
|
||||||
{ Idea }
|
|
||||||
```
|
|
||||||
* **Error**: `404 Not Found` if tour does not exist.
|
|
||||||
|
|
||||||
#### 5. Vote for an Idea
|
|
||||||
|
|
||||||
* **Method**: POST
|
|
||||||
* **URL**: `/tours/{tour_id}/ideas/{idea_id}/vote`
|
|
||||||
* **Path Parameters**:
|
|
||||||
|
|
||||||
* `tour_id` (string): ID of the tour
|
|
||||||
* `idea_id` (string): ID of the idea
|
|
||||||
* **Request Body**:
|
|
||||||
|
|
||||||
```json
|
|
||||||
{ "voterName": "alice@example.com" }
|
|
||||||
```
|
|
||||||
* **Response**: `200 OK`
|
|
||||||
|
|
||||||
```json
|
|
||||||
{ Idea } // Updated idea with new voter
|
|
||||||
```
|
|
||||||
* **Errors**:
|
|
||||||
|
|
||||||
* `404 Not Found` if tour or idea does not exist.
|
|
||||||
* `400 Bad Request` if voter name is missing or already voted.
|
|
||||||
|
|
||||||
## Directory Structure
|
|
||||||
|
|
||||||
```
|
|
||||||
tour-voting-app/
|
|
||||||
├── server.py # FastAPI application
|
|
||||||
├── index.html # Static frontend
|
|
||||||
├── service.sh # Startup helper script
|
|
||||||
├── tour.ini # Supervisor config
|
|
||||||
├── data/ # JSON files for tours (runtime)
|
|
||||||
├── requirements.txt # Python dependencies (optional)
|
|
||||||
└── README.md # Project documentation
|
|
||||||
```
|
|
||||||
|
|
||||||
## Contributing
|
|
||||||
|
|
||||||
1. Fork the repository.
|
|
||||||
2. Create a feature branch:
|
|
||||||
git checkout -b feature/my-feature
|
|
||||||
3. Commit your changes:
|
|
||||||
git commit -m "Add my feature"
|
|
||||||
4. Push to your branch:
|
|
||||||
git push origin feature/my-feature
|
|
||||||
5. Open a pull request.
|
|
||||||
|
|
||||||
## License
|
|
||||||
|
|
||||||
This project is licensed under the WTFPL License.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
.gitignore
|
|
||||||
|
|
||||||
# Byte-compiled files
|
|
||||||
|
|
||||||
**pycache**/
|
|
||||||
\*.py\[cod]
|
|
||||||
\*\$py.class
|
|
||||||
|
|
||||||
# Virtual environments
|
|
||||||
|
|
||||||
.env/
|
|
||||||
.venv/
|
|
||||||
env/
|
|
||||||
venv/
|
|
||||||
|
|
||||||
# Distribution
|
|
||||||
|
|
||||||
build/
|
|
||||||
dist/
|
|
||||||
\*.egg-info/
|
|
||||||
\*.egg
|
|
||||||
|
|
||||||
# Installer logs
|
|
||||||
|
|
||||||
pip-log.txt
|
|
||||||
pip-delete-this-directory.txt
|
|
||||||
|
|
||||||
# Python coverage
|
|
||||||
|
|
||||||
htmlcov/
|
|
||||||
.tox/
|
|
||||||
.coverage
|
|
||||||
.pytest\_cache/
|
|
||||||
.nox/
|
|
||||||
|
|
||||||
# Logs
|
|
||||||
|
|
||||||
\*.log
|
|
||||||
|
|
||||||
# IDEs
|
|
||||||
|
|
||||||
.vscode/
|
|
||||||
.idea/
|
|
||||||
|
|
||||||
# macOS
|
|
||||||
|
|
||||||
.DS\_Store
|
|
||||||
|
|
||||||
# Linux
|
|
||||||
|
|
||||||
\*\~
|
|
||||||
|
|
||||||
# Data files and locks
|
|
||||||
|
|
||||||
data/
|
|
||||||
\*.json
|
|
||||||
\*.lock
|
|
||||||
1080
index.html
1080
index.html
File diff suppressed because it is too large
Load Diff
@ -1,4 +0,0 @@
|
|||||||
fastapi
|
|
||||||
uvicorn
|
|
||||||
pydantic
|
|
||||||
filelock
|
|
||||||
231
server.py
231
server.py
@ -1,231 +0,0 @@
|
|||||||
import os
|
|
||||||
import json
|
|
||||||
import uuid
|
|
||||||
from typing import Optional
|
|
||||||
from datetime import datetime
|
|
||||||
from fastapi import FastAPI, HTTPException, Path
|
|
||||||
from pydantic import BaseModel, Field
|
|
||||||
from filelock import FileLock
|
|
||||||
import argparse
|
|
||||||
import uvicorn
|
|
||||||
|
|
||||||
app = FastAPI(title="Tour Voting API")
|
|
||||||
|
|
||||||
# ─── Pydantic MODELS ──────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
class Idea(BaseModel):
|
|
||||||
id: str
|
|
||||||
name: str
|
|
||||||
start_time: Optional[datetime] = None
|
|
||||||
end_time: Optional[datetime] = None
|
|
||||||
description: str
|
|
||||||
voters: list[str] = Field(default_factory=list)
|
|
||||||
decided: bool = False
|
|
||||||
|
|
||||||
class Tour(BaseModel):
|
|
||||||
id: str
|
|
||||||
name: str
|
|
||||||
start_date: Optional[datetime] = None
|
|
||||||
end_date: Optional[datetime] = None
|
|
||||||
description: str
|
|
||||||
ideas: list[Idea] = Field(default_factory=list)
|
|
||||||
createdAt: datetime
|
|
||||||
|
|
||||||
class TourCreate(BaseModel):
|
|
||||||
name: str
|
|
||||||
start_date: Optional[datetime] = None
|
|
||||||
end_date: Optional[datetime] = None
|
|
||||||
description: str
|
|
||||||
|
|
||||||
class IdeaCreate(BaseModel):
|
|
||||||
name: str
|
|
||||||
start_time: Optional[datetime] = None
|
|
||||||
end_time: Optional[datetime] = None
|
|
||||||
description: str
|
|
||||||
|
|
||||||
class Vote(BaseModel):
|
|
||||||
voterName: str
|
|
||||||
|
|
||||||
# ─── GLOBAL CONFIG ────────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
DATA_DIR = "" # will be overridden via command line
|
|
||||||
|
|
||||||
def get_tour_filepath(tour_id: str) -> str:
|
|
||||||
fname = f"{tour_id}.json"
|
|
||||||
full = os.path.abspath(os.path.join(DATA_DIR, fname))
|
|
||||||
if not full.startswith(os.path.abspath(DATA_DIR) + os.sep):
|
|
||||||
raise HTTPException(status_code=400, detail="Invalid tour ID")
|
|
||||||
return full
|
|
||||||
|
|
||||||
# ─── ENDPOINT: CREATE TOUR ─────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
@app.post("/tour/v1/tours", response_model=Tour)
|
|
||||||
def create_tour(tour: TourCreate):
|
|
||||||
"""
|
|
||||||
Create a new tour. Generates its own UUID and timestamp, then writes to {tour_id}.json.
|
|
||||||
"""
|
|
||||||
tour_id = str(uuid.uuid4())
|
|
||||||
created_at = datetime.utcnow()
|
|
||||||
|
|
||||||
new_tour = Tour(
|
|
||||||
id=tour_id,
|
|
||||||
name=tour.name,
|
|
||||||
start_date=tour.start_date,
|
|
||||||
end_date=tour.end_date,
|
|
||||||
description=tour.description,
|
|
||||||
ideas=[],
|
|
||||||
createdAt=created_at,
|
|
||||||
)
|
|
||||||
|
|
||||||
filepath = get_tour_filepath(tour_id)
|
|
||||||
with open(filepath, "w") as f:
|
|
||||||
# Using default=str so that datetime objects become ISO strings
|
|
||||||
json.dump(new_tour.model_dump(), f, default=str)
|
|
||||||
|
|
||||||
return new_tour
|
|
||||||
|
|
||||||
# ─── ENDPOINT: GET TOUR ────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
@app.get("/tour/v1/tours/{tour_id}", response_model=Tour)
|
|
||||||
def get_tour(tour_id: str = Path(...)):
|
|
||||||
"""
|
|
||||||
Retrieve an existing tour by ID. Loads {tour_id}.json and parses back into a Tour.
|
|
||||||
"""
|
|
||||||
filepath = get_tour_filepath(tour_id)
|
|
||||||
if not os.path.exists(filepath):
|
|
||||||
raise HTTPException(status_code=404, detail="Tour not found")
|
|
||||||
|
|
||||||
with open(filepath) as f:
|
|
||||||
data = json.load(f)
|
|
||||||
|
|
||||||
if data.get("start_date"):
|
|
||||||
data["start_date"] = datetime.fromisoformat(data["start_date"])
|
|
||||||
if data.get("end_date"):
|
|
||||||
data["end_date"] = datetime.fromisoformat(data["end_date"])
|
|
||||||
data["createdAt"] = datetime.fromisoformat(data["createdAt"])
|
|
||||||
|
|
||||||
# Convert nested idea timestamps as well
|
|
||||||
for idea in data["ideas"]:
|
|
||||||
if idea.get("start_time"):
|
|
||||||
idea["start_time"] = datetime.fromisoformat(idea["start_time"])
|
|
||||||
if idea.get("end_time"):
|
|
||||||
idea["end_time"] = datetime.fromisoformat(idea["end_time"])
|
|
||||||
return Tour(**data)
|
|
||||||
|
|
||||||
# ─── ENDPOINT: ADD IDEA ─────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
@app.post("/tour/v1/tours/{tour_id}/ideas", response_model=Idea)
|
|
||||||
def add_idea(tour_id: str, idea: IdeaCreate):
|
|
||||||
"""
|
|
||||||
Add a new idea to the given tour. Generates a new UUID for the idea and appends it.
|
|
||||||
"""
|
|
||||||
filepath = get_tour_filepath(tour_id)
|
|
||||||
if not os.path.exists(filepath):
|
|
||||||
raise HTTPException(status_code=404, detail="Tour not found")
|
|
||||||
|
|
||||||
lock = FileLock(filepath + ".lock")
|
|
||||||
with lock:
|
|
||||||
with open(filepath) as f:
|
|
||||||
data = json.load(f)
|
|
||||||
|
|
||||||
if any(i.get("decided") for i in data["ideas"]):
|
|
||||||
raise HTTPException(status_code=403, detail="Idea already decided")
|
|
||||||
|
|
||||||
idea_id = str(uuid.uuid4())
|
|
||||||
new_idea = Idea(
|
|
||||||
id=idea_id,
|
|
||||||
name=idea.name,
|
|
||||||
description=idea.description,
|
|
||||||
voters=[],
|
|
||||||
start_time=idea.start_time,
|
|
||||||
end_time=idea.end_time,
|
|
||||||
decided=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
data["ideas"].append(new_idea.model_dump())
|
|
||||||
with open(filepath, "w") as f:
|
|
||||||
json.dump(data, f, default=str)
|
|
||||||
|
|
||||||
return new_idea
|
|
||||||
|
|
||||||
# ─── ENDPOINT: VOTE FOR IDEA ───────────────────────────────────────────────────
|
|
||||||
|
|
||||||
@app.post("/tour/v1/tours/{tour_id}/ideas/{idea_id}/vote", response_model=Idea)
|
|
||||||
def vote_idea(tour_id: str, idea_id: str, vote: Vote):
|
|
||||||
"""
|
|
||||||
Cast a vote on a specific idea. Adds voterName if not already present.
|
|
||||||
"""
|
|
||||||
filepath = get_tour_filepath(tour_id)
|
|
||||||
if not os.path.exists(filepath):
|
|
||||||
raise HTTPException(status_code=404, detail="Tour not found")
|
|
||||||
|
|
||||||
lock = FileLock(filepath + ".lock")
|
|
||||||
with lock:
|
|
||||||
with open(filepath) as f:
|
|
||||||
data = json.load(f)
|
|
||||||
|
|
||||||
if any(i.get("decided") for i in data["ideas"]):
|
|
||||||
raise HTTPException(status_code=403, detail="Idea already decided")
|
|
||||||
for idea in data["ideas"]:
|
|
||||||
if idea["id"] == idea_id:
|
|
||||||
if vote.voterName not in idea["voters"]:
|
|
||||||
idea["voters"].append(vote.voterName)
|
|
||||||
# Persist the change
|
|
||||||
with open(filepath, "w") as f:
|
|
||||||
json.dump(data, f, default=str)
|
|
||||||
return Idea(**idea)
|
|
||||||
|
|
||||||
raise HTTPException(status_code=404, detail="Idea not found")
|
|
||||||
|
|
||||||
# ─── ENDPOINT: DECIDE IDEA ────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
@app.post("/tour/v1/tours/{tour_id}/ideas/{idea_id}/decide", response_model=Idea)
|
|
||||||
def decide_idea(tour_id: str, idea_id: str):
|
|
||||||
"""Mark an idea as decided. Only one idea can be decided per tour."""
|
|
||||||
filepath = get_tour_filepath(tour_id)
|
|
||||||
if not os.path.exists(filepath):
|
|
||||||
raise HTTPException(status_code=404, detail="Tour not found")
|
|
||||||
|
|
||||||
lock = FileLock(filepath + ".lock")
|
|
||||||
with lock:
|
|
||||||
with open(filepath) as f:
|
|
||||||
data = json.load(f)
|
|
||||||
|
|
||||||
if any(i.get("decided") for i in data["ideas"]):
|
|
||||||
raise HTTPException(status_code=403, detail="Idea already decided")
|
|
||||||
|
|
||||||
for idea in data["ideas"]:
|
|
||||||
if idea["id"] == idea_id:
|
|
||||||
idea["decided"] = True
|
|
||||||
with open(filepath, "w") as f:
|
|
||||||
json.dump(data, f, default=str)
|
|
||||||
return Idea(**idea)
|
|
||||||
|
|
||||||
raise HTTPException(status_code=404, detail="Idea not found")
|
|
||||||
|
|
||||||
# ─── MAIN: RUN UVIDORN WITH COMMAND-LINE DATA DIR ──────────────────────────────
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
parser = argparse.ArgumentParser(description="Run Tour Voting API server")
|
|
||||||
parser.add_argument(
|
|
||||||
"--data-dir",
|
|
||||||
required=True,
|
|
||||||
help="Directory in which to store tour JSON files (one file per tour_id)",
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"--host",
|
|
||||||
default="0.0.0.0",
|
|
||||||
help="Host to run the server on",
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
|
||||||
"--port",
|
|
||||||
type=int,
|
|
||||||
default=8000,
|
|
||||||
help="Port to run the server on",
|
|
||||||
)
|
|
||||||
args = parser.parse_args()
|
|
||||||
|
|
||||||
DATA_DIR = args.data_dir
|
|
||||||
os.makedirs(DATA_DIR, exist_ok=True)
|
|
||||||
|
|
||||||
uvicorn.run(app, host=args.host, port=args.port, reload=False)
|
|
||||||
@ -1,4 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
cd /home/kaizen/repos/tourplanner || exit 1
|
|
||||||
exec python3.11 server.py --port=3002 --data-dir=/home/kaizen/repos/tourplanner/data
|
|
||||||
Loading…
x
Reference in New Issue
Block a user