1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
|
package disk
import (
"net/http"
"path"
"path/filepath"
"strings"
"gitlab.com/gitlab-org/gitlab-pages/internal/host"
"gitlab.com/gitlab-org/gitlab-pages/internal/serving"
)
const (
subgroupScanLimit int = 21
// maxProjectDepth is set to the maximum nested project depth in gitlab (21) plus 3.
// One for the project, one for the first empty element of the split (URL.Path starts with /),
// and one for the real file path
maxProjectDepth int = subgroupScanLimit + 3
)
// Group represents a GitLab group with project configs and subgroups
type Group struct {
name string
// nested groups
subgroups subgroups
// group domains:
projects projects
}
type projects map[string]*projectConfig
type subgroups map[string]*Group
func (g *Group) digProjectWithSubpath(parentPath string, keys []string) (*projectConfig, string, string) {
if len(keys) >= 1 {
head := keys[0]
tail := keys[1:]
currentPath := path.Join(parentPath, head)
search := strings.ToLower(head)
if project := g.projects[search]; project != nil {
return project, currentPath, path.Join(tail...)
}
if subgroup := g.subgroups[search]; subgroup != nil {
return subgroup.digProjectWithSubpath(currentPath, tail)
}
}
return nil, "", ""
}
// Look up a project inside the domain based on the host and path. Returns the
// project and its name (if applicable)
func (g *Group) getProjectConfigWithSubpath(r *http.Request) (*projectConfig, string, string, string) {
// Check for a project specified in the URL: http://group.gitlab.io/projectA
// If present, these projects shadow the group domain.
split := strings.SplitN(r.URL.Path, "/", maxProjectDepth)
if len(split) >= 2 {
projectConfig, projectPath, urlPath := g.digProjectWithSubpath("", split[1:])
if projectConfig != nil {
return projectConfig, "/" + projectPath, projectPath, urlPath
}
}
// Since the URL doesn't specify a project (e.g. http://mydomain.gitlab.io),
// return the group project if it exists.
if host := host.FromRequest(r); host != "" {
if groupProject := g.projects[host]; groupProject != nil {
return groupProject, "/", host, strings.Join(split[1:], "/")
}
}
return nil, "", "", ""
}
// Resolve tries to find project and its config recursively for a given request
// to a group domain
func (g *Group) Resolve(r *http.Request) (*serving.LookupPath, string, error) {
projectConfig, prefix, projectPath, subPath := g.getProjectConfigWithSubpath(r)
if projectConfig == nil {
return nil, "", nil // it is not an error when project does not exist
}
lookupPath := &serving.LookupPath{
Prefix: prefix,
Path: filepath.Join(g.name, projectPath, "public") + "/",
IsNamespaceProject: projectConfig.NamespaceProject,
IsHTTPSOnly: projectConfig.HTTPSOnly,
HasAccessControl: projectConfig.AccessControl,
ProjectID: projectConfig.ID,
}
return lookupPath, subPath, nil
}
|