Compare commits
3 Commits
5f977022e4
...
5f2cd56325
| Author | SHA1 | Date | |
|---|---|---|---|
| 5f2cd56325 | |||
| 68608a995b | |||
| bc859085b8 |
27
API.apib
Normal file
27
API.apib
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
#Group Bookings
|
||||||
|
|
||||||
|
## Bookings Collection [/time{?cardID}]
|
||||||
|
|
||||||
|
### List all bookings for specific card_id [GET]
|
||||||
|
|
||||||
|
+ Parameters
|
||||||
|
+ cardID:test_card (string) - the id of the rfid card
|
||||||
|
|
||||||
|
+ Response 200 (application/json)
|
||||||
|
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"cradID": "test_card",
|
||||||
|
"readerID": "test_reader",
|
||||||
|
"bookingTyp": 2,
|
||||||
|
"loggedTime": "2024-09-05T08:37:53.117641Z",
|
||||||
|
"id": 5
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"cradID": "test_card",
|
||||||
|
"readerID": "mytest",
|
||||||
|
"bookingTyp": 1,
|
||||||
|
"loggedTime": "2024-09-05T08:51:12.670827Z",
|
||||||
|
"id": 6
|
||||||
|
},
|
||||||
|
]
|
||||||
@@ -2,39 +2,47 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"arbeitszeitmessung/models"
|
"arbeitszeitmessung/models"
|
||||||
"database/sql"
|
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
_ "github.com/lib/pq"
|
_ "github.com/lib/pq"
|
||||||
)
|
)
|
||||||
|
|
||||||
var DB *sql.DB
|
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
var err error
|
var err error
|
||||||
DB, err = OpenDatabase()
|
models.DB, err = OpenDatabase()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
defer DB.Close()
|
defer models.DB.Close()
|
||||||
|
|
||||||
|
http.HandleFunc("/time/new", timeCreateHandler)
|
||||||
http.HandleFunc("/time", timeHandler)
|
http.HandleFunc("/time", timeHandler)
|
||||||
|
|
||||||
fmt.Println("Server is running at http://localhost:8080")
|
fmt.Printf("Server is running at http://localhost:8080 exposed to port %s\n", getEnv("EXPOSED_PORT", "8000"))
|
||||||
log.Fatal(http.ListenAndServe(":8080", nil))
|
log.Fatal(http.ListenAndServe(":8080", nil))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func timeCreateHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch r.Method {
|
||||||
|
case "PUT":
|
||||||
|
createBooking(w, r)
|
||||||
|
default:
|
||||||
|
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func timeHandler(w http.ResponseWriter, r *http.Request) {
|
func timeHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
switch r.Method {
|
switch r.Method {
|
||||||
case "PUT":
|
|
||||||
createBooking(w, r)
|
|
||||||
case "GET":
|
case "GET":
|
||||||
getBookings(w, r)
|
getBookings(w, r)
|
||||||
|
case "PUT":
|
||||||
|
updateBooking(w, r)
|
||||||
default:
|
default:
|
||||||
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
||||||
}
|
}
|
||||||
@@ -43,7 +51,7 @@ func timeHandler(w http.ResponseWriter, r *http.Request) {
|
|||||||
func createBooking(w http.ResponseWriter, r *http.Request) {
|
func createBooking(w http.ResponseWriter, r *http.Request) {
|
||||||
booking := (*models.Booking).FromUrlParams(nil, r.URL.Query())
|
booking := (*models.Booking).FromUrlParams(nil, r.URL.Query())
|
||||||
if booking.Verify() {
|
if booking.Verify() {
|
||||||
err := booking.Insert(DB)
|
err := booking.Insert()
|
||||||
if errors.Is(models.SameBookingError{}, err) {
|
if errors.Is(models.SameBookingError{}, err) {
|
||||||
http.Error(w, "Booking already exists", http.StatusConflict)
|
http.Error(w, "Booking already exists", http.StatusConflict)
|
||||||
return
|
return
|
||||||
@@ -61,7 +69,7 @@ func createBooking(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
func getBookings(w http.ResponseWriter, r *http.Request) {
|
func getBookings(w http.ResponseWriter, r *http.Request) {
|
||||||
card_id := r.URL.Query().Get("cardID")
|
card_id := r.URL.Query().Get("cardID")
|
||||||
bookings, err := GetBookingsByCardID(DB, card_id)
|
bookings, err := (*models.Booking).GetBookingsByCardID(nil, card_id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error getting bookings: ", err)
|
log.Println("Error getting bookings: ", err)
|
||||||
http.Error(w, "Internal Server Error", http.StatusInternalServerError)
|
http.Error(w, "Internal Server Error", http.StatusInternalServerError)
|
||||||
@@ -69,7 +77,42 @@ func getBookings(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
w.Header().Set("Content-Type", "application/json")
|
w.Header().Set("Content-Type", "application/json")
|
||||||
json.NewEncoder(w).Encode(bookings)
|
json.NewEncoder(w).Encode(bookings)
|
||||||
|
}
|
||||||
|
|
||||||
|
func updateBooking(w http.ResponseWriter, r *http.Request) {
|
||||||
|
_booking_id := r.URL.Query().Get("bookingID")
|
||||||
|
if _booking_id == "" {
|
||||||
|
http.Error(w, "Missing bookingID query parameter", http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
booking_id, err := strconv.Atoi(_booking_id)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "Invalid bookingID query parameter", http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
booking, err := (*models.Booking).GetBookingById(nil, booking_id)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("Error getting booking: ", err)
|
||||||
|
http.Error(w, "Internal Server Error", http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if card_id := r.URL.Query().Get("cardID"); card_id != "" {
|
||||||
|
booking.CardID = card_id
|
||||||
|
}
|
||||||
|
if reader_id := r.URL.Query().Get("readerID"); reader_id != "" {
|
||||||
|
booking.ReaderID = reader_id
|
||||||
|
}
|
||||||
|
if _booking_type := r.URL.Query().Get("bookingType"); _booking_type != "" {
|
||||||
|
booking_type, err := strconv.Atoi(_booking_type)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "Invalid booking_type query parameter", http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
booking.BookingType = booking_type
|
||||||
|
}
|
||||||
|
booking.Save()
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
json.NewEncoder(w).Encode(booking)
|
||||||
}
|
}
|
||||||
|
|
||||||
// func getBooking(w http.ResponseWriter, r *http.Request)
|
// func getBooking(w http.ResponseWriter, r *http.Request)
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package models
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"database/sql"
|
"database/sql"
|
||||||
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"net/url"
|
"net/url"
|
||||||
"strconv"
|
"strconv"
|
||||||
@@ -22,6 +23,8 @@ type Booking struct {
|
|||||||
Id int `json:"id"`
|
Id int `json:"id"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var DB *sql.DB
|
||||||
|
|
||||||
func (b Booking) New(card_id string, reader_id string, booking_type int) Booking {
|
func (b Booking) New(card_id string, reader_id string, booking_type int) Booking {
|
||||||
return Booking{
|
return Booking{
|
||||||
CardID: card_id,
|
CardID: card_id,
|
||||||
@@ -48,24 +51,81 @@ func (b Booking) Verify() bool {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b Booking) Insert(db *sql.DB) error {
|
func (b *Booking) Insert() error {
|
||||||
if !checkLastBooking(b, db) {
|
if !checkLastBooking(*b) {
|
||||||
return SameBookingError{}
|
return SameBookingError{}
|
||||||
}
|
}
|
||||||
stmt, err := db.Prepare((`INSERT INTO zeiten (card_id, reader_id, booking_type) VALUES ($1, $2, $3)`))
|
stmt, err := DB.Prepare((`INSERT INTO zeiten (card_id, reader_id, booking_type) VALUES ($1, $2, $3) RETURNING id, logged_time`))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
_, err = stmt.Query(b.CardID, b.ReaderID, b.BookingType)
|
err = stmt.QueryRow(b.CardID, b.ReaderID, b.BookingType).Scan(&b.Id, &b.LoggedTime)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func checkLastBooking(b Booking, db *sql.DB) bool {
|
func (b *Booking) GetBookingById(booking_id int) (Booking, error) {
|
||||||
|
var booking Booking
|
||||||
|
qStr, err := DB.Prepare((`SELECT id, logged_time, card_id, reader_id, booking_type FROM zeiten WHERE id = $1`))
|
||||||
|
if err != nil {
|
||||||
|
return booking, err
|
||||||
|
}
|
||||||
|
err = qStr.QueryRow(booking_id).Scan(&booking.Id, &booking.LoggedTime, &booking.CardID, &booking.ReaderID, &booking.BookingType)
|
||||||
|
if err != nil {
|
||||||
|
return booking, err
|
||||||
|
}
|
||||||
|
if !booking.Verify() {
|
||||||
|
fmt.Printf("Booking verification failed")
|
||||||
|
return booking, nil
|
||||||
|
}
|
||||||
|
return booking, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Booking) GetBookingsByCardID(card_id string) ([]Booking, error) {
|
||||||
|
qStr, err := DB.Prepare((`SELECT id, logged_time, card_id, reader_id, booking_type FROM zeiten WHERE card_id = $1`))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var bookings []Booking
|
||||||
|
rows, err := qStr.Query(card_id)
|
||||||
|
if err == sql.ErrNoRows {
|
||||||
|
return bookings, err
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer rows.Close()
|
||||||
|
for rows.Next() {
|
||||||
|
var booking Booking
|
||||||
|
if err := rows.Scan(&booking.Id, &booking.LoggedTime, &booking.CardID, &booking.ReaderID, &booking.BookingType); err != nil {
|
||||||
|
return bookings, err
|
||||||
|
}
|
||||||
|
bookings = append(bookings, booking)
|
||||||
|
}
|
||||||
|
if err = rows.Err(); err != nil {
|
||||||
|
return bookings, err
|
||||||
|
}
|
||||||
|
return bookings, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b Booking) Save() {
|
||||||
|
qStr, err := DB.Prepare((`UPDATE "zeiten" SET "id" = $1, "card_id" = $2, "reader_id" = $3, "booking_type" = $4 WHERE "id" = $1;`))
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Error preparing query: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
_, err = qStr.Query(b.Id, b.CardID, b.ReaderID, b.BookingType)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Error executing query: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func checkLastBooking(b Booking) bool {
|
||||||
var booking_type int
|
var booking_type int
|
||||||
stmt, err := db.Prepare((`SELECT booking_type FROM "zeiten" WHERE "card_id" = $1 ORDER BY "logged_time" DESC LIMIT 1;`))
|
stmt, err := DB.Prepare((`SELECT booking_type FROM "zeiten" WHERE "card_id" = $1 ORDER BY "logged_time" DESC LIMIT 1;`))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Error preparing query: %v", err)
|
log.Fatalf("Error preparing query: %v", err)
|
||||||
return false
|
return false
|
||||||
|
|||||||
@@ -2,4 +2,4 @@ POSTGRES_USER=arbeit_zeit
|
|||||||
POSTGRES_PASSWORD=password
|
POSTGRES_PASSWORD=password
|
||||||
POSTGRES_PATH=./database
|
POSTGRES_PATH=./database
|
||||||
POSTGRES_DB=arbeitszeitmessung
|
POSTGRES_DB=arbeitszeitmessung
|
||||||
# EXPOSED_PORT=8080
|
EXPOSED_PORT=8000
|
||||||
|
|||||||
35
Docker/dev-docker-compose.yml
Normal file
35
Docker/dev-docker-compose.yml
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
name: arbeitszeitmessung-dev
|
||||||
|
services:
|
||||||
|
db:
|
||||||
|
image: postgres:16
|
||||||
|
restart: unless-stopped
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
environment:
|
||||||
|
POSTGRES_USER: ${POSTGRES_USER}
|
||||||
|
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
|
||||||
|
POSTGRES_DB: ${POSTGRES_DB}
|
||||||
|
PGDATA: /var/lib/postgresql/data/pg_data
|
||||||
|
volumes:
|
||||||
|
- ${POSTGRES_PATH}:/var/lib/postgresql/data
|
||||||
|
ports:
|
||||||
|
- 5432:5432
|
||||||
|
|
||||||
|
adminer:
|
||||||
|
image: adminer
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- 8001:8080
|
||||||
|
backend:
|
||||||
|
build: ../Backend
|
||||||
|
image: git.letsstein.de/tom/arbeitszeit-backend
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
environment:
|
||||||
|
POSTGRES_HOST: db
|
||||||
|
POSTGRES_DB: ${POSTGRES_DB}
|
||||||
|
POSTGRES_USER: ${POSTGRES_USER}
|
||||||
|
POSTGRES_PASS: ${POSTGRES_PASSWORD}
|
||||||
|
EXPOSED_PORT: ${EXPOSED_PORT}
|
||||||
|
ports:
|
||||||
|
- 8000:8080
|
||||||
@@ -12,16 +12,14 @@ services:
|
|||||||
PGDATA: /var/lib/postgresql/data/pg_data
|
PGDATA: /var/lib/postgresql/data/pg_data
|
||||||
volumes:
|
volumes:
|
||||||
- ${POSTGRES_PATH}:/var/lib/postgresql/data
|
- ${POSTGRES_PATH}:/var/lib/postgresql/data
|
||||||
ports:
|
|
||||||
- 5432:5432
|
|
||||||
|
|
||||||
adminer:
|
adminer:
|
||||||
image: adminer
|
image: adminer
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
ports:
|
ports:
|
||||||
- 8000:8080
|
- 8000:8080
|
||||||
|
|
||||||
backend:
|
backend:
|
||||||
build: ../Backend
|
|
||||||
image: git.letsstein.de/tom/arbeitszeit-backend
|
image: git.letsstein.de/tom/arbeitszeit-backend
|
||||||
env_file:
|
env_file:
|
||||||
- .env
|
- .env
|
||||||
@@ -30,5 +28,8 @@ services:
|
|||||||
POSTGRES_DB: ${POSTGRES_DB}
|
POSTGRES_DB: ${POSTGRES_DB}
|
||||||
POSTGRES_USER: ${POSTGRES_USER}
|
POSTGRES_USER: ${POSTGRES_USER}
|
||||||
POSTGRES_PASS: ${POSTGRES_PASSWORD}
|
POSTGRES_PASS: ${POSTGRES_PASSWORD}
|
||||||
|
EXPOSED_PORT: ${EXPOSED_PORT}
|
||||||
ports:
|
ports:
|
||||||
- 8001:8080
|
- ${EXPOSED_PORT}:8080
|
||||||
|
depends_on:
|
||||||
|
- db
|
||||||
|
|||||||
80
Readme.md
80
Readme.md
@@ -5,9 +5,81 @@ bis jetzt ein einfaches Backend mit PostgreSQL Datenbank und GO Webserver um Arb
|
|||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone https://git.letsstein.de/tom/arbeitszeitmessung
|
git clone https://git.letsstein.de/tom/arbeitszeitmessung arbeitszeitmessung
|
||||||
|
|
||||||
# cd in das Verzeichnis
|
cd arbeitszeitmessung/Docker
|
||||||
|
# .env Datei anpassen
|
||||||
cd Docker
|
|
||||||
docker compose up -d
|
docker compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
## API
|
||||||
|
Nutzung der API
|
||||||
|
|
||||||
|
### Buchungen [/time]
|
||||||
|
|
||||||
|
#### [GET] Anfrage
|
||||||
|
Parameter: cardID (string)
|
||||||
|
Antwort: `200`
|
||||||
|
```json
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"cradID": "test_card",
|
||||||
|
"readerID": "test_reader",
|
||||||
|
"bookingTyp": 2,
|
||||||
|
"loggedTime": "2024-09-05T08:37:53.117641Z",
|
||||||
|
"id": 5
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"cradID": "test_card",
|
||||||
|
"readerID": "mytest",
|
||||||
|
"bookingTyp": 1,
|
||||||
|
"loggedTime": "2024-09-05T08:51:12.670827Z",
|
||||||
|
"id": 6
|
||||||
|
},
|
||||||
|
]
|
||||||
|
```
|
||||||
|
Antwort `500`
|
||||||
|
Serverfehler
|
||||||
|
|
||||||
|
#### [PUT] Anfrage
|
||||||
|
Parameter: id (int)
|
||||||
|
Body: (veränderte Parameter)
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"cradID": "test_card",
|
||||||
|
"readerID": "mytest",
|
||||||
|
"bookingTyp": 1,
|
||||||
|
"loggedTime": "2024-09-05T08:51:12.670827Z",
|
||||||
|
}
|
||||||
|
```
|
||||||
|
Antwort `200`
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"cradID": "test_card",
|
||||||
|
"readerID": "mytest",
|
||||||
|
"bookingTyp": 1,
|
||||||
|
"loggedTime": "2024-09-05T08:51:12.670827Z",
|
||||||
|
"id": 6
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Neue Buchung [/time/new]
|
||||||
|
#### [PUT] Anfrage
|
||||||
|
Parameter:
|
||||||
|
- cardID (string)
|
||||||
|
- readerID (string)
|
||||||
|
- bookingType (string)
|
||||||
|
|
||||||
|
Antwort `202` Akzeptiert und eingefügt
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"cradID": "test_card",
|
||||||
|
"readerID": "mytest",
|
||||||
|
"bookingTyp": 1,
|
||||||
|
"loggedTime": "2024-09-05T08:51:12.670827Z",
|
||||||
|
"id": 6
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Antwort `409` Konflikt
|
||||||
|
Die vorherige Buchung am selben Tag hat den gleichen Buchungstyp
|
||||||
|
|||||||
3
db.sql
3
db.sql
@@ -8,7 +8,8 @@ CREATE TABLE zeiten (
|
|||||||
);
|
);
|
||||||
-- @block insert data
|
-- @block insert data
|
||||||
INSERT INTO zeiten (card_id, reader_id, booking_type)
|
INSERT INTO zeiten (card_id, reader_id, booking_type)
|
||||||
VALUES ('test_card', 'test_reader', '2');
|
VALUES ('test_card', 'test_reader', '2')
|
||||||
|
RETURNING id, logged_time;
|
||||||
-- @block select
|
-- @block select
|
||||||
SELECT *
|
SELECT *
|
||||||
FROM zeiten;
|
FROM zeiten;
|
||||||
|
|||||||
Reference in New Issue
Block a user