mirror of
https://github.com/zitadel/zitadel.git
synced 2025-12-06 16:12:13 +00:00
# Which Problems Are Solved - The previous monorepo in monorepo structure for the login app and its related packages was fragmented, complicated and buggy. - The process for building and testing the login container was inconsistent between local development and CI. - Lack of clear documentation as well as easy and reliable ways for non-frontend developers to reproduce and fix failing PR checks locally. # How the Problems Are Solved - Consolidated the login app and its related npm packages by moving the main package to `apps/login/apps/login` and merging `apps/login/packages/integration` and `apps/login/packages/acceptance` into the main `apps/login` package. - Migrated from Docker Compose-based test setups to dev container-based setups, adding support for multiple dev container configurations: - `.devcontainer/base` - `.devcontainer/turbo-lint-unit` - `.devcontainer/turbo-lint-unit-debug` - `.devcontainer/login-integration` - `.devcontainer/login-integration-debug` - Added npm scripts to run the new dev container setups, enabling exact reproduction of GitHub PR checks locally, and updated the pipeline to use these containers. - Cleaned up Dockerfiles and docker-bake.hcl files to only build the production image for the login app. - Cleaned up compose files to focus on dev environments in dev containers. - Updated `CONTRIBUTING.md` with guidance on running and debugging PR checks locally using the new dev container approach. - Introduced separate Dockerfiles for the login app to distinguish between using published client packages and building clients from local protos. - Ensured the login container is always built in the pipeline for use in integration and acceptance tests. - Updated Makefile and GitHub Actions workflows to use `--frozen-lockfile` for installing pnpm packages, ensuring reproducible installs. - Disabled GitHub release creation by the changeset action. - Refactored the `/build` directory structure for clarity and maintainability. - Added a `clean` command to `docks/package.json`. - Experimentally added `knip` to the `zitadel-client` package for improved linting of dependencies and exports. # Additional Changes - Fixed Makefile commands for consistency and reliability. - Improved the structure and clarity of the `/build` directory to support seamless integration of the login build. - Enhanced documentation and developer experience for running and debugging CI checks locally. # Additional Context - See updated `CONTRIBUTING.md` for new local development and debugging instructions. - These changes are a prerequisite for further improvements to the CI pipeline and local development workflow. - Closes #10276
112 lines
3.3 KiB
Go
112 lines
3.3 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
)
|
|
|
|
type serializableData struct {
|
|
ContextInfo map[string]interface{} `json:"contextInfo,omitempty"`
|
|
Args map[string]interface{} `json:"args,omitempty"`
|
|
}
|
|
|
|
type response struct {
|
|
Recipient string `json:"recipient,omitempty"`
|
|
}
|
|
|
|
func main() {
|
|
port := flag.String("port", "3333", "used port for the sink")
|
|
email := flag.String("email", "/email", "path for a sent email")
|
|
emailKey := flag.String("email-key", "recipientEmailAddress", "value in the sent context info of the email used as key to retrieve the notification")
|
|
sms := flag.String("sms", "/sms", "path for a sent sms")
|
|
smsKey := flag.String("sms-key", "recipientPhoneNumber", "value in the sent context info of the sms used as key to retrieve the notification")
|
|
notification := flag.String("notification", "/notification", "path to receive the notification")
|
|
flag.Parse()
|
|
|
|
messages := make(map[string]serializableData)
|
|
|
|
http.HandleFunc(*email, func(w http.ResponseWriter, r *http.Request) {
|
|
data, err := io.ReadAll(r.Body)
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
serializableData := serializableData{}
|
|
if err := json.Unmarshal(data, &serializableData); err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
email, ok := serializableData.ContextInfo[*emailKey].(string)
|
|
if !ok {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
fmt.Println(email + ": " + string(data))
|
|
messages[email] = serializableData
|
|
io.WriteString(w, "Email!\n")
|
|
})
|
|
|
|
http.HandleFunc(*sms, func(w http.ResponseWriter, r *http.Request) {
|
|
data, err := io.ReadAll(r.Body)
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
serializableData := serializableData{}
|
|
if err := json.Unmarshal(data, &serializableData); err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
phone, ok := serializableData.ContextInfo[*smsKey].(string)
|
|
if !ok {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
fmt.Println(phone + ": " + string(data))
|
|
messages[phone] = serializableData
|
|
io.WriteString(w, "SMS!\n")
|
|
})
|
|
|
|
http.HandleFunc(*notification, func(w http.ResponseWriter, r *http.Request) {
|
|
data, err := io.ReadAll(r.Body)
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
response := response{}
|
|
if err := json.Unmarshal(data, &response); err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
msg, ok := messages[response.Recipient]
|
|
if !ok {
|
|
http.Error(w, "No messages found for recipient: "+response.Recipient, http.StatusNotFound)
|
|
return
|
|
}
|
|
serializableData, err := json.Marshal(msg)
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
w.Header().Set("Content-Type", "application/json")
|
|
io.WriteString(w, string(serializableData))
|
|
})
|
|
|
|
fmt.Println("Starting server on", *port)
|
|
fmt.Println(*email, " for email handling")
|
|
fmt.Println(*sms, " for sms handling")
|
|
fmt.Println(*notification, " for retrieving notifications")
|
|
http.Handle("/healthy", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { return }))
|
|
fmt.Println("/healthy returns 200 OK")
|
|
err := http.ListenAndServe(":"+*port, nil)
|
|
if err != nil {
|
|
panic("Server could not be started: " + err.Error())
|
|
}
|
|
}
|