Compare commits
25 Commits
dev/ui
...
dev/loggin
| Author | SHA1 | Date | |
|---|---|---|---|
| 7eda8eb538 | |||
| 0d7696cbc6 | |||
| 5001f24d9b | |||
| ea8e78fd9f | |||
| 6da58d6753 | |||
| 89eb5d255d | |||
| 1b8fb747e8 | |||
| 74cded42d8 | |||
| 22350142fc | |||
| 659fb80049 | |||
| cbc4028f8d | |||
| e4d423385a | |||
| c9c2d801b0 | |||
| 94c7c8a36e | |||
| d69ec600cd | |||
| 95d5c4ab9d | |||
| bf841ad5c6 | |||
| a1aae9dc56 | |||
| 750fb1ff58 | |||
| f4e9915e7f | |||
| 18046bbe18 | |||
| 75929e3b7d | |||
| 9e5dc760d5 | |||
| 4d00143a74 | |||
| 5fbe53faf6 |
@@ -23,7 +23,9 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
# Disabling shallow clone is recommended for improving relevancy of reporting
|
||||||
|
fetch-depth: 0
|
||||||
- name: Setup go
|
- name: Setup go
|
||||||
uses: actions/setup-go@v5
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
@@ -45,4 +47,25 @@ jobs:
|
|||||||
restore-keys: |-
|
restore-keys: |-
|
||||||
arbeitszeitmessung-
|
arbeitszeitmessung-
|
||||||
- name: Run Go Tests
|
- name: Run Go Tests
|
||||||
run: cd Backend && go test ./...
|
run: cd Backend && mkdir .test && go test ./... -coverprofile=.test/coverage.out -json > .test/report.json
|
||||||
|
- name: Verify coverage report exists
|
||||||
|
run: |
|
||||||
|
if [ -f "Backend/.test/coverage.out" ]; then
|
||||||
|
echo "Coverage report found"
|
||||||
|
else
|
||||||
|
echo "Coverage report not found"
|
||||||
|
fi
|
||||||
|
- uses: SonarSource/sonarqube-scan-action@v6
|
||||||
|
env:
|
||||||
|
SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }}
|
||||||
|
SONAR_HOST_URL: ${{ secrets.SONAR_HOST_URL }}
|
||||||
|
with:
|
||||||
|
projectBaseDir: Backend
|
||||||
|
args: >
|
||||||
|
-Dsonar.projectVersion=${{ gitea.sha_short }}
|
||||||
|
- uses: SonarSource/sonarqube-quality-gate-action@v1
|
||||||
|
timeout-minutes: 5
|
||||||
|
env:
|
||||||
|
SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }}
|
||||||
|
with:
|
||||||
|
scanMetadataReportFile: Backend/.scannerwork/report-task.txt
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ RUN go mod download && go mod verify
|
|||||||
COPY . .
|
COPY . .
|
||||||
RUN go build -o server .
|
RUN go build -o server .
|
||||||
|
|
||||||
FROM alpine
|
FROM alpine:3.22
|
||||||
RUN apk add --no-cache tzdata
|
RUN apk add --no-cache tzdata
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY --from=build /app/server /app/server
|
COPY --from=build /app/server /app/server
|
||||||
|
|||||||
@@ -9,9 +9,62 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type typstMetadata struct {
|
||||||
|
ISOWeek string `json:"iso-week"`
|
||||||
|
EmployeeName string `json:"employee-name"`
|
||||||
|
WorkTime string `json:"worktime"`
|
||||||
|
Overtime string `json:"overtime"`
|
||||||
|
OvertimeTotal string `json:"overtime-total"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type typstDayPart struct {
|
||||||
|
BookingFrom string `json:"booking-from"`
|
||||||
|
BookingTo string `json:"booking-to"`
|
||||||
|
WorkType string `json:"worktype"`
|
||||||
|
IsWorkDay bool `json:"is-workday"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type typstDay struct {
|
||||||
|
Date string `json:"date"`
|
||||||
|
DayParts []typstDayPart `json:"day-parts"`
|
||||||
|
Worktime string `json:"worktime"`
|
||||||
|
Pausetime string `json:"pausetime"`
|
||||||
|
Overtime string `json:"overtime"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func ConvertDaysToTypst(days []models.IWorkDay, u models.User) ([]typstDay, error) {
|
||||||
|
var typstDays []typstDay
|
||||||
|
for _, day := range days {
|
||||||
|
var typstDay typstDay
|
||||||
|
var typstDayParts []typstDayPart
|
||||||
|
work, pause, overtime := day.GetAllWorkTimesVirtual(u)
|
||||||
|
typstDay.Date = day.Date().Format("01.02.2006")
|
||||||
|
typstDay.Worktime = helper.FormatDuration(work)
|
||||||
|
typstDay.Pausetime = helper.FormatDuration(pause)
|
||||||
|
typstDay.Overtime = helper.FormatDuration(overtime)
|
||||||
|
if day.IsWorkDay() {
|
||||||
|
workDay, _ := day.(*models.WorkDay)
|
||||||
|
for i := 0; i < len(workDay.Bookings); i += 2 {
|
||||||
|
var typstDayPart typstDayPart
|
||||||
|
typstDayPart.BookingFrom = workDay.Bookings[i].Timestamp.Format("15:04")
|
||||||
|
typstDayPart.BookingTo = workDay.Bookings[i+1].Timestamp.Format("15:04")
|
||||||
|
typstDayPart.WorkType = workDay.Bookings[i].BookingType.Name
|
||||||
|
typstDayPart.IsWorkDay = true
|
||||||
|
typstDayParts = append(typstDayParts, typstDayPart)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
absentDay, _ := day.(*models.Absence)
|
||||||
|
typstDayParts = append(typstDayParts, typstDayPart{IsWorkDay: false, WorkType: absentDay.AbwesenheitTyp.Name})
|
||||||
|
}
|
||||||
|
typstDay.DayParts = typstDayParts
|
||||||
|
typstDays = append(typstDays, typstDay)
|
||||||
|
}
|
||||||
|
return typstDays, nil
|
||||||
|
}
|
||||||
|
|
||||||
func PDFHandler(w http.ResponseWriter, r *http.Request) {
|
func PDFHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
helper.RequiresLogin(Session, w, r)
|
helper.RequiresLogin(Session, w, r)
|
||||||
startDate, err := parseTimestamp(r, "start", time.Now().Format("2006-01-02"))
|
startDate, err := parseTimestamp(r, "start_date", time.Now().Format("2006-01-02"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error parsing 'start_date' time", err)
|
log.Println("Error parsing 'start_date' time", err)
|
||||||
http.Error(w, "Timestamp 'start_date' cannot be parsed!", http.StatusBadRequest)
|
http.Error(w, "Timestamp 'start_date' cannot be parsed!", http.StatusBadRequest)
|
||||||
|
|||||||
@@ -14,8 +14,11 @@ require (
|
|||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
github.com/Dadido3/go-typst v0.3.0 // indirect
|
||||||
github.com/hashicorp/errwrap v1.1.0 // indirect
|
github.com/hashicorp/errwrap v1.1.0 // indirect
|
||||||
github.com/hashicorp/go-multierror v1.1.1 // indirect
|
github.com/hashicorp/go-multierror v1.1.1 // indirect
|
||||||
|
github.com/smasher164/xid v0.1.2 // indirect
|
||||||
go.uber.org/atomic v1.7.0 // indirect
|
go.uber.org/atomic v1.7.0 // indirect
|
||||||
golang.org/x/sys v0.36.0 // indirect
|
golang.org/x/sys v0.36.0 // indirect
|
||||||
|
golang.org/x/text v0.23.0 // indirect
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 h1:L/gRVlceqvL25UVaW/CKtUDjefjrs0SPonmDGUVOYP0=
|
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 h1:L/gRVlceqvL25UVaW/CKtUDjefjrs0SPonmDGUVOYP0=
|
||||||
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E=
|
github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E=
|
||||||
|
github.com/Dadido3/go-typst v0.3.0 h1:Itix2FtQgBiOuHUNqgGUAK11Oo2WMlZGGGpCiQNK1IA=
|
||||||
|
github.com/Dadido3/go-typst v0.3.0/go.mod h1:QYis9sT70u65kn1SkFfyPRmHsPxgoxWbAixwfPReOZA=
|
||||||
github.com/Microsoft/go-winio v0.6.2 h1:F2VQgta7ecxGYO8k3ZZz3RS8fVIXVxONVUPlNERoyfY=
|
github.com/Microsoft/go-winio v0.6.2 h1:F2VQgta7ecxGYO8k3ZZz3RS8fVIXVxONVUPlNERoyfY=
|
||||||
github.com/Microsoft/go-winio v0.6.2/go.mod h1:yd8OoFMLzJbo9gZq8j5qaps8bJ9aShtEA8Ipt1oGCvU=
|
github.com/Microsoft/go-winio v0.6.2/go.mod h1:yd8OoFMLzJbo9gZq8j5qaps8bJ9aShtEA8Ipt1oGCvU=
|
||||||
github.com/a-h/templ v0.3.943 h1:o+mT/4yqhZ33F3ootBiHwaY4HM5EVaOJfIshvd5UNTY=
|
github.com/a-h/templ v0.3.943 h1:o+mT/4yqhZ33F3ootBiHwaY4HM5EVaOJfIshvd5UNTY=
|
||||||
@@ -54,6 +56,8 @@ github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
|||||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
|
github.com/smasher164/xid v0.1.2 h1:erplXSdBRIIw+MrwjJ/m8sLN2XY16UGzpTA0E2Ru6HA=
|
||||||
|
github.com/smasher164/xid v0.1.2/go.mod h1:tgivm8CQl19fH1c5y+8F4mA+qY6n2i6qDRBlY/6nm+I=
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||||
github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
|
github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
|
||||||
@@ -70,5 +74,9 @@ go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw=
|
|||||||
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
||||||
golang.org/x/sys v0.36.0 h1:KVRy2GtZBrk1cBYA7MKu5bEZFxQk4NIDV6RLVcC8o0k=
|
golang.org/x/sys v0.36.0 h1:KVRy2GtZBrk1cBYA7MKu5bEZFxQk4NIDV6RLVcC8o0k=
|
||||||
golang.org/x/sys v0.36.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
|
golang.org/x/sys v0.36.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
|
||||||
|
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
|
golang.org/x/text v0.23.0 h1:D71I7dUrlY+VX0gQShAThNGHFxZ13dGLBHQLVl1mJlY=
|
||||||
|
golang.org/x/text v0.23.0/go.mod h1:/BLNzu4aZCJ1+kcD0DNRotWKage4q2rGVAg4o22unh4=
|
||||||
|
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
|||||||
@@ -5,8 +5,7 @@ import (
|
|||||||
"arbeitszeitmessung/helper"
|
"arbeitszeitmessung/helper"
|
||||||
"arbeitszeitmessung/models"
|
"arbeitszeitmessung/models"
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"log/slog"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"time"
|
"time"
|
||||||
@@ -17,23 +16,25 @@ import (
|
|||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
var err error
|
var err error
|
||||||
|
var logLevel slog.LevelVar
|
||||||
|
logLevel.Set(slog.LevelWarn)
|
||||||
|
|
||||||
|
logger := slog.New(slog.NewJSONHandler(os.Stdout, &slog.HandlerOptions{Level: &logLevel}))
|
||||||
|
slog.SetDefault(logger)
|
||||||
|
|
||||||
err = godotenv.Load(".env")
|
err = godotenv.Load(".env")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("No .env file found in directory!")
|
slog.Info("No .env file found in directory!")
|
||||||
}
|
}
|
||||||
if helper.GetEnv("GO_ENV", "production") == "debug" {
|
if helper.GetEnv("GO_ENV", "production") == "debug" {
|
||||||
log.Println("Debug mode enabled")
|
logLevel.Set(slog.LevelDebug)
|
||||||
log.Println("Environment Variables")
|
|
||||||
envs := os.Environ()
|
envs := os.Environ()
|
||||||
for _, e := range envs {
|
slog.Debug("Debug mode enabled", "Environment Variables", envs)
|
||||||
fmt.Println(e)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
models.DB, err = OpenDatabase()
|
models.DB, err = OpenDatabase()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
slog.Error("Error while opening the database", "Error", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
fs := http.FileServer(http.Dir("./static"))
|
fs := http.FileServer(http.Dir("./static"))
|
||||||
@@ -58,14 +59,15 @@ func main() {
|
|||||||
serverSessionMiddleware := endpoints.Session.LoadAndSave(server)
|
serverSessionMiddleware := endpoints.Session.LoadAndSave(server)
|
||||||
|
|
||||||
// starting the http server
|
// starting the http server
|
||||||
fmt.Printf("Server is running at http://localhost:%s\n", helper.GetEnv("EXPOSED_PORT", "8080"))
|
slog.Info("Server is running at http://localhost:8080")
|
||||||
log.Fatal(http.ListenAndServe(":8080", serverSessionMiddleware))
|
slog.Error("Error starting Server", "Error", http.ListenAndServe(":8080", serverSessionMiddleware))
|
||||||
}
|
}
|
||||||
|
|
||||||
func ParamsMiddleware(next http.HandlerFunc) http.Handler {
|
func ParamsMiddleware(next http.HandlerFunc) http.Handler {
|
||||||
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
queryParams := r.URL.Query()
|
queryParams := r.URL.Query()
|
||||||
ctx := context.WithValue(r.Context(), "urlParams", queryParams)
|
ctx := context.WithValue(r.Context(), "urlParams", queryParams)
|
||||||
|
slog.Debug("ParamsMiddleware added urlParams", slog.Any("urlParams", queryParams))
|
||||||
next.ServeHTTP(w, r.WithContext(ctx))
|
next.ServeHTTP(w, r.WithContext(ctx))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import (
|
|||||||
"database/sql"
|
"database/sql"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
|
"log/slog"
|
||||||
"net/url"
|
"net/url"
|
||||||
"strconv"
|
"strconv"
|
||||||
"time"
|
"time"
|
||||||
@@ -87,6 +88,27 @@ func (b *Booking) Verify() bool {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *Booking) IsSubmittedAndChecked() bool {
|
||||||
|
qStr, err := DB.Prepare(`SELECT bestaetigt from wochen_report WHERE $1 = ANY(anwesenheiten);`)
|
||||||
|
if err != nil {
|
||||||
|
slog.Warn("Error when preparing SQL Statement", "error", err)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
defer qStr.Close()
|
||||||
|
var isSubmittedAndChecked bool = false
|
||||||
|
|
||||||
|
err = qStr.QueryRow(b.CounterId).Scan(&isSubmittedAndChecked)
|
||||||
|
if err == sql.ErrNoRows {
|
||||||
|
// No rows found ==> not even submitted
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
slog.Warn("Unexpected error when executing SQL Statement", "error", err)
|
||||||
|
}
|
||||||
|
return isSubmittedAndChecked
|
||||||
|
}
|
||||||
|
|
||||||
func (b *Booking) Insert() error {
|
func (b *Booking) Insert() error {
|
||||||
if !checkLastBooking(*b) {
|
if !checkLastBooking(*b) {
|
||||||
return SameBookingError{}
|
return SameBookingError{}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
|
"log/slog"
|
||||||
"sort"
|
"sort"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
@@ -53,9 +54,11 @@ func GetDays(user User, tsFrom, tsTo time.Time, orderedForward bool) []IWorkDay
|
|||||||
}
|
}
|
||||||
if day.AbwesenheitTyp.WorkTime == 1 {
|
if day.AbwesenheitTyp.WorkTime == 1 {
|
||||||
if workDay, ok := allDays[day.Date().Format("2006-01-02")].(*WorkDay); ok {
|
if workDay, ok := allDays[day.Date().Format("2006-01-02")].(*WorkDay); ok {
|
||||||
|
if len(workDay.Bookings) > 0 {
|
||||||
workDay.kurzArbeit = true
|
workDay.kurzArbeit = true
|
||||||
workDay.kurzArbeitAbsence = day
|
workDay.kurzArbeitAbsence = day
|
||||||
}
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
allDays[day.Date().Format("2006-01-02")] = &day
|
allDays[day.Date().Format("2006-01-02")] = &day
|
||||||
}
|
}
|
||||||
@@ -80,6 +83,19 @@ func (d *WorkDay) Date() time.Time {
|
|||||||
return d.Day
|
return d.Day
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (d *WorkDay) GenerateKurzArbeitBookings(u User) (time.Time, time.Time) {
|
||||||
|
var timeFrom, timeTo time.Time
|
||||||
|
if d.workTime >= u.ArbeitszeitProTag() {
|
||||||
|
return timeFrom, timeTo
|
||||||
|
}
|
||||||
|
|
||||||
|
timeFrom = d.Bookings[len(d.Bookings)-1].Timestamp.Add(time.Minute)
|
||||||
|
timeTo = timeFrom.Add(u.ArbeitszeitProTag() - d.workTime)
|
||||||
|
slog.Debug("Added duration as Kurzarbeit", "date", d.Date().String(), "duration", timeTo.Sub(timeFrom).String())
|
||||||
|
|
||||||
|
return timeFrom, timeTo
|
||||||
|
}
|
||||||
|
|
||||||
func (d *WorkDay) TimeWorkVirtual(u User) time.Duration {
|
func (d *WorkDay) TimeWorkVirtual(u User) time.Duration {
|
||||||
if d.IsKurzArbeit() {
|
if d.IsKurzArbeit() {
|
||||||
return u.ArbeitszeitProTag()
|
return u.ArbeitszeitProTag()
|
||||||
@@ -107,6 +123,7 @@ func (d *WorkDay) TimeWorkReal(u User) time.Duration {
|
|||||||
if helper.IsSameDate(d.Date(), time.Now()) && len(d.Bookings)%2 == 1 {
|
if helper.IsSameDate(d.Date(), time.Now()) && len(d.Bookings)%2 == 1 {
|
||||||
d.realWorkTime += time.Since(lastBooking.Timestamp.Local())
|
d.realWorkTime += time.Since(lastBooking.Timestamp.Local())
|
||||||
}
|
}
|
||||||
|
slog.Debug("Calculated RealWorkTime for user", "user", u, slog.String("worktime", d.realWorkTime.String()))
|
||||||
return d.realWorkTime
|
return d.realWorkTime
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,10 @@ import (
|
|||||||
"database/sql"
|
"database/sql"
|
||||||
"errors"
|
"errors"
|
||||||
"log"
|
"log"
|
||||||
|
"log/slog"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/lib/pq"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Workweeks are
|
// Workweeks are
|
||||||
@@ -17,10 +20,9 @@ type WorkWeek struct {
|
|||||||
User User
|
User User
|
||||||
WeekStart time.Time
|
WeekStart time.Time
|
||||||
Worktime time.Duration
|
Worktime time.Duration
|
||||||
|
WorkTimeVirtual time.Duration
|
||||||
Overtime time.Duration
|
Overtime time.Duration
|
||||||
Status WeekStatus
|
Status WeekStatus
|
||||||
overtimeDiff time.Duration
|
|
||||||
worktimeDiff time.Duration
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type WeekStatus int8
|
type WeekStatus int8
|
||||||
@@ -45,12 +47,17 @@ func NewWorkWeek(user User, tsMonday time.Time, populate bool) WorkWeek {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (w *WorkWeek) PopulateWithDays(worktime time.Duration, overtime time.Duration) {
|
func (w *WorkWeek) PopulateWithDays(worktime time.Duration, overtime time.Duration) {
|
||||||
|
slog.Debug("Got Days with overtime and worktime", slog.String("worktime", worktime.String()), slog.String("overtime", overtime.String()))
|
||||||
w.Days = GetDays(w.User, w.WeekStart, w.WeekStart.Add(6*24*time.Hour), false)
|
w.Days = GetDays(w.User, w.WeekStart, w.WeekStart.Add(6*24*time.Hour), false)
|
||||||
|
|
||||||
for _, day := range w.Days {
|
for _, day := range w.Days {
|
||||||
w.Worktime += day.TimeWorkVirtual(w.User)
|
work, _ := day.TimePauseReal(w.User)
|
||||||
|
w.Worktime += work
|
||||||
|
w.WorkTimeVirtual += day.TimeWorkVirtual(w.User)
|
||||||
}
|
}
|
||||||
w.Overtime = w.Worktime - w.User.ArbeitszeitProWoche()
|
slog.Debug("Got worktime for user", "user", w.User, "worktime", w.Worktime.String(), "virtualWorkTime", w.WorkTimeVirtual.String())
|
||||||
|
|
||||||
|
w.Overtime = w.WorkTimeVirtual - w.User.ArbeitszeitProWoche()
|
||||||
|
|
||||||
w.Worktime = w.Worktime.Round(time.Minute)
|
w.Worktime = w.Worktime.Round(time.Minute)
|
||||||
w.Overtime = w.Overtime.Round(time.Minute)
|
w.Overtime = w.Overtime.Round(time.Minute)
|
||||||
@@ -61,8 +68,6 @@ func (w *WorkWeek) PopulateWithDays(worktime time.Duration, overtime time.Durati
|
|||||||
|
|
||||||
if overtime != w.Overtime || worktime != w.Worktime {
|
if overtime != w.Overtime || worktime != w.Worktime {
|
||||||
w.Status = WeekStatusDifferences
|
w.Status = WeekStatusDifferences
|
||||||
w.overtimeDiff = overtime
|
|
||||||
w.worktimeDiff = worktime
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -127,12 +132,13 @@ func (w *WorkWeek) GetSendWeeks(user User) []WorkWeek {
|
|||||||
defer rows.Close()
|
defer rows.Close()
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
var week WorkWeek = WorkWeek{User: user}
|
var week WorkWeek = WorkWeek{User: user}
|
||||||
if err := rows.Scan(&week.Id, &week.WeekStart, &week.Worktime, &week.Overtime); err != nil {
|
var workTime, overTime time.Duration
|
||||||
|
if err := rows.Scan(&week.Id, &week.WeekStart, &workTime, &overTime); err != nil {
|
||||||
log.Println("Error scanning row!", err)
|
log.Println("Error scanning row!", err)
|
||||||
return weeks
|
return weeks
|
||||||
}
|
}
|
||||||
|
|
||||||
week.PopulateWithDays(week.Worktime, week.Overtime)
|
week.PopulateWithDays(workTime, overTime)
|
||||||
weeks = append(weeks, week)
|
weeks = append(weeks, week)
|
||||||
}
|
}
|
||||||
if err = rows.Err(); err != nil {
|
if err = rows.Err(); err != nil {
|
||||||
@@ -144,30 +150,70 @@ func (w *WorkWeek) GetSendWeeks(user User) []WorkWeek {
|
|||||||
|
|
||||||
var ErrRunningWeek = errors.New("Week is in running week")
|
var ErrRunningWeek = errors.New("Week is in running week")
|
||||||
|
|
||||||
|
func (w *WorkWeek) GetBookingIds() (anwesenheitsIds, abwesenheitsIds []int64, err error) {
|
||||||
|
qStr, err := DB.Prepare(`
|
||||||
|
SELECT
|
||||||
|
(SELECT array_agg(counter_id ORDER BY counter_id)
|
||||||
|
FROM anwesenheit
|
||||||
|
WHERE card_uid = $1
|
||||||
|
AND timestamp::DATE >= $2
|
||||||
|
AND timestamp::DATE < $3) AS anwesenheit,
|
||||||
|
|
||||||
|
(SELECT array_agg(counter_id ORDER BY counter_id)
|
||||||
|
FROM abwesenheit
|
||||||
|
WHERE card_uid = $1
|
||||||
|
AND datum_from < $3
|
||||||
|
AND datum_to >= $2) AS abwesenheit;
|
||||||
|
`)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
defer qStr.Close()
|
||||||
|
|
||||||
|
slog.Debug("Inserting parameters into qStr:", "user card_uid", w.User.CardUID, "week_start", w.WeekStart, "week_end", w.WeekStart.AddDate(0, 0, 5))
|
||||||
|
|
||||||
|
err = qStr.QueryRow(w.User.CardUID, w.WeekStart, w.WeekStart.AddDate(0, 0, 5)).Scan(pq.Array(&anwesenheitsIds), pq.Array(&abwesenheitsIds))
|
||||||
|
if err != nil {
|
||||||
|
return anwesenheitsIds, abwesenheitsIds, err
|
||||||
|
}
|
||||||
|
return anwesenheitsIds, abwesenheitsIds, nil
|
||||||
|
}
|
||||||
|
|
||||||
// creates a new entry in the woche_report table with the given workweek
|
// creates a new entry in the woche_report table with the given workweek
|
||||||
func (w *WorkWeek) SendWeek() error {
|
func (w *WorkWeek) SendWeek() error {
|
||||||
var qStr *sql.Stmt
|
var qStr *sql.Stmt
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
|
slog.Info("Sending workWeek to team head", "week", w.WeekStart.String())
|
||||||
|
|
||||||
|
anwBookings, awBookings, err := w.GetBookingIds()
|
||||||
|
if err != nil {
|
||||||
|
slog.Warn("Error querying bookings from work week", slog.Any("error", err))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
slog.Debug("Recieved Booking Ids", "anwesenheiten", anwBookings)
|
||||||
|
|
||||||
if time.Since(w.WeekStart) < 5*24*time.Hour {
|
if time.Since(w.WeekStart) < 5*24*time.Hour {
|
||||||
log.Println("Cannot send week, because it's the running week!")
|
slog.Warn("Cannot send week, because it's the running week!")
|
||||||
return ErrRunningWeek
|
return ErrRunningWeek
|
||||||
}
|
}
|
||||||
|
|
||||||
if w.CheckStatus() != WeekStatusNone {
|
if w.CheckStatus() != WeekStatusNone {
|
||||||
qStr, err = DB.Prepare(`UPDATE "wochen_report" SET bestaetigt = FALSE, arbeitszeit = make_interval(secs => $3::numeric / 1000000000), ueberstunden = make_interval(secs => $4::numeric / 1000000000) WHERE personal_nummer = $1 AND woche_start = $2;`)
|
qStr, err = DB.Prepare(`UPDATE "wochen_report" SET bestaetigt = FALSE, arbeitszeit = make_interval(secs => $3::numeric / 1000000000), ueberstunden = make_interval(secs => $4::numeric / 1000000000), anwesenheiten=$5, abwesenheiten=$6 WHERE personal_nummer = $1 AND woche_start = $2;`)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error preparing SQL statement", err)
|
slog.Warn("Error preparing SQL statement", "error", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
qStr, err = DB.Prepare(`INSERT INTO wochen_report (personal_nummer, woche_start, arbeitszeit, ueberstunden) VALUES ($1, $2, make_interval(secs => $3::numeric / 1000000000), make_interval(secs => $4::numeric / 1000000000));`)
|
qStr, err = DB.Prepare(`INSERT INTO wochen_report (personal_nummer, woche_start, arbeitszeit, ueberstunden, anwesenheiten, abwesenheiten) VALUES ($1, $2, make_interval(secs => $3::numeric / 1000000000), make_interval(secs => $4::numeric / 1000000000), $5, $6);`)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error preparing SQL statement", err)
|
slog.Warn("Error preparing SQL statement", "error", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_, err = qStr.Exec(w.User.PersonalNummer, w.WeekStart, int64(w.Worktime), int64(w.Overtime))
|
|
||||||
|
_, err = qStr.Exec(w.User.PersonalNummer, w.WeekStart, int64(w.Worktime), int64(w.Overtime), pq.Array(anwBookings), pq.Array(awBookings))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error executing query!", err)
|
log.Println("Error executing query!", err)
|
||||||
return err
|
return err
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
sonar.projectKey=Arbeitszeitmessung
|
sonar.projectKey=arbeitszeitmessung
|
||||||
sonar.sources=.
|
sonar.sources=.
|
||||||
sonar.exclusions=**/*_test.go
|
sonar.exclusions=**/*_test.go, **/*_templ.go
|
||||||
|
|
||||||
sonar.tests=.
|
sonar.tests=.
|
||||||
sonar.test.inclusions=**/*_test.go
|
sonar.test.inclusions=**/*_test.go
|
||||||
|
|||||||
@@ -1,18 +1,21 @@
|
|||||||
function clearEditState() {
|
function clearEditState() {
|
||||||
for (e of document.querySelectorAll(".edit")) {
|
for (let e of document.querySelectorAll(".edit")) {
|
||||||
e.classList.remove("edit");
|
e.classList.remove("edit");
|
||||||
}
|
}
|
||||||
toggleAbsenceEdit(false);
|
toggleAbsenceEdit(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function clearButtonState() {
|
||||||
|
for (let b of document.querySelectorAll(".change-button-component")) {
|
||||||
|
b.type = "button";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function editWorkday(element, event, id, isWorkDay) {
|
function editWorkday(element, event, id, isWorkDay) {
|
||||||
console.log("editWorkday called", isWorkDay);
|
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
var form = document.getElementById(id);
|
let form = document.getElementById(id);
|
||||||
if (form == null) {
|
if (form == null) {
|
||||||
form = element
|
form = element.closest(".grid-sub").querySelector(".all-booking-component > form");
|
||||||
.closest(".grid-sub")
|
|
||||||
.querySelector(".all-booking-component > form");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
clearEditState();
|
clearEditState();
|
||||||
@@ -21,38 +24,29 @@ function editWorkday(element, event, id, isWorkDay) {
|
|||||||
|
|
||||||
if (isWorkDay) {
|
if (isWorkDay) {
|
||||||
element.classList.add("edit");
|
element.classList.add("edit");
|
||||||
if (element.classList.contains("edit")) {
|
if (element.type == "button") {
|
||||||
form.querySelectorAll("input, select").forEach((input) => {
|
for (let input of form.querySelectorAll("input, select")) {
|
||||||
input.disabled = false;
|
input.disabled = false;
|
||||||
});
|
}
|
||||||
|
clearButtonState();
|
||||||
|
element.type = "submit";
|
||||||
} else {
|
} else {
|
||||||
form.submit();
|
form.submit();
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
var absenceForm = document.getElementById("absence_form");
|
const absenceForm = document.getElementById("absence_form");
|
||||||
|
|
||||||
if (id != 0) {
|
if (id == 0) {
|
||||||
syncFields(form, absenceForm, [
|
absenceForm.querySelector("[name=date_from]").value = form.id.replace("time-", "");
|
||||||
"date_from",
|
absenceForm.querySelector("[name=date_to]").value = form.id.replace("time-", "");
|
||||||
"date_to",
|
|
||||||
"aw_type",
|
|
||||||
"aw_id",
|
|
||||||
]);
|
|
||||||
} else {
|
} else {
|
||||||
absenceForm.querySelector("[name=date_from]").value = form.id.replace(
|
syncFields(form, absenceForm, ["date_from", "date_to", "aw_type", "aw_id"]);
|
||||||
"time-",
|
|
||||||
"",
|
|
||||||
);
|
|
||||||
absenceForm.querySelector("[name=date_to]").value = form.id.replace(
|
|
||||||
"time-",
|
|
||||||
"",
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function toggleAbsenceEdit(state) {
|
function toggleAbsenceEdit(state) {
|
||||||
var form = document.getElementById("absence_form");
|
const form = document.getElementById("absence_form");
|
||||||
if (state) {
|
if (state) {
|
||||||
form.classList.remove("hidden");
|
form.classList.remove("hidden");
|
||||||
form.scrollIntoView({
|
form.scrollIntoView({
|
||||||
@@ -66,22 +60,22 @@ function toggleAbsenceEdit(state) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function syncFields(from, to, fieldsToSync) {
|
function syncFields(from, to, fieldsToSync) {
|
||||||
fieldsToSync.forEach((name) => {
|
for (let field of fieldsToSync) {
|
||||||
const src = from.querySelector(`[name=${name}]`);
|
const src = from.querySelector(`[name=${field}]`);
|
||||||
const target = to.querySelector(`[name=${name}]`);
|
const target = to.querySelector(`[name=${field}]`);
|
||||||
if (!src || !target) return;
|
if (!src || !target) return;
|
||||||
target.value = src.value;
|
target.value = src.value;
|
||||||
});
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function navigateWeek(element, event, direction) {
|
function navigateWeek(element, event, direction) {
|
||||||
var dateInput = element.closest("form").querySelector("input[type=date]");
|
const dateInput = element.closest("form").querySelector("input[type=date]");
|
||||||
var date = dateInput.valueAsDate;
|
const date = dateInput.valueAsDate;
|
||||||
date.setDate(date.getDate() + 7 * direction);
|
date.setDate(date.getDate() + 7 * direction);
|
||||||
date.setHours(10);
|
date.setHours(10);
|
||||||
dateInput.valueAsDate = date;
|
dateInput.valueAsDate = date;
|
||||||
}
|
}
|
||||||
|
|
||||||
function logoutUser() {
|
function logoutUser() {
|
||||||
fetch("/user/logout", {}).then(() => window.location.reload());
|
fetch("/user/logout", {}).then(() => globalThis.location.reload());
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -93,61 +93,6 @@ templ TeamPage(weeks []models.WorkWeek, userWeek models.WorkWeek) {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@workWeekComponent(userWeek, false)
|
@workWeekComponent(userWeek, false)
|
||||||
// <div class="grid-sub lg:divide-x-1 max-md:divide-y-1 responsive @container">
|
|
||||||
// <div class="grid-cell col-span-full bg-neutral-300 lg:border-0">
|
|
||||||
// <h2 class="text-2xl uppercase font-bold">Eigene Abrechnung</h2>
|
|
||||||
// </div>
|
|
||||||
// <div class="grid-cell flex flex-col max-md:border-b-1 max-md:bg-neutral-300 gap-2 ">
|
|
||||||
// <div class="lg:hidden">
|
|
||||||
// @weekPicker(userWeek.WeekStart)
|
|
||||||
// </div>
|
|
||||||
// <h2 class="uppercase font-bold">{ fmt.Sprintf("%s %s", userWeek.User.Vorname, userWeek.User.Name) }</h2>
|
|
||||||
// <div class="grid grid-cols-5 gap-2 lg:grid-cols-1">
|
|
||||||
// <div class="col-span-2">
|
|
||||||
// <span class="flex flex-row gap-2 items-center">
|
|
||||||
// @statusCheckMark(userWeek.CheckStatus(), models.WeekStatusSent)
|
|
||||||
// Gesendet
|
|
||||||
// </span>
|
|
||||||
// <span class="flex flex-row gap-2 items-center">
|
|
||||||
// @statusCheckMark(userWeek.CheckStatus(), models.WeekStatusAccepted)
|
|
||||||
// Akzeptiert
|
|
||||||
// </span>
|
|
||||||
// </div>
|
|
||||||
// <div class="flex flex-row gap-2 col-span-3">
|
|
||||||
// @timeGaugeComponent(int8(progress), false)
|
|
||||||
// <div>
|
|
||||||
// <p>Arbeitszeit: { fmt.Sprintf("%s", helper.FormatDuration(userWeek.Worktime)) }</p>
|
|
||||||
// <p>Überstunden: { fmt.Sprintf("%s", helper.FormatDurationFill(userWeek.Overtime, true)) }</p>
|
|
||||||
// </div>
|
|
||||||
// </div>
|
|
||||||
// </div>
|
|
||||||
// </div>
|
|
||||||
// <div class="grid-cell col-span-3 flex flex-col @7xl:grid @7xl:grid-cols-5 gap-2 py-4 content-baseline">
|
|
||||||
// for _, day := range userWeek.Days {
|
|
||||||
// @defaultWeekDayComponent(userWeek.User, day)
|
|
||||||
// }
|
|
||||||
// </div>
|
|
||||||
// <div class="grid-cell flex flex-col gap-2 justify-between">
|
|
||||||
// <div class="max-md:hidden">
|
|
||||||
// @weekPicker(userWeek.WeekStart)
|
|
||||||
// </div>
|
|
||||||
// <form method="post" class="flex flex-col gap-2">
|
|
||||||
// <input type="hidden" name="method" value="send"/>
|
|
||||||
// <input type="hidden" name="user" value={ strconv.Itoa(userWeek.User.PersonalNummer) }/>
|
|
||||||
// <input type="hidden" name="week" value={ userWeek.WeekStart.Format(time.DateOnly) }/>
|
|
||||||
// switch userWeek.CheckStatus() {
|
|
||||||
// case models.WeekStatusNone:
|
|
||||||
// <p class="text-sm">an Vorgesetzten senden</p>
|
|
||||||
// case models.WeekStatusSent:
|
|
||||||
// <p class="text-sm">an Vorgesetzten gesendet</p>
|
|
||||||
// case models.WeekStatusAccepted:
|
|
||||||
// <p class="text-sm">vom Vorgesetzten bestätigt</p>
|
|
||||||
// }
|
|
||||||
// <button disabled?={ userWeek.Status < models.WeekStatusSent } type="submit" class="btn">Korrigieren</button>
|
|
||||||
// <button disabled?={ time.Since(userWeek.WeekStart) < 24*7*time.Hour || userWeek.Status >= models.WeekStatusSent || userWeek.RequiresAction() } type="submit" class="btn">Senden</button>
|
|
||||||
// </form>
|
|
||||||
// </div>
|
|
||||||
// </div>
|
|
||||||
if len(weeks) > 0 {
|
if len(weeks) > 0 {
|
||||||
<div class="grid-cell col-span-full bg-neutral-300">
|
<div class="grid-cell col-span-full bg-neutral-300">
|
||||||
<h2 class="text-2xl uppercase font-bold">Abrechnung Mitarbeiter</h2>
|
<h2 class="text-2xl uppercase font-bold">Abrechnung Mitarbeiter</h2>
|
||||||
|
|||||||
@@ -337,7 +337,7 @@ func TeamPresencePage(teamPresence map[models.User]bool) templ.Component {
|
|||||||
var templ_7745c5c3_Var11 string
|
var templ_7745c5c3_Var11 string
|
||||||
templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(user.Vorname)
|
templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(user.Vorname)
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 173, Col: 22}
|
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 118, Col: 22}
|
||||||
}
|
}
|
||||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11))
|
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11))
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
@@ -350,7 +350,7 @@ func TeamPresencePage(teamPresence map[models.User]bool) templ.Component {
|
|||||||
var templ_7745c5c3_Var12 string
|
var templ_7745c5c3_Var12 string
|
||||||
templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(user.Name)
|
templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(user.Name)
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 173, Col: 36}
|
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pages.templ`, Line: 118, Col: 36}
|
||||||
}
|
}
|
||||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12))
|
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12))
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
|
|||||||
@@ -45,7 +45,12 @@ templ PDFReportEmploye(e models.User, overtime, worktime time.Duration, workDays
|
|||||||
<p>{ workDay.Bookings[bookingI].BookingType.Name } </p>
|
<p>{ workDay.Bookings[bookingI].BookingType.Name } </p>
|
||||||
}
|
}
|
||||||
if workDay.IsKurzArbeit() {
|
if workDay.IsKurzArbeit() {
|
||||||
<p class="col-span-full">Kurzarbeit</p>
|
{{
|
||||||
|
timeFrom, timeTo := workDay.GenerateKurzArbeitBookings(e)
|
||||||
|
}}
|
||||||
|
<p>{ timeFrom.Format("15:04") }</p>
|
||||||
|
<p>{ timeTo.Format("15:04") }</p>
|
||||||
|
<p>Kurzarbeit</p>
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
{{
|
{{
|
||||||
|
|||||||
@@ -252,7 +252,35 @@ func PDFReportEmploye(e models.User, overtime, worktime time.Duration, workDays
|
|||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
if workDay.IsKurzArbeit() {
|
if workDay.IsKurzArbeit() {
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "<p class=\"col-span-full\">Kurzarbeit</p>")
|
|
||||||
|
timeFrom, timeTo := workDay.GenerateKurzArbeitBookings(e)
|
||||||
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "<p>")
|
||||||
|
if templ_7745c5c3_Err != nil {
|
||||||
|
return templ_7745c5c3_Err
|
||||||
|
}
|
||||||
|
var templ_7745c5c3_Var17 string
|
||||||
|
templ_7745c5c3_Var17, templ_7745c5c3_Err = templ.JoinStringErrs(timeFrom.Format("15:04"))
|
||||||
|
if templ_7745c5c3_Err != nil {
|
||||||
|
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pdf.templ`, Line: 51, Col: 36}
|
||||||
|
}
|
||||||
|
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var17))
|
||||||
|
if templ_7745c5c3_Err != nil {
|
||||||
|
return templ_7745c5c3_Err
|
||||||
|
}
|
||||||
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "</p><p>")
|
||||||
|
if templ_7745c5c3_Err != nil {
|
||||||
|
return templ_7745c5c3_Err
|
||||||
|
}
|
||||||
|
var templ_7745c5c3_Var18 string
|
||||||
|
templ_7745c5c3_Var18, templ_7745c5c3_Err = templ.JoinStringErrs(timeTo.Format("15:04"))
|
||||||
|
if templ_7745c5c3_Err != nil {
|
||||||
|
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pdf.templ`, Line: 52, Col: 34}
|
||||||
|
}
|
||||||
|
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var18))
|
||||||
|
if templ_7745c5c3_Err != nil {
|
||||||
|
return templ_7745c5c3_Err
|
||||||
|
}
|
||||||
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, "</p><p>Kurzarbeit</p>")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
@@ -260,25 +288,25 @@ func PDFReportEmploye(e models.User, overtime, worktime time.Duration, workDays
|
|||||||
} else {
|
} else {
|
||||||
|
|
||||||
absentDay, _ := day.(*models.Absence)
|
absentDay, _ := day.(*models.Absence)
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "<p class=\"col-span-full\">")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 22, "<p class=\"col-span-full\">")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
var templ_7745c5c3_Var17 string
|
var templ_7745c5c3_Var19 string
|
||||||
templ_7745c5c3_Var17, templ_7745c5c3_Err = templ.JoinStringErrs(absentDay.AbwesenheitTyp.Name)
|
templ_7745c5c3_Var19, templ_7745c5c3_Err = templ.JoinStringErrs(absentDay.AbwesenheitTyp.Name)
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pdf.templ`, Line: 54, Col: 62}
|
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pdf.templ`, Line: 59, Col: 62}
|
||||||
}
|
}
|
||||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var17))
|
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var19))
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, "</p>")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, "</p>")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 22, "</div>")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, "</div>")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
@@ -287,7 +315,7 @@ func PDFReportEmploye(e models.User, overtime, worktime time.Duration, workDays
|
|||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, " ")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, " ")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
@@ -295,7 +323,7 @@ func PDFReportEmploye(e models.User, overtime, worktime time.Duration, workDays
|
|||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, " ")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, " ")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
@@ -303,18 +331,18 @@ func PDFReportEmploye(e models.User, overtime, worktime time.Duration, workDays
|
|||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, " ")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 27, " ")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
if day.Date().Weekday() == time.Friday {
|
if day.Date().Weekday() == time.Friday {
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, "<p class=\"col-span-full bg-neutral-300\">Wochenende</p>")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, "<p class=\"col-span-full bg-neutral-300\">Wochenende</p>")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 27, "</div></content>")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, "</div></content>")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
@@ -338,9 +366,9 @@ func ColorDuration(d time.Duration, classes string) templ.Component {
|
|||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
ctx = templ.InitializeContext(ctx)
|
ctx = templ.InitializeContext(ctx)
|
||||||
templ_7745c5c3_Var18 := templ.GetChildren(ctx)
|
templ_7745c5c3_Var20 := templ.GetChildren(ctx)
|
||||||
if templ_7745c5c3_Var18 == nil {
|
if templ_7745c5c3_Var20 == nil {
|
||||||
templ_7745c5c3_Var18 = templ.NopComponent
|
templ_7745c5c3_Var20 = templ.NopComponent
|
||||||
}
|
}
|
||||||
ctx = templ.ClearChildren(ctx)
|
ctx = templ.ClearChildren(ctx)
|
||||||
|
|
||||||
@@ -348,38 +376,38 @@ func ColorDuration(d time.Duration, classes string) templ.Component {
|
|||||||
if d.Abs() < time.Minute {
|
if d.Abs() < time.Minute {
|
||||||
color = "text-neutral-300"
|
color = "text-neutral-300"
|
||||||
}
|
}
|
||||||
var templ_7745c5c3_Var19 = []any{color + " " + classes}
|
var templ_7745c5c3_Var21 = []any{color + " " + classes}
|
||||||
templ_7745c5c3_Err = templ.RenderCSSItems(ctx, templ_7745c5c3_Buffer, templ_7745c5c3_Var19...)
|
templ_7745c5c3_Err = templ.RenderCSSItems(ctx, templ_7745c5c3_Buffer, templ_7745c5c3_Var21...)
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, "<p class=\"")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 30, "<p class=\"")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
var templ_7745c5c3_Var20 string
|
var templ_7745c5c3_Var22 string
|
||||||
templ_7745c5c3_Var20, templ_7745c5c3_Err = templ.JoinStringErrs(templ.CSSClasses(templ_7745c5c3_Var19).String())
|
templ_7745c5c3_Var22, templ_7745c5c3_Err = templ.JoinStringErrs(templ.CSSClasses(templ_7745c5c3_Var21).String())
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pdf.templ`, Line: 1, Col: 0}
|
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pdf.templ`, Line: 1, Col: 0}
|
||||||
}
|
}
|
||||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var20))
|
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var22))
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, "\">")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 31, "\">")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
var templ_7745c5c3_Var21 string
|
var templ_7745c5c3_Var23 string
|
||||||
templ_7745c5c3_Var21, templ_7745c5c3_Err = templ.JoinStringErrs(helper.FormatDurationFill(d, true))
|
templ_7745c5c3_Var23, templ_7745c5c3_Err = templ.JoinStringErrs(helper.FormatDurationFill(d, true))
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pdf.templ`, Line: 76, Col: 72}
|
return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/pdf.templ`, Line: 81, Col: 72}
|
||||||
}
|
}
|
||||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var21))
|
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var23))
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 30, "</p>")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 32, "</p>")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -81,7 +81,7 @@ templ weekDayComponent(user models.User, day models.WorkDay) {
|
|||||||
templ workWeekComponent(week models.WorkWeek, onlyAccept bool) {
|
templ workWeekComponent(week models.WorkWeek, onlyAccept bool) {
|
||||||
{{
|
{{
|
||||||
year, kw := week.WeekStart.ISOWeek()
|
year, kw := week.WeekStart.ISOWeek()
|
||||||
progress := (float32(week.Worktime.Hours()) / week.User.ArbeitszeitPerWoche) * 100
|
progress := (float32(week.WorkTimeVirtual.Hours()) / week.User.ArbeitszeitPerWoche) * 100
|
||||||
}}
|
}}
|
||||||
<div class="employeComponent grid-sub responsive lg:divide-x-1 max-md:divide-y-1 @container">
|
<div class="employeComponent grid-sub responsive lg:divide-x-1 max-md:divide-y-1 @container">
|
||||||
<div class="grid-cell flex flex-col max-md:bg-neutral-300 gap-2">
|
<div class="grid-cell flex flex-col max-md:bg-neutral-300 gap-2">
|
||||||
|
|||||||
@@ -346,7 +346,7 @@ func workWeekComponent(week models.WorkWeek, onlyAccept bool) templ.Component {
|
|||||||
ctx = templ.ClearChildren(ctx)
|
ctx = templ.ClearChildren(ctx)
|
||||||
|
|
||||||
year, kw := week.WeekStart.ISOWeek()
|
year, kw := week.WeekStart.ISOWeek()
|
||||||
progress := (float32(week.Worktime.Hours()) / week.User.ArbeitszeitPerWoche) * 100
|
progress := (float32(week.WorkTimeVirtual.Hours()) / week.User.ArbeitszeitPerWoche) * 100
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, "<div class=\"employeComponent grid-sub responsive lg:divide-x-1 max-md:divide-y-1 @container\"><div class=\"grid-cell flex flex-col max-md:bg-neutral-300 gap-2\">")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, "<div class=\"employeComponent grid-sub responsive lg:divide-x-1 max-md:divide-y-1 @container\"><div class=\"grid-cell flex flex-col max-md:bg-neutral-300 gap-2\">")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ templ lineComponent() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
templ changeButtonComponent(id string, workDay bool) {
|
templ changeButtonComponent(id string, workDay bool) {
|
||||||
<button class="btn w-auto group/button" type="submit" onclick={ templ.JSFuncCall("editWorkday", templ.JSExpression("this"), templ.JSExpression("event"), id, workDay) }>
|
<button class="change-button-component btn w-auto group/button" type="button" onclick={ templ.JSFuncCall("editWorkday", templ.JSExpression("this"), templ.JSExpression("event"), id, workDay) }>
|
||||||
<p class="hidden md:block group-[.edit]/button:hidden">Ändern</p>
|
<p class="hidden md:block group-[.edit]/button:hidden">Ändern</p>
|
||||||
<p class="hidden group-[.edit]/button:md:block">Absenden</p>
|
<p class="hidden group-[.edit]/button:md:block">Absenden</p>
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" fill="currentColor" class="w-4 h-4 md:hidden">
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" fill="currentColor" class="w-4 h-4 md:hidden">
|
||||||
@@ -89,7 +89,7 @@ templ absenceComponent(a *models.Absence, isKurzarbeit bool) {
|
|||||||
</p>
|
</p>
|
||||||
<div class="w-full"></div>
|
<div class="w-full"></div>
|
||||||
if isKurzarbeit {
|
if isKurzarbeit {
|
||||||
<button onclick={ templ.JSFuncCall("editAbsence", templ.JSExpression("this"), templ.JSExpression("event"), "time-"+a.Date().Format("2006-01-02")) } class="hidden btn border-0 rounded-none grow-0 w-auto group-[.edit]:inline">Bearbeiten</button>
|
<button type="button" onclick={ templ.JSFuncCall("editWorkday", templ.JSExpression("this"), templ.JSExpression("event"), "time-"+a.Date().Format("2006-01-02"), false) } class="hidden btn border-0 rounded-none grow-0 w-auto group-[.edit]:inline">Bearbeiten</button>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
@@ -120,6 +120,9 @@ templ bookingComponent(booking models.Booking) {
|
|||||||
<input disabled name={ "booking_" + strconv.Itoa(booking.CounterId) } type="time" value={ booking.Timestamp.Format("15:04") } class="text-neutral-700 group-[.edit]:inline hidden bg-neutral-100 text-sm px-3 py-2 cursor-pointer"/>
|
<input disabled name={ "booking_" + strconv.Itoa(booking.CounterId) } type="time" value={ booking.Timestamp.Format("15:04") } class="text-neutral-700 group-[.edit]:inline hidden bg-neutral-100 text-sm px-3 py-2 cursor-pointer"/>
|
||||||
{ booking.GetBookingType() }
|
{ booking.GetBookingType() }
|
||||||
</p>
|
</p>
|
||||||
|
if booking.IsSubmittedAndChecked() {
|
||||||
|
<p>submitted</p>
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -69,7 +69,7 @@ func changeButtonComponent(id string, workDay bool) templ.Component {
|
|||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "<button class=\"btn w-auto group/button\" type=\"submit\" onclick=\"")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "<button class=\"change-button-component btn w-auto group/button\" type=\"button\" onclick=\"")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
@@ -394,15 +394,15 @@ func absenceComponent(a *models.Absence, isKurzarbeit bool) templ.Component {
|
|||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
if isKurzarbeit {
|
if isKurzarbeit {
|
||||||
templ_7745c5c3_Err = templ.RenderScriptItems(ctx, templ_7745c5c3_Buffer, templ.JSFuncCall("editAbsence", templ.JSExpression("this"), templ.JSExpression("event"), "time-"+a.Date().Format("2006-01-02")))
|
templ_7745c5c3_Err = templ.RenderScriptItems(ctx, templ_7745c5c3_Buffer, templ.JSFuncCall("editWorkday", templ.JSExpression("this"), templ.JSExpression("event"), "time-"+a.Date().Format("2006-01-02"), false))
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, "<button onclick=\"")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, "<button type=\"button\" onclick=\"")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
var templ_7745c5c3_Var22 templ.ComponentScript = templ.JSFuncCall("editAbsence", templ.JSExpression("this"), templ.JSExpression("event"), "time-"+a.Date().Format("2006-01-02"))
|
var templ_7745c5c3_Var22 templ.ComponentScript = templ.JSFuncCall("editWorkday", templ.JSExpression("this"), templ.JSExpression("event"), "time-"+a.Date().Format("2006-01-02"), false)
|
||||||
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ_7745c5c3_Var22.Call)
|
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ_7745c5c3_Var22.Call)
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
@@ -568,7 +568,17 @@ func bookingComponent(booking models.Booking) templ.Component {
|
|||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 39, "</p></div>")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 39, "</p>")
|
||||||
|
if templ_7745c5c3_Err != nil {
|
||||||
|
return templ_7745c5c3_Err
|
||||||
|
}
|
||||||
|
if booking.IsSubmittedAndChecked() {
|
||||||
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 40, "<p>submitted</p>")
|
||||||
|
if templ_7745c5c3_Err != nil {
|
||||||
|
return templ_7745c5c3_Err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 41, "</div>")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
@@ -597,7 +607,7 @@ func LegendComponent() templ.Component {
|
|||||||
templ_7745c5c3_Var31 = templ.NopComponent
|
templ_7745c5c3_Var31 = templ.NopComponent
|
||||||
}
|
}
|
||||||
ctx = templ.ClearChildren(ctx)
|
ctx = templ.ClearChildren(ctx)
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 40, "<div class=\"flex flex-row gap-4 md:mx-[10%] print:hidden\"><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-red-600\"></div><span>Fehler</span></div><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-orange-500\"></div><span>Arbeitszeit unter regulär</span></div><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-accent\"></div><span>Arbeitszeit vollständig</span></div><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-purple-600\"></div><span>Überstunden</span></div><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-neutral-400\"></div><span>Keine Buchungen</span></div></div>")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 42, "<div class=\"flex flex-row gap-4 md:mx-[10%] print:hidden\"><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-red-600\"></div><span>Fehler</span></div><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-orange-500\"></div><span>Arbeitszeit unter regulär</span></div><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-accent\"></div><span>Arbeitszeit vollständig</span></div><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-purple-600\"></div><span>Überstunden</span></div><div class=\"flex flex-row items-center gap-2\"><div class=\"rounded-full size-4 bg-neutral-400\"></div><span>Keine Buchungen</span></div></div>")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
return templ_7745c5c3_Err
|
return templ_7745c5c3_Err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -103,7 +103,7 @@ templ defaultDayComponent(day models.IWorkDay) {
|
|||||||
if day.IsWorkDay() {
|
if day.IsWorkDay() {
|
||||||
{{
|
{{
|
||||||
workDay, _ := day.(*models.WorkDay)
|
workDay, _ := day.(*models.WorkDay)
|
||||||
work, pause, overtime := workDay.GetAllWorkTimesVirtual(user)
|
work, pause, overtime := workDay.GetAllWorkTimesReal(user)
|
||||||
}}
|
}}
|
||||||
if day.RequiresAction() {
|
if day.RequiresAction() {
|
||||||
<p class="text-red-600">Bitte anpassen</p>
|
<p class="text-red-600">Bitte anpassen</p>
|
||||||
|
|||||||
@@ -297,7 +297,7 @@ func defaultDayComponent(day models.IWorkDay) templ.Component {
|
|||||||
if day.IsWorkDay() {
|
if day.IsWorkDay() {
|
||||||
|
|
||||||
workDay, _ := day.(*models.WorkDay)
|
workDay, _ := day.(*models.WorkDay)
|
||||||
work, pause, overtime := workDay.GetAllWorkTimesVirtual(user)
|
work, pause, overtime := workDay.GetAllWorkTimesReal(user)
|
||||||
if day.RequiresAction() {
|
if day.RequiresAction() {
|
||||||
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "<p class=\"text-red-600\">Bitte anpassen</p>")
|
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "<p class=\"text-red-600\">Bitte anpassen</p>")
|
||||||
if templ_7745c5c3_Err != nil {
|
if templ_7745c5c3_Err != nil {
|
||||||
|
|||||||
28
DB/initdb/01_schema.sql
Normal file → Executable file
28
DB/initdb/01_schema.sql
Normal file → Executable file
@@ -22,7 +22,7 @@ COMMENT ON COLUMN "anwesenheit"."geraet_id" IS 'ID des Lesegerätes';
|
|||||||
DROP TABLE IF EXISTS "s_anwesenheit_typen";
|
DROP TABLE IF EXISTS "s_anwesenheit_typen";
|
||||||
CREATE TABLE "s_anwesenheit_typen" (
|
CREATE TABLE "s_anwesenheit_typen" (
|
||||||
"anwesenheit_id" int2 PRIMARY KEY,
|
"anwesenheit_id" int2 PRIMARY KEY,
|
||||||
"anwesenheit_name" varchar(255)
|
"anwesenheit_name" varchar(255) NOT NULL
|
||||||
);
|
);
|
||||||
|
|
||||||
-- ----------------------------
|
-- ----------------------------
|
||||||
@@ -78,30 +78,34 @@ EXECUTE FUNCTION update_zuletzt_geandert();
|
|||||||
DROP TABLE IF EXISTS "wochen_report";
|
DROP TABLE IF EXISTS "wochen_report";
|
||||||
CREATE TABLE "wochen_report" (
|
CREATE TABLE "wochen_report" (
|
||||||
"id" serial PRIMARY KEY,
|
"id" serial PRIMARY KEY,
|
||||||
"personal_nummer" int4,
|
"personal_nummer" int4 NOT NULL,
|
||||||
"woche_start" date,
|
"woche_start" date NOT NULL,
|
||||||
"bestaetigt" bool DEFAULT FALSE,
|
"bestaetigt" bool DEFAULT FALSE,
|
||||||
"arbeitszeit" interval,
|
"arbeitszeit" interval NOT NULL,
|
||||||
"ueberstunden" interval,
|
"ueberstunden" interval NOT NULL,
|
||||||
|
"anwesenheiten" int ARRAY,
|
||||||
|
"abwesenheiten" int ARRAY,
|
||||||
UNIQUE ("personal_nummer", "woche_start")
|
UNIQUE ("personal_nummer", "woche_start")
|
||||||
);
|
);
|
||||||
|
|
||||||
DROP TABLE IF EXISTS "abwesenheit";
|
DROP TABLE IF EXISTS "abwesenheit";
|
||||||
CREATE TABLE "abwesenheit" (
|
CREATE TABLE "abwesenheit" (
|
||||||
"counter_id" bigserial PRIMARY KEY,
|
"counter_id" bigserial PRIMARY KEY,
|
||||||
"card_uid" varchar(255),
|
"card_uid" varchar(255) NOT NULL,
|
||||||
"abwesenheit_typ" int2,
|
"abwesenheit_typ" int2 NOT NULL,
|
||||||
"datum_from" timestamptz DEFAULT NOW()::DATE,
|
"datum_from" timestamptz DEFAULT NOW()::DATE NOT NULL,
|
||||||
"datum_to" timestamptz
|
"datum_to" timestamptz NOT NULL
|
||||||
);
|
);
|
||||||
|
|
||||||
DROP TABLE IF EXISTS "s_abwesenheit_typen";
|
DROP TABLE IF EXISTS "s_abwesenheit_typen";
|
||||||
CREATE TABLE "s_abwesenheit_typen" (
|
CREATE TABLE "s_abwesenheit_typen" (
|
||||||
"abwesenheit_id" int2 PRIMARY KEY,
|
"abwesenheit_id" int2 PRIMARY KEY NOT NULL,
|
||||||
"abwesenheit_name" varchar(255),
|
"abwesenheit_name" varchar(255) NOT NULL,
|
||||||
"arbeitszeit_equivalent" float4
|
"arbeitszeit_equivalent" float4 NOT NULL
|
||||||
);
|
);
|
||||||
|
|
||||||
|
COMMENT ON COLUMN "s_abwesenheit_typen"."arbeitszeit_equivalent" IS '0=keine Arbeitszeit; 1=Arbeitszeit auffüllen; 2=Arbeitszeit austauschen';
|
||||||
|
|
||||||
-- Adds crypto extension
|
-- Adds crypto extension
|
||||||
|
|
||||||
CREATE EXTENSION IF NOT EXISTS pgcrypto;
|
CREATE EXTENSION IF NOT EXISTS pgcrypto;
|
||||||
|
|||||||
0
DB/initdb/02_sample_data.sql
Normal file → Executable file
0
DB/initdb/02_sample_data.sql
Normal file → Executable file
0
DB/initdb/03_create_user.sh
Normal file → Executable file
0
DB/initdb/03_create_user.sh
Normal file → Executable file
2
Makefile
2
Makefile
@@ -44,7 +44,7 @@ generateFrontend:
|
|||||||
|
|
||||||
|
|
||||||
backend: generateFrontend login_registry
|
backend: generateFrontend login_registry
|
||||||
docker buildx build --platform linux/amd64,linux/arm64 -t ${IMAGE_REGISTRY}/${PACKAGE_OWNER}/arbeitszeitmessung:latest Backend --load #--push
|
docker buildx build --platform linux/amd64,linux/arm64 -t ${IMAGE_REGISTRY}/${PACKAGE_OWNER}/arbeitszeitmessung:latest Backend --push
|
||||||
# docker buildx build --platform linux/amd64,linux/arm64 -t ${IMAGE_REGISTRY}/${PACKAGE_OWNER}/arbeitszeitmessung:${GIT_COMMIT} Backend //--push
|
# docker buildx build --platform linux/amd64,linux/arm64 -t ${IMAGE_REGISTRY}/${PACKAGE_OWNER}/arbeitszeitmessung:${GIT_COMMIT} Backend //--push
|
||||||
|
|
||||||
test:
|
test:
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
# Arbeitszeitmessung
|
# Arbeitszeitmessung
|
||||||
|
|
||||||
|
[](https://sonar.letsstein.de/dashboard?id=arbeitszeitmessung)
|
||||||
|
|
||||||
bis jetzt ein einfaches Backend mit PostgreSQL Datenbank und GO Webserver um Arbeitszeitbuchungen per HTTP PUT einzufügen
|
bis jetzt ein einfaches Backend mit PostgreSQL Datenbank und GO Webserver um Arbeitszeitbuchungen per HTTP PUT einzufügen
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|||||||
63
WIR-typst/main.typ
Normal file
63
WIR-typst/main.typ
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
#set page("a4")
|
||||||
|
#set text(font: "Lato")
|
||||||
|
|
||||||
|
= Stunden
|
||||||
|
|
||||||
|
== Kim Mustermensch
|
||||||
|
|
||||||
|
Zeitraum: 01.10.2025 - 31.10.2025
|
||||||
|
|
||||||
|
Arbeitszeit: 136h 19min
|
||||||
|
|
||||||
|
Überstunden: -39h 41min
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// #show table.cell: it => {
|
||||||
|
// if it.y == 0 {
|
||||||
|
// set text(white)
|
||||||
|
// strong(it)
|
||||||
|
// } else if it.body == [] {
|
||||||
|
// // Replace empty cells with 'N/A'
|
||||||
|
// pad(..it.inset)[0min]
|
||||||
|
|
||||||
|
// } else {
|
||||||
|
// it
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
#let subgrid(body) = {
|
||||||
|
table.cell(colspan: 3, inset: 0em)[
|
||||||
|
#table(
|
||||||
|
columns: (1fr, 1fr, 1fr),
|
||||||
|
gutter: 0em,
|
||||||
|
stroke: black,
|
||||||
|
[..#body]
|
||||||
|
)
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
"01.09.2025",
|
||||||
|
"08:07",
|
||||||
|
"16:28",
|
||||||
|
"Büro",
|
||||||
|
"7h 51min",
|
||||||
|
"30min",
|
||||||
|
"-9min",
|
||||||
|
"02.09.2025",
|
||||||
|
// return work, pause, overtime
|
||||||
|
table.cell(colspan: 3, inset: 0em)[#table(
|
||||||
|
columns: (1fr, 1fr, 1fr),
|
||||||
|
gutter: 0em,
|
||||||
|
stroke: black,
|
||||||
|
[08:12], [16:24], [Büro],
|
||||||
|
[16:30], [17:24], [Homeoffice]
|
||||||
|
)],
|
||||||
|
"6h",
|
||||||
|
"0min",
|
||||||
|
"-1h 15min"
|
||||||
|
|
||||||
|
)
|
||||||
BIN
WIR-typst/template.pdf
Normal file
BIN
WIR-typst/template.pdf
Normal file
Binary file not shown.
58
WIR-typst/template.typ
Normal file
58
WIR-typst/template.typ
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
#let table-header(..headers) = {
|
||||||
|
table.header(
|
||||||
|
..headers.pos().map(h => strong(text(fill: white, h)))
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#set table(
|
||||||
|
stroke: black,
|
||||||
|
inset: .5em,
|
||||||
|
align: center,
|
||||||
|
)
|
||||||
|
|
||||||
|
#let abrechnung(meta, days) = {
|
||||||
|
set page(paper: "a4")
|
||||||
|
|
||||||
|
[= Abrechnung Arbeitszeit -- #meta.employee-name]
|
||||||
|
|
||||||
|
[Zeitraum: #meta.Zeitraum
|
||||||
|
|
||||||
|
Arbeitszeit: #user.Arbeitszeit
|
||||||
|
|
||||||
|
Überstunden: #user.Überstunden
|
||||||
|
]
|
||||||
|
|
||||||
|
table(
|
||||||
|
columns: (1fr, 1fr, 1fr, 1fr, 1fr, 1fr, 1.25fr),
|
||||||
|
fill: (x, y) =>
|
||||||
|
if y == 0 { gray },
|
||||||
|
align: center,
|
||||||
|
table-header(
|
||||||
|
[Datum], [Kommen], [Gehen], [Arbeitsart], [Stunden], [Pause], [Überstunden]
|
||||||
|
),
|
||||||
|
.. for day in days {
|
||||||
|
(
|
||||||
|
[#day.Date],
|
||||||
|
table.cell(colspan: 3, inset: 0em)[
|
||||||
|
#table(
|
||||||
|
columns: (1fr, 1fr, 1fr),
|
||||||
|
gutter: 0em,
|
||||||
|
stroke: black,
|
||||||
|
.. for Zeit in day.Zeiten {
|
||||||
|
(
|
||||||
|
[#Zeit.Kommen],
|
||||||
|
[#Zeit.Gehen],
|
||||||
|
[#Zeit.Art],
|
||||||
|
)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
],
|
||||||
|
[#day.Arbeitszeit],
|
||||||
|
[#day.Pause],
|
||||||
|
[#day.Überstunden],
|
||||||
|
|
||||||
|
)
|
||||||
|
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
36
WIR-typst/test.typ
Normal file
36
WIR-typst/test.typ
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
#let user = (
|
||||||
|
Name: "Mustermensch",
|
||||||
|
Vorname: "Kim",
|
||||||
|
Arbeitszeit: "139h 12min",
|
||||||
|
Überstunden: "-14h 12min"
|
||||||
|
)
|
||||||
|
|
||||||
|
#let meta = (
|
||||||
|
Zeitraum: "01.09.2025 - 30.09.2025",
|
||||||
|
KW: "26"
|
||||||
|
)
|
||||||
|
|
||||||
|
#let days = (
|
||||||
|
(
|
||||||
|
Date: "01.09.2025",
|
||||||
|
Zeiten: (
|
||||||
|
(Kommen: "07:17", Gehen: "14:13", Art: "Büro"),
|
||||||
|
(Kommen: "14:24", Gehen: "16:13", Art: "Homeoffice")
|
||||||
|
),
|
||||||
|
Arbeitszeit: "7h 32min",
|
||||||
|
Pause: "34min",
|
||||||
|
Überstunden: "12min"
|
||||||
|
),(
|
||||||
|
Date: "02.09.2025",
|
||||||
|
Zeiten: (
|
||||||
|
(Kommen: "07:23", Gehen: "14:21", Art: "Büro"),
|
||||||
|
(Kommen: "14:38", Gehen: "17:13", Art: "Homeoffice")
|
||||||
|
),
|
||||||
|
Arbeitszeit: "6h 22min",
|
||||||
|
Pause: "45min",
|
||||||
|
Überstunden: "-23min"
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
#import "template.typ": abrechnung
|
||||||
|
#show: doc => abrechnung(meta, user, days)
|
||||||
4
migrations/20251013212224_buchungs_array.down.sql
Normal file
4
migrations/20251013212224_buchungs_array.down.sql
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
-- reverse: modify "wochen_report" table
|
||||||
|
ALTER TABLE "wochen_report" DROP COLUMN "abwesenheiten", DROP COLUMN "anwesenheiten", ALTER COLUMN "arbeitszeit" DROP NOT NULL, ALTER COLUMN "ueberstunden" DROP NOT NULL, ALTER COLUMN "woche_start" DROP NOT NULL;
|
||||||
|
-- reverse: modify "abwesenheit" table
|
||||||
|
ALTER TABLE "abwesenheit" ALTER COLUMN "datum_to" DROP NOT NULL;
|
||||||
4
migrations/20251013212224_buchungs_array.up.sql
Normal file
4
migrations/20251013212224_buchungs_array.up.sql
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
-- modify "abwesenheit" table
|
||||||
|
ALTER TABLE "abwesenheit" ALTER COLUMN "datum_to" SET NOT NULL;
|
||||||
|
-- modify "wochen_report" table
|
||||||
|
ALTER TABLE "wochen_report" ALTER COLUMN "woche_start" SET NOT NULL, ALTER COLUMN "ueberstunden" SET NOT NULL, ALTER COLUMN "arbeitszeit" SET NOT NULL, ADD COLUMN "anwesenheiten" integer[] NULL, ADD COLUMN "abwesenheiten" integer[] NULL;
|
||||||
@@ -1,15 +1,9 @@
|
|||||||
h1:3AxgD8mnu/F+JGtJu9FZvA9Ro0UUtGPgyjskKtfTYUQ=
|
h1:gE7ikkZS7bQbedAjVspQKftSo5ODij2eiQGWbfQEYmI=
|
||||||
20250901201159_initial.down.sql h1:cmF5CvNGqEfcmbRgiqaqDWERdNNRaMzarbNLJ/Y35o4=
|
20250901201159_initial.up.sql h1:Mb1RlVdFvcxqU9HrSK6oNeURqFa3O4KzB3rDa+6+3gc=
|
||||||
20250901201159_initial.up.sql h1:Yrak/+wfQ4Tu/dVR/cUZ/75DlAcv4G/OJXDqpgSw47U=
|
20250901201250_control_tables.up.sql h1:a5LATgR/CRiC4GsqxkJ94TyJOxeTcW74eCnodIy+c1E=
|
||||||
20250901201250_control_tables.down.sql h1:f/KmhO9pOI45J8ZRjFonvD3CypB+rOoGOPN2WMFHvOw=
|
20250901201710_triggers_extension.up.sql h1:z9b6Hk9btE2Ns4mU7B16HjvYBP6EEwHAXVlvPpkn978=
|
||||||
20250901201250_control_tables.up.sql h1:of5E07p0N1aen9CdQNEOrO7ffbKZC6kp4oK5KPzU9+g=
|
20250903221313_overtime.up.sql h1:t/B435ShW5ZEnzC81jRABWVZ5gNm7tPZPnOO6/ZY6ow=
|
||||||
20250901201710_triggers_extension.down.sql h1:a9va3FSfHBWzODJSJO+ywNa2hiZwjG/vmvYGb3L1lnM=
|
20250903233030_non_null_contraints.up.sql h1:YKeYgazfh+jPyh7hFT/pV+By8eHnk1taXnlgSLyXSA0=
|
||||||
20250901201710_triggers_extension.up.sql h1:nUBPd2eDssi/TwMVF/nOJkIM5rUM0iINdg1K9pZRZN0=
|
20250904114004_intervals.up.sql h1:gDdN8cJ4xH1vQhAbbhqD5lwdyEO1N9EIqEYkmWGiWIU=
|
||||||
20250903221313_overtime.down.sql h1:X+jJESqcZ6ZTd2H563z6kRaXb4dn4sA02D3ck2795v8=
|
20250916093608_kurzarbeit.up.sql h1:yDAAMLyUXz6b7+MI6XK/HZMPzutKoT2NNNOCjFaqSts=
|
||||||
20250903221313_overtime.up.sql h1:C3DSiNVpe9v0Un1DEQ0lsy5yToR8iqcggv91GSr6tRE=
|
20251013212224_buchungs_array.up.sql h1:mbhvnwMUkEFFQQ41NC47auqxbtvNkztziWvpLDFm6tA=
|
||||||
20250903233030_non_null_contraints.down.sql h1:42TZzPsji2Ze50k6sLwgIuNo4Trk3m3ni/aIfQJ97dE=
|
|
||||||
20250903233030_non_null_contraints.up.sql h1:k6zR5YNSAP4fo5QEc58KZ0LxvEz1nl0X/AAcZ+TG3I4=
|
|
||||||
20250904114004_intervals.down.sql h1:SquJAPinzFIRN6fJjLLIRsz59Tyr4RwGiGuOFI/N1SQ=
|
|
||||||
20250904114004_intervals.up.sql h1:AFqncTGOiEZVBbhWFqN2zlQ7DyhybB5wJr6a36Atk1E=
|
|
||||||
20250916093608_kurzarbeit.down.sql h1:ljM1a1pQCxOQiXRaXU04GC4V9yy2y20x5eUNQ/zyx+o=
|
|
||||||
20250916093608_kurzarbeit.up.sql h1:pTiw0VfGaf26mhJg4wf98Fqwn1kShJ+PiN2PiM4q1kk=
|
|
||||||
|
|||||||
Reference in New Issue
Block a user