package suites import ( "context" "crypto/tls" "encoding/json" "fmt" "io" "log" "net/http" "os" "path/filepath" "runtime" "strings" "text/template" "github.com/go-rod/rod" "github.com/google/uuid" ) var browserPaths = []string{"/usr/bin/chromium-browser", "/usr/bin/chromium"} // ValidateBrowserPath validates the appropriate chromium browser path. func ValidateBrowserPath(path string) (browserPath string, err error) { var info os.FileInfo if info, err = os.Stat(path); err != nil { return "", err } else if info.IsDir() { return "", fmt.Errorf("browser cannot be a directory") } return path, nil } // GetBrowserPath retrieves the appropriate chromium browser path. func GetBrowserPath() (path string, err error) { browserPath := os.Getenv("BROWSER_PATH") if browserPath != "" { return ValidateBrowserPath(browserPath) } for _, browserPath = range browserPaths { if browserPath, err = ValidateBrowserPath(browserPath); err == nil { return browserPath, nil } } return "", fmt.Errorf("no chromium browser was detected in the known paths, set the BROWSER_PATH environment variable to override the path") } // GetLoginBaseURL returns the URL of the login portal and the path prefix if specified. func GetLoginBaseURL(baseDomain string) string { if PathPrefix != "" { return LoginBaseURLFmt(baseDomain) + PathPrefix } return LoginBaseURLFmt(baseDomain) } func (rs *RodSession) collectCoverage(page *rod.Page) { coverageDir := "../../web/.nyc_output" resp, err := page.Eval("() => JSON.stringify(window.__coverage__)") if err != nil { log.Fatal(err) } coverageData := fmt.Sprintf("%v", resp.Value) _ = os.MkdirAll(coverageDir, 0775) if coverageData != "" { err = os.WriteFile(fmt.Sprintf("%s/coverage-%s.json", coverageDir, uuid.New().String()), []byte(coverageData), 0664) //nolint:gosec if err != nil { log.Fatal(err) } err = filepath.Walk("../../web/.nyc_output", fixCoveragePath) if err != nil { log.Fatal(err) } } } func (rs *RodSession) collectScreenshot(err error, page *rod.Page) { if err == context.DeadlineExceeded && os.Getenv("CI") == t { base := "/buildkite/screenshots" build := os.Getenv("BUILDKITE_BUILD_NUMBER") suite := strings.ToLower(os.Getenv("SUITE")) job := os.Getenv("BUILDKITE_JOB_ID") path := filepath.Join(base, build, suite, job) if err := os.MkdirAll(path, 0755); err != nil { log.Fatal(err) } pc, _, _, _ := runtime.Caller(2) fn := runtime.FuncForPC(pc) p := "github.com/authelia/authelia/v4/internal/suites." r := strings.NewReplacer(p, "", "(", "", ")", "", "*", "", ".", "-") page.MustScreenshotFullPage(fmt.Sprintf("%s/%s.jpg", path, r.Replace(fn.Name()))) } } func fixCoveragePath(path string, file os.FileInfo, err error) error { if err != nil { return err } if file.IsDir() { return nil } coverage, err := filepath.Match("*.json", file.Name()) if err != nil { return err } if coverage { read, err := os.ReadFile(path) if err != nil { return err } wd, _ := os.Getwd() ciPath := strings.TrimSuffix(wd, "internal/suites") content := strings.ReplaceAll(string(read), "/node/src/app/", ciPath+"web/") err = os.WriteFile(path, []byte(content), 0) if err != nil { return err } } return nil } // getEnvInfoFromURL gets environments variables for specified cookie domain // this func makes a http call to https://login./override and is only useful for suite tests. func getDomainEnvInfo(domain string) (map[string]string, error) { info := make(map[string]string) client := &http.Client{ Transport: &http.Transport{ TLSClientConfig: &tls.Config{ InsecureSkipVerify: true, //nolint:gosec }, }, } var ( req *http.Request resp *http.Response body []byte err error ) targetURL := LoginBaseURLFmt(domain) + "/devworkflow" if req, err = http.NewRequest(http.MethodGet, targetURL, nil); err != nil { return info, err } req.Header.Set("X-Forwarded-Proto", "https") req.Header.Set("X-Forwarded-Host", domain) if resp, err = client.Do(req); err != nil { return info, err } if body, err = io.ReadAll(resp.Body); err != nil { return info, err } defer resp.Body.Close() if err = json.Unmarshal(body, &info); err != nil { return info, err } return info, nil } // generateDevEnvFile generates web/.env.development based on opts. func generateDevEnvFile(opts map[string]string) error { wd, _ := os.Getwd() path := strings.TrimSuffix(wd, "internal/suites") src := fmt.Sprintf("%s/web/.env.production", path) dst := fmt.Sprintf("%s/web/.env.development", path) tmpl, err := template.ParseFiles(src) if err != nil { return err } file, _ := os.Create(dst) defer file.Close() if err := tmpl.Execute(file, opts); err != nil { return err } return nil } // updateDevEnvFileForDomain updates web/.env.development. // this function only affects local dev environments. func updateDevEnvFileForDomain(domain string, setup bool) error { if os.Getenv("CI") == "true" { return nil } info, err := getDomainEnvInfo(domain) if err != nil { return err } err = generateDevEnvFile(info) if err != nil { return err } if !setup { err = waitUntilAutheliaFrontendIsReady(multiCookieDomainDockerEnvironment) if err != nil { return err } } return nil }