diff --git a/.gitignore b/.gitignore
index 3292d647c680aeb8fe718e8094d1274c5149a844..ecdfdf7be9d0f55f55e571a8229a175d5ff279cd 100644
--- a/.gitignore
+++ b/.gitignore
@@ -4,6 +4,7 @@
 /pkg
 /logs
 /vendor
+/config
 
 media-repo*.yaml
 homeserver.yaml
diff --git a/Dockerfile b/Dockerfile
index f972452a1b174228dc02b4f15d7be2271440590f..7e3eae95aa5e9523f3a79ca59e9716e616fdae22 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -25,6 +25,8 @@ COPY ./config.sample.yaml /etc/media-repo.yaml.sample
 COPY ./migrations /var/lib/media-repo-migrations
 COPY ./docker/run.sh /usr/local/bin/
 
+ENV REPO_CONFIG=/data/media-repo.yaml
+
 CMD /usr/local/bin/run.sh
 VOLUME ["/data", "/media"]
 EXPOSE 8000
diff --git a/cmd/media_repo/main.go b/cmd/media_repo/main.go
index 7c485c57a284d2e9171a4606c68fd4e17af2a5a4..a29cdb12a73821c70b997e84a10cf6b30612b902 100644
--- a/cmd/media_repo/main.go
+++ b/cmd/media_repo/main.go
@@ -4,6 +4,7 @@ import (
 	"context"
 	"flag"
 	"fmt"
+	"os"
 
 	"github.com/sirupsen/logrus"
 	"github.com/turt2live/matrix-media-repo/api/webserver"
@@ -23,6 +24,12 @@ func main() {
 	templatesPath := flag.String("templates", "./templates", "The absolute path for the templates folder")
 	flag.Parse()
 
+	// Override config path with config for Docker users
+	configEnv := os.Getenv("REPO_CONFIG")
+	if configEnv != "" {
+		configPath = &configEnv
+	}
+
 	config.Path = *configPath
 	config.Runtime.MigrationsPath = *migrationsPath
 	config.Runtime.TemplatesPath = *templatesPath
diff --git a/common/config/config.go b/common/config/config.go
index 234eb527df558f3d3b10a5e49bcc105be93a0dde..659adc7e8b1163c4bb0e3082c2fee961c64dfd39 100644
--- a/common/config/config.go
+++ b/common/config/config.go
@@ -4,8 +4,11 @@ import (
 	"fmt"
 	"io/ioutil"
 	"os"
+	"path"
+	"sort"
 	"sync"
 
+	"github.com/sirupsen/logrus"
 	"gopkg.in/yaml.v2"
 )
 
@@ -172,7 +175,7 @@ func ReloadConfig() error {
 	c := NewDefaultConfig()
 
 	// Write a default config if the one given doesn't exist
-	_, err := os.Stat(Path)
+	info, err := os.Stat(Path)
 	exists := err == nil || !os.IsNotExist(err)
 	if !exists {
 		fmt.Println("Generating new configuration...")
@@ -197,16 +200,45 @@ func ReloadConfig() error {
 		}
 	}
 
-	f, err := os.Open(Path)
+	// Get new info about the possible directory after creating
+	info, err = os.Stat(Path)
 	if err != nil {
 		return err
 	}
-	defer f.Close()
 
-	buffer, err := ioutil.ReadAll(f)
-	err = yaml.Unmarshal(buffer, &c)
-	if err != nil {
-		return err
+	pathsOrdered := make([]string, 0)
+	if info.IsDir() {
+		logrus.Info("Config is a directory - loading all files over top of each other")
+
+		files, err := ioutil.ReadDir(Path)
+		if err != nil {
+			return err
+		}
+
+		for _, f := range files {
+			pathsOrdered = append(pathsOrdered, path.Join(Path, f.Name()))
+		}
+
+		sort.Strings(pathsOrdered)
+	} else {
+		pathsOrdered = append(pathsOrdered, Path)
+	}
+
+	for _, p := range pathsOrdered {
+		logrus.Info("Loading config file: ", p)
+		f, err := os.Open(p)
+		if err != nil {
+			return err
+		}
+
+		//noinspection GoDeferInLoop
+		defer f.Close()
+
+		buffer, err := ioutil.ReadAll(f)
+		err = yaml.Unmarshal(buffer, &c)
+		if err != nil {
+			return err
+		}
 	}
 
 	instance = c
diff --git a/docs/config.md b/docs/config.md
new file mode 100644
index 0000000000000000000000000000000000000000..a91f250411f336f23d705de48375878bdbfbd12e
--- /dev/null
+++ b/docs/config.md
@@ -0,0 +1,57 @@
+# Media repository configuration
+
+Simple deployments are going to make the best use of copy/pasting the sample config and using that. More complicated
+deployments might want to make use of layering/splitting out their configs more verbosely. Splitting out the configs
+also gives admins the opportunity to have more detailed per-domain control over their repository.
+
+To make use of the layering, use the `-config` switch to point it at the directory containing your config files. In
+Docker this takes the shape of setting the environment variable `REPO_CONFIG` to something like `/data/config`.
+
+Config files/directories are automatically watched to apply changes on the fly.
+
+## Structure
+
+Config directories are shallow and do not recurse (this means all your files go into one giant directory). Files are
+read in alphabetical order - it is recommended to prefix config files with a number for predictable ordering.
+
+Files override one another unless they are indicated as being per-domain (more on that later on in this doc). Before
+reading the config directory, the media repo will apply the default config, which is the same as the sample config.
+As it loads each file, it overrides whatever the file has defined.
+
+Simply put, given `00-main.yaml`, `01-bind.yaml`, and `02-datastores.yaml` the media repo will read the defaults, then
+apply 00, then 01, then 02. The file names do not matter aside from application order.
+
+## Per-domain configs
+
+When using per-domain configs the `homeservers` field of the main config can be ignored. The `homeservers` option
+is still respected for simple configuration of domains, though it is recommended to use per-domain configs if you're
+splitting out your overall config.
+
+Per-domain configs go in the same directory as all the other config files with the same ordering behaviour. To flag
+a config as a per-domain config, ensure the `homeserver` property is set.
+
+A minimal per-domain config would be:
+```yaml
+homeserver: example.org
+csApi: "https://example.org"
+backoffAt: 10
+adminApiKind: "matrix"
+```
+
+Any options from the main config can then be overridden per-domain with the exception of:
+* `homeservers` - because why would you.
+* `database` - because the database is for the whole process.
+* `repo` - because the listener configuration is for the whole process.
+* `sharedSecretAuth` - because the option doesn't apply to a particular domain.
+* `rateLimit` - because this configuration is applied before the host is known.
+* `metrics` - because this affects the whole process.
+
+To override a value, simply provide it in any valid per-domain config:
+
+```yaml
+homeserver: example.org
+identicons:
+  enabled: false
+```
+
+Per-domain configs can also be layered - just ensure that each layer has the `homeserver` property in it.