diff --git a/lib/app/app.js b/lib/app/app.js
--- a/lib/app/app.js
+++ b/lib/app/app.js
@@ -17,6 +17,7 @@
 
 const WwwServerFactory = locreq("lib/http/http.js");
 const DatastoreMongoFactory = locreq("lib/datastore/db.js");
+const MetadataFactory = locreq("lib/app/metadata.js");
 
 const load_base_chips = locreq("lib/app/load-base-chips");
 const default_config = locreq("default_config.json");
@@ -26,6 +27,7 @@
 
 class App {
 	constructor(custom_config, manifest) {
+		this.status = "stopped";
 		this.Sealious = Sealious;
 		this.setupEventEmitter();
 
@@ -55,6 +57,7 @@
 		load_base_chips(this);
 
 		this.Datastore = new DatastoreMongoFactory(this);
+		this.Metadata = MetadataFactory(this);
 
 		this.FieldType = Sealious.FieldType.bind(Sealious.FieldType, this);
 		this.Collection = Sealious.Collection.bind(Sealious.Collection, this);
@@ -107,6 +110,7 @@
 	}
 
 	async start() {
+		this.status = "starting";
 		assert(
 			["dev", "production"].includes(
 				this.ConfigManager.get("core.environment")
@@ -118,13 +122,16 @@
 		await this.Mail.init();
 		await this.ChipManager.start_chips();
 		await this.emit("start");
+		this.status = "running";
 		return this;
 	}
 
 	async stop() {
+		this.status = "stopping";
 		await this.emit("stop");
 		await this.WwwServer.stop();
 		await this.Datastore.stop();
+		this.status = "stopped";
 	}
 }
 
diff --git a/lib/app/base-chips/collections/password-reset-intents.subtest.js b/lib/app/base-chips/collections/password-reset-intents.subtest.js
--- a/lib/app/base-chips/collections/password-reset-intents.subtest.js
+++ b/lib/app/base-chips/collections/password-reset-intents.subtest.js
@@ -2,7 +2,7 @@
 const axios = require("axios");
 const assert = require("assert");
 const { promise_timeout } = locreq("test_utils");
-const with_test_app = locreq("test_utils/with-test-app.js");
+const { with_running_app } = locreq("test_utils/with-test-app.js");
 
 describe("password-reset-intents", () => {
 	async function create_a_user(app) {
@@ -19,7 +19,7 @@
 	}
 
 	it("tells you if the email address doesn't exist", async () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			try {
 				await axios.post(
 					`${base_url}/api/v1/collections/password-reset-intents`,
@@ -38,7 +38,7 @@
 		}));
 
 	it("allows anyone to create an intent, if the email exists", async () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			await create_a_user(app);
 			const data = (await axios.post(
 				`${base_url}/api/v1/collections/password-reset-intents`,
@@ -53,7 +53,7 @@
 		}));
 
 	it("tells you if the email address is malformed", async () =>
-		with_test_app(async ({ base_url }) => {
+		with_running_app(async ({ base_url }) => {
 			try {
 				await axios.post(
 					`${base_url}/api/v1/collections/password-reset-intents`,
@@ -72,7 +72,7 @@
 		}));
 
 	it("sends an email with the reset password link", async () =>
-		with_test_app(async ({ app, base_url, mail_api }) => {
+		with_running_app(async ({ app, base_url, mail_api }) => {
 			await create_a_user(app);
 			const data = (await axios.post(
 				`${base_url}/api/v1/collections/password-reset-intents`,
diff --git a/lib/app/base-chips/collections/users.subtest.js b/lib/app/base-chips/collections/users.subtest.js
--- a/lib/app/base-chips/collections/users.subtest.js
+++ b/lib/app/base-chips/collections/users.subtest.js
@@ -1,11 +1,11 @@
 const locreq = require("locreq")(__dirname);
 const assert = require("assert");
-const with_test_app = locreq("test_utils/with-test-app.js");
+const { with_running_app } = locreq("test_utils/with-test-app.js");
 
 describe("users", () => {
 	describe("auto create admin", () => {
 		it("should automatically create a registration intent for the admin user", async () =>
-			with_test_app(async ({ app, mail_api }) => {
+			with_running_app(async ({ app, mail_api }) => {
 				const registration_intents = await app.run_action(
 					new app.Sealious.SuperContext(),
 					["collections", "registration-intents"],
diff --git a/lib/app/base-chips/field-types/field-types.test.js b/lib/app/base-chips/field-types/field-types.test.js
--- a/lib/app/base-chips/field-types/field-types.test.js
+++ b/lib/app/base-chips/field-types/field-types.test.js
@@ -1,4 +1,5 @@
 describe("field types", () => {
 	require("./single_reference.subtest.js");
 	require("./text.subtest.js");
+	require("./reverse-single-reference.subtest.js");
 });
diff --git a/lib/app/base-chips/field-types/reverse-single-reference.js b/lib/app/base-chips/field-types/reverse-single-reference.js
new file mode 100644
--- /dev/null
+++ b/lib/app/base-chips/field-types/reverse-single-reference.js
@@ -0,0 +1,211 @@
+"use strict";
+const locreq = require("locreq")(__dirname);
+const Promise = require("bluebird");
+const Errors = locreq("lib/response/error.js");
+const assert = require("assert");
+
+function params_to_cache_key(collection, field_name, params) {
+	return `${collection.name}___${field_name}-reverse-single-reference(${
+		params.collection.name
+	},${params.collection.field_name}).last_update`;
+}
+
+async function update_cache(
+	app,
+	collection,
+	field_name,
+	params,
+	resource_ids = undefined
+) {
+	let pipeline;
+	const referencing_field_name = params.field_name;
+	const referencing_collection = params.collection;
+	if (resource_ids) {
+		assert(Array.isArray(resource_ids));
+		pipeline = [
+			{ $match: { [`body.${referencing_field_name}`]: { $in: resource_ids } } },
+		];
+	} else {
+		pipeline = [];
+	}
+	pipeline.push({
+		$group: {
+			_id: `$body.${referencing_field_name}`,
+			referenced_by: { $push: `$sealious_id` },
+		},
+	});
+	const to_update = await app.Datastore.aggregate(
+		referencing_collection.name,
+		pipeline
+	);
+	if (resource_ids) {
+		for (let resource_id of resource_ids) {
+			if (to_update.filter(e => e._id === resource_id).length === 0) {
+				to_update.push({ _id: resource_id, referenced_by: [] });
+			}
+		}
+	}
+	for (let entry of to_update) {
+		await app.Datastore.update(
+			collection.name,
+			{ sealious_id: entry._id },
+			{ $set: { [`body.${field_name}`]: entry.referenced_by } }
+		);
+	}
+}
+
+const reverse_single_reference_factory = app => {
+	return {
+		name: "reverse-single-reference",
+
+		get_description: function() {
+			return "Shows which resources from given collection point to this resource in a given field.";
+		},
+
+		get_default_value: async () => [],
+
+		is_proper_value: function(context, params, new_value) {
+			return context.is_super
+				? Promise.resolve()
+				: Promise.reject("This is a read-only field");
+		},
+
+		filter_to_query: async function(context, params, field_filter) {
+			if (typeof field_filter !== "object") {
+				return {
+					$eq: field_filter,
+				};
+			}
+			const matches = await app.run_action(
+				context,
+				["collections", params.collection.name],
+				"show",
+				{
+					filter: field_filter,
+				}
+			);
+			const ids = matches.map(resource => resource.id);
+			return {
+				$in: ids,
+			};
+		},
+
+		format: function(context, params, decoded_value, format) {
+			// format can be "expand" or "deep-expand:<depth>", like "deep-expand:3"
+			if (!format) {
+				return decoded_value; // just the IDs
+			}
+
+			const format_params = format.split(":");
+
+			if (!["expand", "deep-expand"].includes(format_params[0])) {
+				return decoded_value;
+			}
+
+			const query_format = {};
+			if (format_params[0] === "deep-expand" && format_params[1] > 1) {
+				for (const field_name in params.collection.fields) {
+					const field = params.collection.fields[field_name];
+					if (field.type.name === "single_reference") {
+						query_format[field_name] = `deep-expand:${parseInt(
+							format_params[1],
+							10
+						) - 1}`;
+					}
+				}
+			}
+			const resource_ids = decoded_value;
+			return Promise.map(resource_ids, async resource_id =>
+				app.run_action(
+					context,
+					["collections", params.collection.name, resource_id],
+					"show",
+					{ format: query_format }
+				)
+			);
+		},
+
+		init: async (collection, field_name, params) => {
+			assert(
+				params.collection instanceof app.Sealious.Collection,
+				"'params.collection' should be an instance of Collection"
+			);
+			assert(
+				params.collection.fields[params.field_name],
+				`Collection '${
+					params.collection.name
+				}' does not contain a field named ${params.field_name}.`
+			);
+			app.on("start", async () => {
+				const last_modified_resource_in_reference_collection = (await app.run_action(
+					new app.Sealious.SuperContext(),
+					["collections", params.collection.name],
+					"show",
+					{
+						sort: { "last_modified_context.timestamp": "desc" },
+						pagination: { items: 1 },
+					}
+				))[0];
+
+				if (last_modified_resource_in_reference_collection) {
+					const last_modified_resource_timestamp =
+						last_modified_resource_in_reference_collection.last_modified_context
+							.timestamp;
+					const last_field_cache_update =
+						(await app.Metadata.get(
+							params_to_cache_key(collection, field_name, params)
+						)) || 0;
+					if (last_modified_resource_timestamp > last_field_cache_update) {
+						await update_cache(app, collection, field_name, params);
+						await app.Metadata.set(
+							params_to_cache_key(collection, field_name, params),
+							Date.now()
+						);
+					}
+				}
+			});
+			app.on(
+				new RegExp(`post:collections\.${params.collection.name}:create`),
+				async (path, event_params, resource) => {
+					const referenced_id = resource.body[params.field_name];
+					await update_cache(app, collection, field_name, params, [
+						referenced_id,
+					]);
+				}
+			);
+			app.on(
+				new RegExp(`post:collections\.${params.collection.name}\..*:delete`),
+				async (path, event_params, resource) => {
+					const deleted_id = path[2];
+					const affected = await app.Datastore.find(collection.name, {
+						[`body.${field_name}`]: deleted_id,
+					});
+					const affected_ids = affected.map(document => document.sealious_id);
+					await update_cache(app, collection, field_name, params, affected_ids);
+				}
+			);
+			app.on(
+				new RegExp(`post:collections\.${params.collection.name}\..*:edit`),
+				async (path, event_params, resource) => {
+					if (!event_params.hasOwnProperty(params.field_name)) return;
+					const edited_id = path[2];
+					const no_longer_referenced = await app.Datastore.find(
+						collection.name,
+						{
+							[`body.${field_name}`]: edited_id,
+						}
+					);
+					const affected_ids = no_longer_referenced.map(
+						document => document.sealious_id
+					);
+					if (event_params[params.field_name]) {
+						affected_ids.push(event_params[params.field_name]);
+					}
+					await update_cache(app, collection, field_name, params, affected_ids);
+				}
+			);
+		},
+	};
+};
+
+module.exports = reverse_single_reference_factory;
diff --git a/lib/app/base-chips/field-types/reverse-single-reference.subtest.js b/lib/app/base-chips/field-types/reverse-single-reference.subtest.js
new file mode 100644
--- /dev/null
+++ b/lib/app/base-chips/field-types/reverse-single-reference.subtest.js
@@ -0,0 +1,185 @@
+const assert = require("assert");
+const Promise = require("bluebird");
+const locreq = require("locreq")(__dirname);
+const axios = require("axios");
+const { create_resource_as } = locreq("test_utils");
+const { with_stopped_app, with_running_app } = locreq(
+	"test_utils/with-test-app.js"
+);
+const DatastoreMongoFactory = locreq("lib/datastore/db.js");
+
+describe("reverse-single-reference", () => {
+	async function create_referencing_collections(app, with_reverse) {
+		const A = app.createChip(app.Sealious.Collection, {
+			name: "A",
+			fields: [
+				{
+					name: "reference_to_b",
+					type: "single_reference",
+					params: { collection: "B" },
+				},
+				{
+					name: "pairity",
+					type: "text",
+				},
+			],
+		});
+		const B = app.createChip(app.Sealious.Collection, {
+			name: "B",
+			fields: [{ name: "number", type: "int" }],
+		});
+		if (with_reverse) {
+			B.add_field({
+				name: "references_in_a",
+				type: "reverse-single-reference",
+				params: { collection: A, field_name: "reference_to_b" },
+			});
+		}
+	}
+
+	async function create_resources(app) {
+		const numbers = [1, 2, 3];
+		const bs = await Promise.map(numbers, number =>
+			app.run_action(
+				new app.Sealious.SuperContext(),
+				["collections", "B"],
+				"create",
+				{ number }
+			)
+		);
+		for (let b of bs) {
+			for (let i = 1; i <= b.body.number; i++) {
+				await app.run_action(
+					new app.Sealious.SuperContext(),
+					["collections", "A"],
+					"create",
+					{ reference_to_b: b.id, pairity: i % 2 ? "odd" : "even" }
+				);
+			}
+		}
+	}
+
+	async function with_reverse(fn) {
+		return with_stopped_app(async args => {
+			await create_referencing_collections(args.app, "with_reverse" && true);
+			await args.app.start();
+			await create_resources(args.app);
+			await fn(args);
+		});
+	}
+
+	it("recreates the cached values if the field has just been added", async () => {
+		await with_stopped_app(async ({ app, dont_clear_database_on_stop }) => {
+			await create_referencing_collections(app, "with_reverse" && false);
+			await app.start();
+			await create_resources(app);
+			dont_clear_database_on_stop();
+		});
+		await with_stopped_app(async ({ base_url, app, rest_api }) => {
+			await create_referencing_collections(app, "with_reverse" && true);
+			await app.start();
+			const result = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=1"
+			))[0];
+			assert(result.body.references_in_a);
+			assert.equal(result.body.references_in_a.length, 1);
+			const result2 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=2"
+			))[0];
+			assert(result2.body.references_in_a);
+			assert.equal(result2.body.references_in_a.length, 2);
+		});
+	});
+
+	it("updates the cached value when a new reference is created", async () => {
+		await with_stopped_app(async ({ app, rest_api }) => {
+			await create_referencing_collections(app, "with_reverse" && true);
+			await app.start();
+			await create_resources(app);
+			const result2 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=2"
+			))[0];
+			assert(result2.body.references_in_a instanceof Array);
+			assert.equal(result2.body.references_in_a.length, 2);
+		});
+	});
+
+	it("updates the cached value when an old reference is deleted", async () =>
+		with_reverse(async ({ app, rest_api }) => {
+			const result2 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=2"
+			))[0];
+			const referencing_id = result2.body.references_in_a[0];
+			await rest_api.delete(`/api/v1/collections/A/${referencing_id}`);
+			const new_result2 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=2"
+			))[0];
+			assert.equal(new_result2.body.references_in_a.length, 1);
+		}));
+
+	it("updates the cached value when an old reference is edited to a new one", async () =>
+		with_reverse(async ({ app, rest_api }) => {
+			const result1 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=1"
+			))[0];
+			const result2 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=2"
+			))[0];
+			const referencing_id = result2.body.references_in_a[0];
+
+			await rest_api.patch(`/api/v1/collections/A/${referencing_id}`, {
+				reference_to_b: result1.id,
+			});
+			const new_result2 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=2"
+			))[0];
+			assert.equal(new_result2.body.references_in_a.length, 1);
+			const new_result1 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=1"
+			))[0];
+			assert.equal(new_result1.body.references_in_a.length, 2);
+		}));
+
+	it("updates the cached value when an old reference is edited to an empty one", async () =>
+		with_reverse(async ({ app, rest_api }) => {
+			const result1 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=1"
+			))[0];
+			const result2 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=2"
+			))[0];
+			const referencing_id = result2.body.references_in_a[0];
+
+			await rest_api.patch(`/api/v1/collections/A/${referencing_id}`, {
+				reference_to_b: "",
+			});
+			const new_result2 = (await rest_api.get(
+				"/api/v1/collections/B?filter[number]=2"
+			))[0];
+			assert.equal(new_result2.body.references_in_a.length, 1);
+		}));
+
+	it("allows to filter by a value of the referencing resource", async () =>
+		with_reverse(async ({ app, rest_api }) => {
+			let results = await rest_api.get(
+				"/api/v1/collections/B?filter[references_in_a][pairity]=non-existant"
+			);
+			assert.equal(results.length, 0);
+			results = await rest_api.get(
+				"/api/v1/collections/B?filter[references_in_a][pairity]=odd"
+			);
+			assert.equal(results.length, 3);
+			results = await rest_api.get(
+				"/api/v1/collections/B?filter[references_in_a][pairity]=even&filter[number]=3"
+			);
+			assert.equal(results.length, 1);
+		}));
+
+	it("allows to display the full body of the referencing resources", async () =>
+		with_reverse(async ({ app, rest_api }) => {
+			let results = await rest_api.get(
+				"/api/v1/collections/B?format[references_in_a]=expand"
+			);
+			assert(results[0].body.references_in_a[0].body);
+		}));
+});
diff --git a/lib/app/base-chips/field-types/single_reference.subtest.js b/lib/app/base-chips/field-types/single_reference.subtest.js
--- a/lib/app/base-chips/field-types/single_reference.subtest.js
+++ b/lib/app/base-chips/field-types/single_reference.subtest.js
@@ -2,7 +2,7 @@
 const locreq = require("locreq")(__dirname);
 const axios = require("axios");
 const { create_resource_as } = locreq("test_utils");
-const with_test_app = locreq("test_utils/with-test-app.js");
+const { with_running_app } = locreq("test_utils/with-test-app.js");
 
 describe("single_reference", () => {
 	async function create_referencing_collections(app) {
@@ -28,7 +28,7 @@
 	}
 
 	it("should not allow a value that is not an existing id", async () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			await create_referencing_collections(app);
 			return axios
 				.post(`${base_url}/api/v1/collections/A`, {
@@ -46,7 +46,7 @@
 		}));
 
 	it("should allow a value that exists in B", async () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			create_referencing_collections(app);
 			const b_id = (await axios.post(`${base_url}/api/v1/collections/B`, {
 				number: 1,
@@ -57,7 +57,7 @@
 		}));
 
 	it("should not allow a value that exists in B but does not meet the filter criteria", async () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			create_referencing_collections(app);
 			const b_id = (await axios.post(`${base_url}/api/v1/collections/B`, {
 				number: 0,
@@ -79,7 +79,7 @@
 		}));
 
 	it("should allow a value that exists in B but does not meet the filter criteria", async () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			create_referencing_collections(app);
 			const b_id = (await axios.post(`${base_url}/api/v1/collections/B`, {
 				number: 1,
diff --git a/lib/app/base-chips/special_filters/IsReferencedByResourcesMatching.subtest.js b/lib/app/base-chips/special_filters/IsReferencedByResourcesMatching.subtest.js
--- a/lib/app/base-chips/special_filters/IsReferencedByResourcesMatching.subtest.js
+++ b/lib/app/base-chips/special_filters/IsReferencedByResourcesMatching.subtest.js
@@ -5,7 +5,7 @@
 
 const { create_resource_as } = locreq("test_utils");
 const IsReferencedByResourcesMatching = require("./IsReferencedByResourcesMatching");
-const with_test_app = locreq("test_utils/with-test-app.js");
+const { with_running_app } = locreq("test_utils/with-test-app.js");
 
 describe("IsReferencedByResourcesMatching", () => {
 	async function setup(app) {
@@ -86,7 +86,7 @@
 	}
 
 	it("returns only users with role matching `allowed_values`", () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			await setup(app);
 			return axios
 				.get(`${base_url}/api/v1/collections/users/@staff`)
diff --git a/lib/app/base-chips/special_filters/matches.subtest.js b/lib/app/base-chips/special_filters/matches.subtest.js
--- a/lib/app/base-chips/special_filters/matches.subtest.js
+++ b/lib/app/base-chips/special_filters/matches.subtest.js
@@ -5,7 +5,7 @@
 
 const { create_resource_as } = locreq("test_utils");
 const matches = require("./matches");
-const with_test_app = locreq("test_utils/with-test-app.js");
+const { with_running_app } = locreq("test_utils/with-test-app.js");
 
 describe("Matches", () => {
 	async function setup(app) {
@@ -35,7 +35,7 @@
 	}
 
 	it("returns only positive numbers when using @positive filter", () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			await setup(app);
 			return axios
 				.get(
@@ -50,7 +50,7 @@
 		}));
 
 	it("returns empty array when using both @positive and @negative filters", () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			await setup(app);
 			return axios
 				.get(`${base_url}/api/v1/collections/numbers/@positive/@negative`)
diff --git a/lib/app/load-base-chips.js b/lib/app/load-base-chips.js
--- a/lib/app/load-base-chips.js
+++ b/lib/app/load-base-chips.js
@@ -45,6 +45,7 @@
 	"value-existing-in-collection",
 	"value-not-existing-in-collection",
 	"secret-token",
+	"reverse-single-reference",
 ]);
 
 BaseChips.set(CalculatedFieldType, ["map-reduce", "aggregate", "custom"]);
diff --git a/lib/app/metadata.js b/lib/app/metadata.js
new file mode 100644
--- /dev/null
+++ b/lib/app/metadata.js
@@ -0,0 +1,25 @@
+const COLLECTION_NAME = "_metadata";
+
+module.exports = app => ({
+	db_collection_name: COLLECTION_NAME,
+	async get(key) {
+		const matches = await app.Datastore.find(COLLECTION_NAME, { key });
+		if (matches.length) {
+			return matches[0].value;
+		} else {
+			undefined;
+		}
+	},
+	async set(key, value) {
+		const matches = await app.Datastore.find(COLLECTION_NAME, { key });
+		if (matches.length) {
+			await app.Datastore.update(
+				COLLECTION_NAME,
+				{ key: key },
+				{ $set: { value: value } }
+			);
+		} else {
+			await app.Datastore.insert(COLLECTION_NAME, { key, value });
+		}
+	},
+});
diff --git a/lib/chip-types/collection.js b/lib/chip-types/collection.js
--- a/lib/chip-types/collection.js
+++ b/lib/chip-types/collection.js
@@ -73,8 +73,13 @@
 Collection.type_name = "collection";
 
 Collection.pure = {
-	add_field: function(app, field_type, fields, field_declaration) {
-		const field_object = new Field(app, field_declaration, field_type);
+	add_field: function(app, field_type, fields, field_declaration, collection) {
+		const field_object = new Field(
+			app,
+			field_declaration,
+			field_type,
+			collection
+		);
 		const field_name = field_object.name;
 		if (!fields[field_name]) {
 			fields[field_name] = field_object;
@@ -521,7 +526,7 @@
 
 Collection.prototype = {
 	add_field(field_declaration) {
-		return pure.add_field(this.app, this, this.fields, field_declaration);
+		return pure.add_field(this.app, this, this.fields, field_declaration, this);
 	},
 	add_fields(field_declarations_array) {
 		return pure.add_fields(
diff --git a/lib/chip-types/field-type-default-methods.js b/lib/chip-types/field-type-default-methods.js
--- a/lib/chip-types/field-type-default-methods.js
+++ b/lib/chip-types/field-type-default-methods.js
@@ -5,67 +5,73 @@
 const FieldTypeDescription = require("../data-structures/field-type-description.js");
 
 const default_methods = {
-	has_index: function(params){
+	init: function() {
+		return null;
+	},
+	has_index: function(params) {
 		return false;
 	},
-	is_proper_value: function(context, params, new_value, old_value){
+	is_proper_value: function(context, params, new_value, old_value) {
 		return Promise.resolve();
 	},
-	format: function(context, params, decoded_value, format_params){
+	format: function(context, params, decoded_value, format_params) {
 		return decoded_value;
 	},
-	encode: function(context, params, value_in_code){
+	encode: function(context, params, value_in_code) {
 		return value_in_code;
 	},
-	get_description: function(context, params){
+	get_description: function(context, params) {
 		return new FieldTypeDescription(this.name);
 	},
-	decode: function(context, params, value_in_database){
+	decode: function(context, params, value_in_database) {
 		return value_in_database;
 	},
-	filter_to_query: function(context, params, query){
-		return Promise.resolve(this.encode(context, params, query))
-		.then(function(encoded_value){
+	filter_to_query: function(context, params, query) {
+		return Promise.resolve(this.encode(context, params, query)).then(function(
+			encoded_value
+		) {
 			return {
 				$eq: encoded_value,
 			};
 		});
 	},
-	full_text_search_enabled: function(){
+	full_text_search_enabled: function() {
 		return false;
 	},
-	get_aggregation_stages: function(context, params, field_name, query_params){
+	get_aggregation_stages: function(context, params, field_name, query_params) {
 		const self = this;
-		if(!query_params || !query_params.filter) return Promise.resolve([]);
+		if (!query_params || !query_params.filter) return Promise.resolve([]);
 		const expanded_filter = expandHash(query_params.filter);
 		let field_filter = expanded_filter[field_name];
-		if(field_filter && field_filter.length === 1 && field_filter[0] instanceof Array){
+		if (
+			field_filter &&
+			field_filter.length === 1 &&
+			field_filter[0] instanceof Array
+		) {
 			field_filter = field_filter[0]; // to fix an edge case where instead of array of values the array is wrapped within another array
 		}
-		if(!(field_name in expanded_filter)){
+		if (!(field_name in expanded_filter)) {
 			return Promise.resolve([]);
 		}
-		if(field_name in expanded_filter && field_filter === undefined)
-			return Promise.resolve([{$match: {[`body.${field_name}`]: {$exists: false}}}]);
+		if (field_name in expanded_filter && field_filter === undefined)
+			return Promise.resolve([
+				{ $match: { [`body.${field_name}`]: { $exists: false } } },
+			]);
 		let new_filter = null;
-		if(field_filter instanceof Array){
+		if (field_filter instanceof Array) {
 			new_filter = Promise.all(
-				field_filter.map(function(element){
+				field_filter.map(function(element) {
 					return self.encode(context, params, element);
 				})
-			)
-			.then((filters)=> {
-				return {$in: filters};
+			).then(filters => {
+				return { $in: filters };
 			});
-		}else{
+		} else {
 			new_filter = self.filter_to_query(context, params, field_filter);
 		}
-		return new_filter
-			.then(function(filter){
-				return [
-					{$match: {[`body.${field_name}`]: filter}},
-				];
-			});
+		return new_filter.then(function(filter) {
+			return [{ $match: { [`body.${field_name}`]: filter } }];
+		});
 	},
 };
 
diff --git a/lib/chip-types/field.js b/lib/chip-types/field.js
--- a/lib/chip-types/field.js
+++ b/lib/chip-types/field.js
@@ -3,27 +3,29 @@
 const default_methods = require("./field-type-default-methods.js");
 const FieldType = locreq("lib/chip-types/field-type.js");
 
-function Field (app, declaration){
-
+function Field(app, declaration, collection) {
 	this.name = declaration.name;
 	this.declaration = declaration;
 	this.type = new FieldType(app, declaration.type);
 	this.required = declaration.required || false;
 	this.params = declaration.params || {};
+	this.type.init(collection, declaration.name, this.params);
 
 	const self = this;
 
-	for (const method_name in default_methods){
-		this[method_name] = (function(method_name){
-			return function(){
-				const arguments_array = Object.keys(arguments).map((key)=>arguments[key]);
+	for (const method_name in default_methods) {
+		this[method_name] = (function(method_name) {
+			return function() {
+				const arguments_array = Object.keys(arguments).map(
+					key => arguments[key]
+				);
 				arguments_array.splice(1, 0, self.params);
 				return self.type[method_name].apply(self.type, arguments_array);
 			};
 		})(method_name);
 	}
 
-	this.get_specification = function(){
+	this.get_specification = function() {
 		return {
 			name: this.name,
 			type: this.type,
@@ -31,14 +33,17 @@
 		};
 	};
 
-	this.get_aggregation_stages = function(context, query_params){
+	this.get_aggregation_stages = function(context, query_params) {
 		const self = this;
 		return Promise.resolve(
-			self.type.get_aggregation_stages(context, self.params, self.name, query_params)
+			self.type.get_aggregation_stages(
+				context,
+				self.params,
+				self.name,
+				query_params
+			)
 		);
-
 	};
-
 }
 
 module.exports = Field;
diff --git a/lib/email/templates/simple.test.js b/lib/email/templates/simple.test.js
--- a/lib/email/templates/simple.test.js
+++ b/lib/email/templates/simple.test.js
@@ -1,10 +1,10 @@
 const locreq = require("locreq")(__dirname);
 const assert = require("assert");
-const with_test_app = locreq("test_utils/with-test-app.js");
+const { with_running_app } = locreq("test_utils/with-test-app.js");
 
 describe("simpleTemplate", () => {
 	it("sends an email", async () =>
-		with_test_app(async ({ app, mail_api }) => {
+		with_running_app(async ({ app, mail_api }) => {
 			const message = await app.EmailTemplates.Simple(app, {
 				to: "test@example.com",
 				subject: "Congratulations!",
diff --git a/lib/http/routes/confirm-password-reset.test.js b/lib/http/routes/confirm-password-reset.test.js
--- a/lib/http/routes/confirm-password-reset.test.js
+++ b/lib/http/routes/confirm-password-reset.test.js
@@ -1,11 +1,11 @@
 const locreq = require("locreq")(__dirname);
 const axios = require("axios");
 const assert = require("assert");
-const with_test_app = locreq("test_utils/with-test-app.js");
+const { with_running_app } = locreq("test_utils/with-test-app.js");
 
 describe("confirm-password-reset", () => {
 	it("displays an html form", async () =>
-		with_test_app(async ({ app, base_url }) => {
+		with_running_app(async ({ app, base_url }) => {
 			const response = await axios.get(
 				`${base_url}/confirm-password-reset?token=kupcia&email=dupcia`
 			);
diff --git a/lib/http/routes/finalize-password-reset.test.js b/lib/http/routes/finalize-password-reset.test.js
--- a/lib/http/routes/finalize-password-reset.test.js
+++ b/lib/http/routes/finalize-password-reset.test.js
@@ -3,7 +3,7 @@
 const axios = require("axios");
 const tough = require("tough-cookie");
 const { promise_timeout, assert_throws_async } = locreq("test_utils");
-const with_test_app = locreq("test_utils/with-test-app.js");
+const { with_running_app } = locreq("test_utils/with-test-app.js");
 
 describe("finalize password reset", () => {
 	async function create_a_user(app) {
@@ -20,7 +20,7 @@
 	}
 
 	it("allows to change a password (entire flow)", async () =>
-		with_test_app(async ({ app, base_url, mail_api }) => {
+		with_running_app(async ({ app, base_url, mail_api }) => {
 			await create_a_user(app);
 			const cookieJar = new tough.CookieJar();
 			const options = {
diff --git a/lib/http/routes/finalize-registration-intent.test.js b/lib/http/routes/finalize-registration-intent.test.js
--- a/lib/http/routes/finalize-registration-intent.test.js
+++ b/lib/http/routes/finalize-registration-intent.test.js
@@ -3,11 +3,11 @@
 const axios = require("axios");
 const tough = require("tough-cookie");
 const { promise_timeout, assert_throws_async } = locreq("test_utils");
-const with_test_app = locreq("test_utils/with-test-app.js");
+const { with_running_app } = locreq("test_utils/with-test-app.js");
 
 describe("finalize registration", () => {
 	it("allows to register an account (entire flow)", async () =>
-		with_test_app(async ({ app, base_url, mail_api }) => {
+		with_running_app(async ({ app, base_url, mail_api }) => {
 			const cookieJar = new tough.CookieJar();
 			const options = {
 				jar: cookieJar,
diff --git a/test_utils/with-test-app.js b/test_utils/with-test-app.js
--- a/test_utils/with-test-app.js
+++ b/test_utils/with-test-app.js
@@ -1,7 +1,12 @@
 const locreq = require("locreq")(__dirname);
 const axios = require("axios");
 
-module.exports = async function with_test_app(fn) {
+module.exports = {
+	with_stopped_app: with_test_app.bind(global, "auto_start" && false),
+	with_running_app: with_test_app.bind(global, "auto_start" && true),
+};
+
+async function with_test_app(auto_start, fn) {
 	let app = null;
 	const port = 8888;
 	const base_url = `http://localhost:${port}`;
@@ -41,15 +46,26 @@
 		}
 	);
 
-	app.on("stop", async () =>
-		Promise.all(
-			app.ChipManager.get_all_collections().map(collection_name =>
-				app.Datastore.remove(collection_name, {}, "just_one" && false)
-			)
-		)
-	);
+	let clear_database_on_stop = true;
+
+	app.on("stop", async () => {
+		if (clear_database_on_stop) {
+			await Promise.all(
+				app.ChipManager.get_all_collections().map(collection_name =>
+					app.Datastore.remove(collection_name, {}, "just_one" && false)
+				)
+			);
+			await app.Datastore.remove(
+				app.Metadata.db_collection_name,
+				{},
+				"just_one" && false
+			);
+		}
+	});
 
-	await app.start();
+	if (auto_start) {
+		await app.start();
+	}
 
 	try {
 		await axios.delete(`${smtp_api_url}/messages`);
@@ -64,11 +80,19 @@
 				get_message_by_id: async id =>
 					(await axios.get(`${smtp_api_url}/messages/${id}.html`)).data,
 			},
+			dont_clear_database_on_stop: () => (clear_database_on_stop = false),
+			rest_api: {
+				get: async url => (await axios.get(`${base_url}${url}`)).data,
+				delete: async url => (await axios.delete(`${base_url}${url}`)).data,
+				patch: async (url, data) =>
+					(await axios.patch(`${base_url}${url}`, data)).data,
+			},
 		});
-
-		return await app.stop();
 	} catch (e) {
-		await app.stop();
 		throw e;
+	} finally {
+		if (app.status !== "stopped") {
+			await app.stop();
+		}
 	}
-};
+}