Add local.settings.php to every instance
This commit adds a new file 'local.settings.php' to each distillery instance. This file can be used to automatically edit global distillery settings.
This commit is contained in:
parent
6eab3ac311
commit
24ff81f7cd
13 changed files with 98 additions and 42 deletions
|
|
@ -58,7 +58,7 @@ func (bk backup) Run(context wisski_distillery.Context) error {
|
||||||
})
|
})
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errBackupFailed.Wrap(err)
|
return errBackupFailed.WrapError(err)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -83,15 +83,7 @@ func (rb rebuild) Run(context wisski_distillery.Context) (err error) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
smanager := instance.SystemManager()
|
return instance.SystemManager().Apply(context.Context, writer, sys)
|
||||||
|
|
||||||
if err := smanager.Apply(context.Context, writer, sys, true); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := smanager.RebuildSettings(context.Context, writer); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}, wissKIs, status.SmartMessage(func(item *wisski.WissKI) string {
|
}, wissKIs, status.SmartMessage(func(item *wisski.WissKI) string {
|
||||||
return fmt.Sprintf("rebuild %q", item.Slug)
|
return fmt.Sprintf("rebuild %q", item.Slug)
|
||||||
}))
|
}))
|
||||||
|
|
|
||||||
|
|
@ -30,19 +30,11 @@ func (*Rebuild) Action() InstanceAction {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *Rebuild) Act(ctx context.Context, instance *wisski.WissKI, in io.Reader, out io.Writer, params ...string) error {
|
func (r *Rebuild) Act(ctx context.Context, instance *wisski.WissKI, in io.Reader, out io.Writer, params ...string) error {
|
||||||
// read the flags of the instance to be provisioned
|
// read the flags of the instance to be rebuilt
|
||||||
var system models.System
|
var system models.System
|
||||||
if err := json.Unmarshal([]byte(params[0]), &system); err != nil {
|
if err := json.Unmarshal([]byte(params[0]), &system); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
smanager := instance.SystemManager()
|
return instance.SystemManager().Apply(ctx, out, system)
|
||||||
|
|
||||||
if err := smanager.Apply(ctx, out, system, true); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := smanager.RebuildSettings(ctx, out); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -13,6 +13,7 @@ import (
|
||||||
"github.com/FAU-CDI/wisski-distillery/pkg/execx"
|
"github.com/FAU-CDI/wisski-distillery/pkg/execx"
|
||||||
"github.com/FAU-CDI/wisski-distillery/pkg/unpack"
|
"github.com/FAU-CDI/wisski-distillery/pkg/unpack"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
"github.com/tkw1536/pkglib/fsx"
|
||||||
"github.com/tkw1536/pkglib/fsx/umaskfree"
|
"github.com/tkw1536/pkglib/fsx/umaskfree"
|
||||||
"github.com/tkw1536/pkglib/stream"
|
"github.com/tkw1536/pkglib/stream"
|
||||||
"gopkg.in/yaml.v3"
|
"gopkg.in/yaml.v3"
|
||||||
|
|
@ -188,8 +189,9 @@ type StackWithResources struct {
|
||||||
MakeDirsPerm fs.FileMode // permission for dirctories, defaults to [environment.DefaultDirCreate]
|
MakeDirsPerm fs.FileMode // permission for dirctories, defaults to [environment.DefaultDirCreate]
|
||||||
MakeDirs []string // directories to ensure that exist
|
MakeDirs []string // directories to ensure that exist
|
||||||
|
|
||||||
TouchFilesPerm fs.FileMode // permission for new files to touch, defaults to [environment.DefaultFileCreate]
|
TouchFilesPerm fs.FileMode // permission for new files to touch or create, defaults to [environment.DefaultFileCreate]
|
||||||
TouchFiles []string // Files to 'touch', i.e. ensure that exist; guaranteed to be run after MakeDirs
|
TouchFiles []string // Files to 'touch', i.e. ensure that exist; guaranteed to be run after MakeDirs
|
||||||
|
CreateFiles map[string]string // Files to 'create' but not update after they are setup; guaranteed to be run after MakeDirs
|
||||||
}
|
}
|
||||||
|
|
||||||
// InstallationContext is a context to install data in
|
// InstallationContext is a context to install data in
|
||||||
|
|
@ -270,7 +272,7 @@ func (is StackWithResources) Install(ctx context.Context, progress io.Writer, co
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// make sure that certain files exist
|
// touch files that should be created empty
|
||||||
for _, name := range is.TouchFiles {
|
for _, name := range is.TouchFiles {
|
||||||
// find the destination!
|
// find the destination!
|
||||||
dst := filepath.Join(is.Dir, name)
|
dst := filepath.Join(is.Dir, name)
|
||||||
|
|
@ -280,6 +282,26 @@ func (is StackWithResources) Install(ctx context.Context, progress io.Writer, co
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// make sure that certain files exist
|
||||||
|
for name, content := range is.CreateFiles {
|
||||||
|
// find the destination!
|
||||||
|
dst := filepath.Join(is.Dir, name)
|
||||||
|
|
||||||
|
exists, err := fsx.Exists(dst)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// create the file if it doesn't exist
|
||||||
|
if !exists {
|
||||||
|
fmt.Fprintf(progress, "[create] %s\n", dst)
|
||||||
|
if err := umaskfree.WriteFile(dst, []byte(content), umaskfree.DefaultFilePerm); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
fmt.Fprintf(progress, "[skip] %s\n", dst)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// check that the stack can be loaded
|
// check that the stack can be loaded
|
||||||
{
|
{
|
||||||
|
|
|
||||||
|
|
@ -22,4 +22,7 @@ const (
|
||||||
OntologyDirectory = SitesDirectory + "/default/files/ontology"
|
OntologyDirectory = SitesDirectory + "/default/files/ontology"
|
||||||
SitesDirectory = WebDirectory + "/sites"
|
SitesDirectory = WebDirectory + "/sites"
|
||||||
WissKIDirectory = WebDirectory + "/modules/contrib/wisski"
|
WissKIDirectory = WebDirectory + "/modules/contrib/wisski"
|
||||||
|
|
||||||
|
LocalSettingsPath = "/settings/local.php"
|
||||||
|
GlobalSettingsPath = "/settings/global.php"
|
||||||
)
|
)
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,4 @@
|
||||||
|
# THIS FILE IS GENERATED AUTOMATICALLY. DO NOT EDIT.
|
||||||
version: "3.7"
|
version: "3.7"
|
||||||
|
|
||||||
services:
|
services:
|
||||||
|
|
@ -43,6 +44,7 @@ services:
|
||||||
- ${DATA_PATH}/data:/var/www/data:rw
|
- ${DATA_PATH}/data:/var/www/data:rw
|
||||||
- ${DATA_PATH}/home:/var/www:rw
|
- ${DATA_PATH}/home:/var/www:rw
|
||||||
- ${DATA_PATH}/hostkeys:/ssh/hostkeys:rw
|
- ${DATA_PATH}/hostkeys:/ssh/hostkeys:rw
|
||||||
|
- ${LOCAL_SETTINGS_PATH}:${LOCAL_SETTINGS_MOUNT}:ro
|
||||||
|
|
||||||
networks:
|
networks:
|
||||||
default:
|
default:
|
||||||
|
|
|
||||||
16
internal/wisski/ingredient/barrel/local.settings.php
Normal file
16
internal/wisski/ingredient/barrel/local.settings.php
Normal file
|
|
@ -0,0 +1,16 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
// Use this file to manually tweak setttings of this instance in an update-agnostic way.
|
||||||
|
// This file will not be updated by future distillery updates.
|
||||||
|
//
|
||||||
|
// The settings.php file contains settings in the following order:
|
||||||
|
//
|
||||||
|
// - settings generated by the drupal installer
|
||||||
|
// - global distillery settings files
|
||||||
|
// - distillery generated configuration files
|
||||||
|
// - this file
|
||||||
|
//
|
||||||
|
// Because of caching, changes may require an instance restart to take effect.
|
||||||
|
|
||||||
|
// // e.g. to turn on verbose logging, uncomment the following line:
|
||||||
|
// $config["system.logging"]["error_level"] = "verbose";
|
||||||
|
|
@ -23,7 +23,7 @@ import (
|
||||||
// Provision applies defaults to flags, to ensure some values are set
|
// Provision applies defaults to flags, to ensure some values are set
|
||||||
func (manager *Manager) Provision(ctx context.Context, progress io.Writer, system models.System, flags Profile) error {
|
func (manager *Manager) Provision(ctx context.Context, progress io.Writer, system models.System, flags Profile) error {
|
||||||
// Force building and applying the system!
|
// Force building and applying the system!
|
||||||
if err := manager.dependencies.SystemManager.Apply(ctx, progress, system, false); err != nil {
|
if err := manager.dependencies.SystemManager.ApplyInitial(ctx, progress, system); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -130,7 +130,7 @@ func (provision *Manager) bootstrap(ctx context.Context, progress io.Writer, fla
|
||||||
// Rebuild the settings file
|
// Rebuild the settings file
|
||||||
logging.LogMessage(progress, "Rebuilding Settings")
|
logging.LogMessage(progress, "Rebuilding Settings")
|
||||||
{
|
{
|
||||||
if err := provision.dependencies.SystemManager.RebuildSettings(ctx, progress); err != nil {
|
if err := provision.dependencies.SystemManager.BuildSettings(ctx, progress); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -4,12 +4,19 @@ import (
|
||||||
"embed"
|
"embed"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
||||||
|
_ "embed"
|
||||||
|
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/dis/component"
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component"
|
||||||
)
|
)
|
||||||
|
|
||||||
//go:embed all:barrel
|
//go:embed all:barrel
|
||||||
var barrelResources embed.FS
|
var barrelResources embed.FS
|
||||||
|
|
||||||
|
const localSettingsName = "settings.local.php"
|
||||||
|
|
||||||
|
//go:embed local.settings.php
|
||||||
|
var localSettingsTemplate string
|
||||||
|
|
||||||
// Barrel returns a stack representing the running WissKI Instance
|
// Barrel returns a stack representing the running WissKI Instance
|
||||||
func (barrel *Barrel) Stack() component.StackWithResources {
|
func (barrel *Barrel) Stack() component.StackWithResources {
|
||||||
return component.StackWithResources{
|
return component.StackWithResources{
|
||||||
|
|
@ -20,6 +27,10 @@ func (barrel *Barrel) Stack() component.StackWithResources {
|
||||||
Resources: barrelResources,
|
Resources: barrelResources,
|
||||||
ContextPath: filepath.Join("barrel"),
|
ContextPath: filepath.Join("barrel"),
|
||||||
|
|
||||||
|
CreateFiles: map[string]string{
|
||||||
|
localSettingsName: localSettingsTemplate,
|
||||||
|
},
|
||||||
|
|
||||||
EnvContext: map[string]string{
|
EnvContext: map[string]string{
|
||||||
"DOCKER_NETWORK_NAME": barrel.Malt.Config.Docker.Network(),
|
"DOCKER_NETWORK_NAME": barrel.Malt.Config.Docker.Network(),
|
||||||
|
|
||||||
|
|
@ -31,6 +42,9 @@ func (barrel *Barrel) Stack() component.StackWithResources {
|
||||||
"DATA_PATH": filepath.Join(barrel.FilesystemBase, "data"),
|
"DATA_PATH": filepath.Join(barrel.FilesystemBase, "data"),
|
||||||
"RUNTIME_DIR": barrel.Malt.Config.Paths.RuntimeDir(),
|
"RUNTIME_DIR": barrel.Malt.Config.Paths.RuntimeDir(),
|
||||||
|
|
||||||
|
"LOCAL_SETTINGS_PATH": filepath.Join(barrel.FilesystemBase, localSettingsName),
|
||||||
|
"LOCAL_SETTINGS_MOUNT": LocalSettingsPath,
|
||||||
|
|
||||||
"BARREL_BASE_IMAGE": barrel.GetDockerBaseImage(),
|
"BARREL_BASE_IMAGE": barrel.GetDockerBaseImage(),
|
||||||
"IIP_SERVER_ENABLED": barrel.GetIIPServerEnabled(),
|
"IIP_SERVER_ENABLED": barrel.GetIIPServerEnabled(),
|
||||||
"OPCACHE_MODE": barrel.OpCacheMode(),
|
"OPCACHE_MODE": barrel.OpCacheMode(),
|
||||||
|
|
|
||||||
|
|
@ -7,8 +7,9 @@ import (
|
||||||
"github.com/FAU-CDI/wisski-distillery/pkg/logging"
|
"github.com/FAU-CDI/wisski-distillery/pkg/logging"
|
||||||
)
|
)
|
||||||
|
|
||||||
// RebuildSettings (re-)configures settings.php for the provided running instance
|
// BuildSettings sets up global settings.php configuration settings.php for the provided running instance
|
||||||
func (smanager *SystemManager) RebuildSettings(ctx context.Context, progress io.Writer) (err error) {
|
// This doesn't need to be called manually.
|
||||||
|
func (smanager *SystemManager) BuildSettings(ctx context.Context, progress io.Writer) (err error) {
|
||||||
logging.LogMessage(progress, "Updating TRUSTED_HOST_PATTERNS in settings.php")
|
logging.LogMessage(progress, "Updating TRUSTED_HOST_PATTERNS in settings.php")
|
||||||
{
|
{
|
||||||
if err := smanager.dependencies.Settings.SetTrustedDomain(ctx, nil, smanager.Domain()); err != nil {
|
if err := smanager.dependencies.Settings.SetTrustedDomain(ctx, nil, smanager.Domain()); err != nil {
|
||||||
|
|
|
||||||
|
|
@ -21,19 +21,33 @@ type SystemManager struct {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Apply applies a specific system version to this barrel.
|
// Apply applies the given system configuration to this instance and (re-)starts the system.
|
||||||
// If start is true, also starts the container.
|
func (smanager *SystemManager) Apply(ctx context.Context, progress io.Writer, system models.System) (err error) {
|
||||||
func (smanager *SystemManager) Apply(ctx context.Context, progress io.Writer, system models.System, start bool) (err error) {
|
if err := smanager.apply(ctx, progress, system, true); err != nil {
|
||||||
// setup the new docker image
|
return err
|
||||||
smanager.Instance.System = system
|
}
|
||||||
|
|
||||||
// save in bookkeeping
|
if err := smanager.BuildSettings(ctx, progress); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ApplyInitial builds the base image, but does not start it
|
||||||
|
func (smanager *SystemManager) ApplyInitial(ctx context.Context, progress io.Writer, system models.System) error {
|
||||||
|
return smanager.apply(ctx, progress, system, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
// apply stores the new configuration and builds the base image
|
||||||
|
// start inidicates if the image should be started afterwards
|
||||||
|
func (smanager *SystemManager) apply(ctx context.Context, progress io.Writer, system models.System, start bool) error {
|
||||||
|
// store the new system configuration
|
||||||
|
smanager.Instance.System = system
|
||||||
if err := smanager.dependencies.Bookkeeping.Save(ctx); err != nil {
|
if err := smanager.dependencies.Bookkeeping.Save(ctx); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Apply Content-Security-Policy!
|
// build and start the barrel
|
||||||
|
|
||||||
// and rebuild
|
|
||||||
return smanager.dependencies.Barrel.Build(ctx, progress, start)
|
return smanager.dependencies.Barrel.Build(ctx, progress, start)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -7,6 +7,7 @@ import (
|
||||||
|
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/phpx"
|
"github.com/FAU-CDI/wisski-distillery/internal/phpx"
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/wisski/ingredient"
|
"github.com/FAU-CDI/wisski-distillery/internal/wisski/ingredient"
|
||||||
|
"github.com/FAU-CDI/wisski-distillery/internal/wisski/ingredient/barrel"
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/wisski/ingredient/php"
|
"github.com/FAU-CDI/wisski-distillery/internal/wisski/ingredient/php"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
@ -41,6 +42,8 @@ var (
|
||||||
errFailedInstallDistillerySettings = errors.New("failed to install distillery settings")
|
errFailedInstallDistillerySettings = errors.New("failed to install distillery settings")
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// SetTrustedDomain configures the trusted domain setting for the given instance.
|
||||||
|
// Note that this removes any installed distillery settings.
|
||||||
func (settings *Settings) SetTrustedDomain(ctx context.Context, server *phpx.Server, domain string) error {
|
func (settings *Settings) SetTrustedDomain(ctx context.Context, server *phpx.Server, domain string) error {
|
||||||
var ok bool
|
var ok bool
|
||||||
|
|
||||||
|
|
@ -51,14 +54,12 @@ func (settings *Settings) SetTrustedDomain(ctx context.Context, server *phpx.Ser
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// GlobalSettingsPath is the global path to distillery settings
|
|
||||||
const GlobalSettingsPath = "/distillery_settings.php"
|
|
||||||
|
|
||||||
func (settings *Settings) InstallDistillerySettings(ctx context.Context, server *phpx.Server) error {
|
func (settings *Settings) InstallDistillerySettings(ctx context.Context, server *phpx.Server) error {
|
||||||
var ok bool
|
var ok bool
|
||||||
|
|
||||||
err := settings.dependencies.PHP.ExecScript(ctx, server, &ok, settingsPHP, "install_settings_include", []string{
|
err := settings.dependencies.PHP.ExecScript(ctx, server, &ok, settingsPHP, "install_settings_include", []string{
|
||||||
GlobalSettingsPath,
|
barrel.LocalSettingsPath,
|
||||||
|
barrel.GlobalSettingsPath,
|
||||||
})
|
})
|
||||||
if err == nil && !ok {
|
if err == nil && !ok {
|
||||||
err = errFailedInstallDistillerySettings
|
err = errFailedInstallDistillerySettings
|
||||||
|
|
|
||||||
|
|
@ -43,7 +43,6 @@ function set_setting(string $name, mixed $value): bool {
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// reset the file mode
|
// reset the file mode
|
||||||
return chmod($filename, $old);
|
return chmod($filename, $old);
|
||||||
}
|
}
|
||||||
|
|
@ -80,7 +79,7 @@ function install_settings_include(array $paths): bool {
|
||||||
// add code to include the file if it exists
|
// add code to include the file if it exists
|
||||||
$code = $code . 'if (file_exists(' . $the_path . ')) { include_once ' . $the_path . '; }' . "\n";
|
$code = $code . 'if (file_exists(' . $the_path . ')) { include_once ' . $the_path . '; }' . "\n";
|
||||||
}
|
}
|
||||||
$code = $code . "// </distillery-settings-include>";
|
$code = $code . "// </distillery-settings-include>\n";
|
||||||
|
|
||||||
// and store the settings
|
// and store the settings
|
||||||
try {
|
try {
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue