{"id":45964369,"url":"https://github.com/phash/playtogether","last_synced_at":"2026-02-28T14:02:33.031Z","repository":{"id":336612957,"uuid":"1150378285","full_name":"phash/playtogether","owner":"phash","description":"Play nice little games together","archived":false,"fork":false,"pushed_at":"2026-02-05T16:45:37.000Z","size":251,"stargazers_count":0,"open_issues_count":0,"forks_count":0,"subscribers_count":0,"default_branch":"master","last_synced_at":"2026-02-05T19:59:42.577Z","etag":null,"topics":[],"latest_commit_sha":null,"homepage":null,"language":"TypeScript","has_issues":true,"has_wiki":null,"has_pages":null,"mirror_url":null,"source_name":null,"license":null,"status":null,"scm":"git","pull_requests_enabled":true,"icon_url":"https://github.com/phash.png","metadata":{"files":{"readme":"README.md","changelog":null,"contributing":null,"funding":null,"license":null,"code_of_conduct":null,"threat_model":null,"audit":null,"citation":null,"codeowners":null,"security":null,"support":null,"governance":null,"roadmap":null,"authors":null,"dei":null,"publiccode":null,"codemeta":null,"zenodo":null,"notice":null,"maintainers":null,"copyright":null,"agents":null,"dco":null,"cla":null}},"created_at":"2026-02-05T07:54:21.000Z","updated_at":"2026-02-05T16:45:42.000Z","dependencies_parsed_at":null,"dependency_job_id":null,"html_url":"https://github.com/phash/playtogether","commit_stats":null,"previous_names":["phash/playtogether"],"tags_count":null,"template":false,"template_full_name":null,"purl":"pkg:github/phash/playtogether","repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/phash%2Fplaytogether","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/phash%2Fplaytogether/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/phash%2Fplaytogether/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/phash%2Fplaytogether/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/phash","download_url":"https://codeload.github.com/phash/playtogether/tar.gz/refs/heads/master","sbom_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/phash%2Fplaytogether/sbom","scorecard":null,"host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":286080680,"owners_count":29936368,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2026-02-28T13:49:17.081Z","status":"ssl_error","status_checked_at":"2026-02-28T13:48:50.396Z","response_time":90,"last_error":"SSL_connect returned=1 errno=0 peeraddr=140.82.121.5:443 state=error: unexpected eof while reading","robots_txt_status":"success","robots_txt_updated_at":"2025-07-24T06:49:26.215Z","robots_txt_url":"https://github.com/robots.txt","online":false,"can_crawl_api":true,"host_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub","repositories_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories","repository_names_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repository_names","owners_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners"}},"keywords":[],"created_at":"2026-02-28T14:01:59.368Z","updated_at":"2026-02-28T14:02:32.994Z","avatar_url":"https://github.com/phash.png","language":"TypeScript","funding_links":[],"categories":[],"sub_categories":[],"readme":"# PlayTogether\n\nEine Multiplayer-Plattform für kleine Spiele, die man gemeinsam über Smartphones spielen kann.\n\n## Features\n\n- **Echtzeit-Multiplayer** via WebSocket\n- **Mobile-First Design** - optimiert für Smartphones\n- **Native Android App** mit Jetpack Compose\n- **Raum-System** mit einfachen 4-stelligen Codes\n- **10 Spielmodi** in 3 Kategorien (Klassisch, Party, Wort-Spiele)\n- **Moody-System** - Virtuelles Haustier mit Belohnungen\n- **Achievement-System** - 18 Achievements zum Freischalten\n- **Freundschafts-System** - Freunde hinzufuegen und verwalten\n- **Playlist-System** - Mehrere Spiele hintereinander spielen\n\n## Projektstruktur\n\n```\nplaytogether/\n├── packages/\n│   ├── shared/     # Gemeinsame Typen und Utilities\n│   ├── server/     # Backend (Express + Socket.io + Prisma)\n│   └── client/     # Frontend (React + Vite)\n├── android/        # Native Android App (Kotlin + Jetpack Compose)\n└── docker-compose.yml\n```\n\n## Schnellstart\n\n### Mit Docker (Empfohlen)\n\n**Voraussetzungen:** Docker und Docker Compose\n\n```bash\n# .env Datei erstellen\ncp .env.example .env\n\n# Container bauen und starten\nmake up\n\n# Oder mit npm:\nnpm run docker:up\n```\n\nDie Anwendung ist dann verfügbar:\n- **Client:** http://localhost (Port 80)\n- **Server:** http://localhost:3001\n\n**Weitere Docker-Befehle:**\n```bash\nmake logs      # Logs anzeigen\nmake down      # Container stoppen\nmake restart   # Container neustarten\nmake dev       # Entwicklungsmodus mit Hot-Reload\n```\n\n### Ohne Docker (Lokale Entwicklung)\n\n**Voraussetzungen:** Node.js 18+\n\n```bash\n# Dependencies installieren\nnpm install\n\n# Shared-Paket bauen\nnpm run build:shared\n\n# Server und Client gleichzeitig starten\nnpm run dev\n```\n\n- Server: http://localhost:3001\n- Client: http://localhost:5173\n\n## Wie es funktioniert\n\n1. **Spiel erstellen**: Ein Spieler erstellt einen Raum und wählt ein Spiel\n2. **Code teilen**: Der 4-stellige Raum-Code wird mit Freunden geteilt\n3. **Beitreten**: Freunde öffnen die App und geben den Code ein\n4. **Spielen**: Der Host startet das Spiel, wenn genug Spieler da sind\n\n## Technologie-Stack\n\n- **Frontend**: React 18, TypeScript, Vite, Zustand\n- **Backend**: Node.js, Express, Socket.io, Prisma\n- **Android**: Kotlin, Jetpack Compose, Material 3\n- **Database**: PostgreSQL, Redis\n- **Shared**: TypeScript Typen und Utilities\n- **Deployment**: Docker, Docker Compose, nginx\n\n## Docker Deployment\n\n### Produktion\n\n```bash\n# Einfaches Deployment\n./scripts/deploy.sh\n\n# Oder manuell:\ndocker compose build\ndocker compose up -d\n```\n\n### Konfiguration\n\nUmgebungsvariablen in `.env`:\n\n| Variable | Beschreibung | Standard |\n|----------|--------------|----------|\n| `SERVER_PORT` | Server-Port (extern) | 3001 |\n| `CLIENT_PORT` | Client-Port (extern) | 80 |\n| `VITE_SERVER_URL` | WebSocket URL | http://localhost:3001 |\n| `CLIENT_URL` | CORS Origin | http://localhost |\n\n### Container\n\n| Service | Image | Port | Beschreibung |\n|---------|-------|------|--------------|\n| `server` | Node.js Alpine | 3001 | WebSocket + REST API |\n| `client` | nginx Alpine | 80 | Static Files + SPA |\n\n### Health Checks\n\n```bash\n# Server\ncurl http://localhost:3001/api/health\n\n# Client\ncurl http://localhost/health\n```\n\n## Verfuegbare Spiele (10)\n\n### Klassische Spiele\n| Spiel | Spieler | Beschreibung |\n|-------|---------|--------------|\n| Quiz Champ | 2-8 | 145 Fragen, 20s Timer, Streak-Bonus |\n| Tic Tac Toe | 2-8 | Klassisches Strategiespiel im Turniermodus |\n| Schere Stein Papier | 2-16 | Turnier-Paarungen |\n| Reaktions-Test | 2-8 | Reagiere so schnell wie moeglich auf das Signal |\n\n### Party \u0026 Spass\n| Spiel | Spieler | Beschreibung |\n|-------|---------|--------------|\n| Entweder/Oder | 3-20 | Was waehlt die Mehrheit? Stimme ab! |\n| Gluecksrad | 2-6 | Drehe das Rad und loese die Phrase |\n| Emoji Malen | 3-10 | Male mit Emojis und lass andere raten |\n\n### Wort-Spiele\n| Spiel | Spieler | Beschreibung |\n|-------|---------|--------------|\n| Anagramme | 2-8 | Entwirre das verwuerfelte Wort |\n| Galgenmaennchen | 2-8 | Errate das Wort Buchstabe fuer Buchstabe |\n| Wort-Raten | 4-12 | Erklaere Woerter und lass Freunde raten |\n\n## Android App\n\nDie native Android-App ist verfügbar als APK-Download:\n\n```\nhttp://localhost:3003/download/playtogether.apk\n```\n\n**Features:**\n- Native Jetpack Compose UI\n- Material Design 3\n- WebSocket-Integration\n- Offline-fähig (Lobby)\n\n## Lizenz\n\nMIT\n","project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fphash%2Fplaytogether","html_url":"https://awesome.ecosyste.ms/projects/github.com%2Fphash%2Fplaytogether","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fphash%2Fplaytogether/lists"}