Allow exposing graphdb and phpmyadmin via config

This commit is contained in:
Tom Wiesing 2023-11-02 13:31:54 +01:00
parent 9f7ec20593
commit a90c68bf5e
No known key found for this signature in database
8 changed files with 139 additions and 14 deletions

View file

@ -43,6 +43,13 @@ http:
# In the future, it will be enabled by default, but at this point it is not. # In the future, it will be enabled by default, but at this point it is not.
api: null api: null
# Enable serving the triplestore webinterface publically under the "_ts" domain
ts: null
# Enable serving the phpmyadmin webinterface publically under the "_phymyadmin" domain
phpmyadmin: null
# Configuration for the (public) homepage of the distillery. # Configuration for the (public) homepage of the distillery.
home: home:
# the title of the distillery to be set # the title of the distillery to be set

View file

@ -2,6 +2,7 @@ package config
import ( import (
"fmt" "fmt"
"html/template"
"net/url" "net/url"
"strings" "strings"
@ -27,6 +28,38 @@ type HTTPConfig struct {
// API determines if the API is enabled. // API determines if the API is enabled.
// In a future version of the distillery, it will be enabled by default. // In a future version of the distillery, it will be enabled by default.
API validators.NullableBool `yaml:"api" validate:"bool" default:"false"` API validators.NullableBool `yaml:"api" validate:"bool" default:"false"`
// TS determintes if the special Triplestore domain is enabled.
TS validators.NullableBool `yaml:"ts" validate:"bool" default:"false"`
// PhpMyAdmin determines if the special PhpMyAdmin domain is enabled.
PhpMyAdmin validators.NullableBool `yaml:"phpmyadmin" validate:"bool" default:"false"`
}
// TSDomain returns the full url to the triplestore, if any
func (hcfg HTTPConfig) TSURL() template.URL {
return hcfg.optionalURL(TriplestoreDomain.Domain(), hcfg.TS)
}
func (hcfg HTTPConfig) PhpMyAdminURL() template.URL {
return hcfg.optionalURL(PHPMyAdminDomain.Domain(), hcfg.PhpMyAdmin)
}
// optionalURL returns the public-facing url to domain if enabled is true.
func (hcfg HTTPConfig) optionalURL(domain string, enabled validators.NullableBool) template.URL {
if !(enabled.Set && enabled.Value) {
return ""
}
u := url.URL{
Scheme: "http",
Host: hcfg.Domains(domain)[0],
Path: "/",
}
if hcfg.HTTPSEnabled() {
u.Scheme = "https"
}
return template.URL(u.String())
} }
// JoinPath returns the root public url joined with the provided parts. // JoinPath returns the root public url joined with the provided parts.
@ -56,20 +89,24 @@ func (hcfg HTTPConfig) HTTPSEnabled() bool {
return hcfg.CertbotEmail != "" return hcfg.CertbotEmail != ""
} }
// SpecialDomain represents a reserved domain
type SpecialDomain string type SpecialDomain string
var ( var (
TriplestoreDomain SpecialDomain = "ts" TriplestoreDomain SpecialDomain = "ts"
PHPMyAdminDomain SpecialDomain = "phpmyadmin"
) )
func (hcfg HTTPConfig) SpecialDomain(domain SpecialDomain) string { func (sd SpecialDomain) Domain() string {
return fmt.Sprintf("%s.%s", string(domain)+"_", hcfg.PrimaryDomain) return "_" + string(sd)
} }
// Domains adds the given subdomain to the primary and alias domains. // Domains adds the given subdomain to the primary and alias domains.
// If sub is empty, returns only the domains. // If sub is empty, returns only the domains.
// //
// sub is not otherwise validated, and should be normalized by the caller. // sub is not otherwise validated, and should be normalized by the caller.
//
// It is guaranteed that the first domain returned will always be the primary domain.
func (hcfg HTTPConfig) Domains(sub string) []string { func (hcfg HTTPConfig) Domains(sub string) []string {
domains := append([]string{hcfg.PrimaryDomain}, hcfg.ExtraDomains...) domains := append([]string{hcfg.PrimaryDomain}, hcfg.ExtraDomains...)
if sub == "" { if sub == "" {

View file

@ -32,22 +32,14 @@
</table> </table>
</div> </div>
<div class="pure-u-1-1">
<h2 id="overview">Distillery Configuration</h2>
</div>
<div class="pure-u-1-1"> <div class="pure-u-1-1">
<pre> <h2 id="overview">Distillery Services</h2>
<code class="language-yaml">{{ .Config.MarshalSensitive }}</code>
</pre>
</div>
<div class="pure-u-1-1">
<h2 id="overview">Distillery Passwords</h2>
<table class="pure-table pure-table-bordered padding"> <table class="pure-table pure-table-bordered padding">
<thead> <thead>
<tr> <tr>
<th>Service</th> <th>Service</th>
<th>Interface</th>
<th>Username</th> <th>Username</th>
<th>Password</th> <th>Password</th>
</tr> </tr>
@ -57,6 +49,14 @@
<td> <td>
SQL SQL
</td> </td>
<td>
{{ $pma := .Config.HTTP.PhpMyAdminURL }}
{{ if $pma }}
<a href="{{ $pma }}" class="pure-button pure-button-small" target="_blank" rel="noopener noreferer">PHPMyAdmin</a>
{{ else }}
(accessible via SSH)
{{ end }}
</td>
<td> <td>
<code class="copy">{{ .Config.SQL.AdminUsername }}</code> <code class="copy">{{ .Config.SQL.AdminUsername }}</code>
</td> </td>
@ -68,6 +68,14 @@
<td> <td>
Triplestore Triplestore
</td> </td>
<td>
{{ $ts := .Config.HTTP.TSURL }}
{{ if $ts }}
<a href="{{ $ts }}" class="pure-button pure-button-small" target="_blank" rel="noopener noreferer">Workbench</a>
{{ else }}
(accessible via SSH)
{{ end }}
</td>
<td> <td>
<code class="copy">{{ .Config.TS.AdminUsername }}</code> <code class="copy">{{ .Config.TS.AdminUsername }}</code>
</td> </td>
@ -77,4 +85,14 @@
</tr> </tr>
</tbody> </tbody>
</table> </table>
</div> </div>
<div class="pure-u-1-1">
<h2 id="overview">Distillery Configuration</h2>
</div>
<div class="pure-u-1-1">
<pre>
<code class="language-yaml">{{ .Config.MarshalSensitive }}</code>
</pre>
</div>

View file

@ -5,8 +5,11 @@ import (
"path/filepath" "path/filepath"
"time" "time"
"github.com/FAU-CDI/wisski-distillery/internal/config"
"github.com/FAU-CDI/wisski-distillery/internal/dis/component" "github.com/FAU-CDI/wisski-distillery/internal/dis/component"
"github.com/tkw1536/pkglib/fsx/umaskfree" "github.com/tkw1536/pkglib/fsx/umaskfree"
"github.com/tkw1536/pkglib/yamlx"
"gopkg.in/yaml.v3"
) )
type SQL struct { type SQL struct {
@ -47,6 +50,23 @@ func (sql *SQL) Stack() component.StackWithResources {
EnvContext: map[string]string{ EnvContext: map[string]string{
"DOCKER_NETWORK_NAME": sql.Config.Docker.Network(), "DOCKER_NETWORK_NAME": sql.Config.Docker.Network(),
"HTTPS_ENABLED": sql.Config.HTTP.HTTPSEnabledEnv(), "HTTPS_ENABLED": sql.Config.HTTP.HTTPSEnabledEnv(),
"HOST_RULE": sql.Config.HTTP.HostRule(config.PHPMyAdminDomain.Domain()),
},
ComposerYML: func(root *yaml.Node) (*yaml.Node, error) {
// phpmyadmin is exposed => everything is fine
if sql.Config.HTTP.PhpMyAdmin.Set && sql.Config.HTTP.PhpMyAdmin.Value {
return root, nil
}
// not exposed => remove the appropriate labels
if err := yamlx.ReplaceWith(root, []string{
"eu.wiss-ki.barrel.distillery=${DOCKER_NETWORK_NAME}",
}, "services", "phpmyadmin", "labels"); err != nil {
return nil, err
}
return root, nil
}, },
MakeDirsPerm: umaskfree.DefaultDirPerm, MakeDirsPerm: umaskfree.DefaultDirPerm,

View file

@ -29,6 +29,15 @@ services:
- 127.0.0.1:8080:80 - 127.0.0.1:8080:80
labels: labels:
- "eu.wiss-ki.barrel.distillery=${DOCKER_NETWORK_NAME}" - "eu.wiss-ki.barrel.distillery=${DOCKER_NETWORK_NAME}"
# expose phpmyadmin (removed when disabled in global config)
- "traefik.enable=True"
- "traefik.http.routers.core_phpmyadmin.rule=${HOST_RULE}"
- "traefik.http.routers.core_phpmyadmin.tls=${HTTPS_ENABLED}"
- "traefik.http.routers.core_phpmyadmin.tls.certresolver=distillery"
- "traefik.http.services.core_phpmyadmin.loadbalancer.server.port=80"
depends_on: depends_on:
- sql - sql
restart: always restart: always

View file

@ -5,7 +5,10 @@ import (
"path/filepath" "path/filepath"
"time" "time"
"github.com/FAU-CDI/wisski-distillery/internal/config"
"github.com/FAU-CDI/wisski-distillery/internal/dis/component" "github.com/FAU-CDI/wisski-distillery/internal/dis/component"
"github.com/tkw1536/pkglib/yamlx"
"gopkg.in/yaml.v3"
) )
type Triplestore struct { type Triplestore struct {
@ -43,6 +46,24 @@ func (ts *Triplestore) Stack() component.StackWithResources {
EnvContext: map[string]string{ EnvContext: map[string]string{
"DOCKER_NETWORK_NAME": ts.Config.Docker.Network(), "DOCKER_NETWORK_NAME": ts.Config.Docker.Network(),
"HOST_RULE": ts.Config.HTTP.HostRule(config.TriplestoreDomain.Domain()),
"HTTPS_ENABLED": ts.Config.HTTP.HTTPSEnabledEnv(),
},
ComposerYML: func(root *yaml.Node) (*yaml.Node, error) {
// ts is exposed => everything is fine
if ts.Config.HTTP.TS.Set && ts.Config.HTTP.TS.Value {
return root, nil
}
// not exposed => remove the appropriate labels
if err := yamlx.ReplaceWith(root, []string{
"eu.wiss-ki.barrel.distillery=${DOCKER_NETWORK_NAME}",
}, "services", "triplestore", "labels"); err != nil {
return nil, err
}
return root, nil
}, },
MakeDirs: []string{ MakeDirs: []string{

View file

@ -51,7 +51,7 @@ EXPOSE 7200
# setup a healthcheck, that checks if the server is up. # setup a healthcheck, that checks if the server is up.
RUN apt-get update && apt-get install -y curl RUN apt-get update && apt-get install -y curl
HEALTHCHECK --interval=30s --timeout=30s --start-period=5s --retries=3 CMD curl --fail 127.0.0.1:7200/rest/repositories || exit 1 # HEALTHCHECK --interval=30s --timeout=30s --start-period=5s --retries=3 CMD curl --fail 127.0.0.1:7200/rest/repositories || exit 1
# Add volumes for data, work and logs as these might be accessible from the outside. # Add volumes for data, work and logs as these might be accessible from the outside.
# To add your own configuration, manually mount a config file into /opt/graphdb/work # To add your own configuration, manually mount a config file into /opt/graphdb/work

View file

@ -15,6 +15,19 @@ services:
GDB_HEAP_SIZE: 16G GDB_HEAP_SIZE: 16G
labels: labels:
- "eu.wiss-ki.barrel.distillery=${DOCKER_NETWORK_NAME}" - "eu.wiss-ki.barrel.distillery=${DOCKER_NETWORK_NAME}"
# expose the triplestore (removed when disabled in global config)
- "traefik.enable=True"
- "traefik.http.routers.core_ts.rule=${HOST_RULE}"
- "traefik.http.routers.core_ts.tls=${HTTPS_ENABLED}"
- "traefik.http.routers.core_ts.tls.certresolver=distillery"
- "traefik.http.routers.core_ts.middlewares=core_ts@docker"
- "traefik.http.services.core_ts.loadbalancer.server.port=7200"
- "traefik.http.services.core_ts.loadbalancer.passHostHeader=false"
- "traefik.http.middlewares.core_ts.headers.customRequestHeaders.host=localhost"
restart: always restart: always