diff --git a/briar-android/res/values/strings.xml b/briar-android/res/values/strings.xml
index a74ff9c72136d1e088477e78162506aef16794b9..2f6f1d7d11cf359db1f7725c2b383594b9b7c46c 100644
--- a/briar-android/res/values/strings.xml
+++ b/briar-android/res/values/strings.xml
@@ -15,6 +15,8 @@
 	<string name="passwords_do_not_match">Passwords do not match</string>
 	<string name="enter_password">Enter your password:</string>
 	<string name="try_again">Wrong password, try again:</string>
+	<string name="startup_failed_notification_title">Briar could not start up</string>
+	<string name="startup_failed_notification_text">You may need to reinstall Briar.</string>
 	<string name="expiry_warning">This software has expired.\nPlease install a newer version.</string>
 	<string name="contact_list_button">Contacts</string>
 	<string name="forums_button">Forums</string>
diff --git a/briar-android/src/org/briarproject/android/BriarService.java b/briar-android/src/org/briarproject/android/BriarService.java
index 2fc9b01724fb54774e474770fa9138708080b194..c56714bdf571808cdf37e2f01bbf50c77d177344 100644
--- a/briar-android/src/org/briarproject/android/BriarService.java
+++ b/briar-android/src/org/briarproject/android/BriarService.java
@@ -1,6 +1,7 @@
 package org.briarproject.android;
 
 import static android.content.Intent.FLAG_ACTIVITY_CLEAR_TOP;
+import static android.content.Intent.FLAG_ACTIVITY_NEW_TASK;
 import static android.content.Intent.FLAG_ACTIVITY_SINGLE_TOP;
 import static java.util.logging.Level.INFO;
 
@@ -13,6 +14,7 @@ import org.briarproject.R;
 import org.briarproject.api.android.AndroidExecutor;
 import org.briarproject.api.db.DatabaseConfig;
 import org.briarproject.api.lifecycle.LifecycleManager;
+
 import roboguice.service.RoboService;
 import android.app.PendingIntent;
 import android.content.ComponentName;
@@ -30,11 +32,11 @@ public class BriarService extends RoboService {
 	private final Binder binder = new BriarBinder();
 
 	@Inject private DatabaseConfig databaseConfig;
-	private boolean started = false;
 
 	// Fields that are accessed from background threads must be volatile
 	@Inject private volatile LifecycleManager lifecycleManager;
 	@Inject private volatile AndroidExecutor androidExecutor;
+	private volatile boolean started = false;
 
 	@Override
 	public void onCreate() {
@@ -51,21 +53,31 @@ public class BriarService extends RoboService {
 		b.setContentTitle(getText(R.string.notification_title));
 		b.setContentText(getText(R.string.notification_text));
 		b.setWhen(0); // Don't show the time
+		b.setOngoing(true);
 		// Touch the notification to show the home screen
 		Intent i = new Intent(this, HomeScreenActivity.class);
-		i.setFlags(FLAG_ACTIVITY_CLEAR_TOP | FLAG_ACTIVITY_SINGLE_TOP);
-		PendingIntent pi = PendingIntent.getActivity(this, 0, i, 0);
-		b.setContentIntent(pi);
-		b.setOngoing(true);
+		i.setFlags(FLAG_ACTIVITY_NEW_TASK | FLAG_ACTIVITY_CLEAR_TOP |
+				FLAG_ACTIVITY_SINGLE_TOP);
+		b.setContentIntent(PendingIntent.getActivity(this, 0, i, 0));
 		startForeground(1, b.build());
 		// Start the services in a background thread
 		new Thread() {
 			@Override
 			public void run() {
-				lifecycleManager.startServices();
+				if(lifecycleManager.startServices()) {
+					started = true;
+				} else {
+					if(LOG.isLoggable(INFO)) LOG.info("Startup failed");
+					Intent i = new Intent(BriarService.this,
+							HomeScreenActivity.class);
+					i.setFlags(FLAG_ACTIVITY_NEW_TASK |
+							FLAG_ACTIVITY_CLEAR_TOP);
+					i.putExtra("briar.STARTUP_FAILED", true);
+					startActivity(i);
+					stopSelf();
+				}
 			}
 		}.start();
-		started = true;
 	}
 
 	@Override
@@ -84,11 +96,11 @@ public class BriarService extends RoboService {
 		super.onDestroy();
 		if(LOG.isLoggable(INFO)) LOG.info("Destroyed");
 		// Stop the services in a background thread
-		if(started) new Thread() {
+		new Thread() {
 			@Override
 			public void run() {
 				androidExecutor.shutdown();
-				lifecycleManager.stopServices();
+				if(started) lifecycleManager.stopServices();
 			}
 		}.start();
 	}
diff --git a/briar-android/src/org/briarproject/android/HomeScreenActivity.java b/briar-android/src/org/briarproject/android/HomeScreenActivity.java
index 7024486e5b50e74dc10155856b6b28c540f83d24..488dce28bea07ee9c626722f0afaa6adede02b0c 100644
--- a/briar-android/src/org/briarproject/android/HomeScreenActivity.java
+++ b/briar-android/src/org/briarproject/android/HomeScreenActivity.java
@@ -1,5 +1,6 @@
 package org.briarproject.android;
 
+import static android.content.Intent.FLAG_ACTIVITY_NEW_TASK;
 import static android.text.InputType.TYPE_CLASS_TEXT;
 import static android.text.InputType.TYPE_TEXT_VARIATION_PASSWORD;
 import static android.view.Gravity.CENTER;
@@ -41,10 +42,15 @@ import org.briarproject.api.lifecycle.LifecycleManager;
 import org.briarproject.util.StringUtils;
 
 import roboguice.activity.RoboActivity;
+import android.app.Notification;
+import android.app.NotificationManager;
+import android.app.PendingIntent;
+import android.content.Context;
 import android.content.Intent;
 import android.content.SharedPreferences;
 import android.os.Bundle;
 import android.os.IBinder;
+import android.support.v4.app.NotificationCompat;
 import android.text.Editable;
 import android.view.KeyEvent;
 import android.view.View;
@@ -90,9 +96,17 @@ public class HomeScreenActivity extends RoboActivity {
 	@Override
 	public void onCreate(Bundle state) {
 		super.onCreate(state);
+		if(LOG.isLoggable(INFO)) LOG.info("Created");
 		Intent i = getIntent();
-		long handle = i.getLongExtra("org.briarproject.LOCAL_AUTHOR_HANDLE", -1);
-		if(handle != -1) {
+		boolean failed = i.getBooleanExtra("briar.STARTUP_FAILED", false);
+		long handle = i.getLongExtra("briar.LOCAL_AUTHOR_HANDLE", -1);
+		if(failed) {
+			// LifecycleManager failed to start all necessary services
+			showStartupFailureNotification();
+			finish();
+			if(LOG.isLoggable(INFO)) LOG.info("Exiting");
+			System.exit(0);
+		} else if(handle != -1) {
 			// The activity was launched from the setup wizard
 			if(System.currentTimeMillis() < EXPIRY_DATE) {
 				showSpinner();
@@ -118,13 +132,30 @@ public class HomeScreenActivity extends RoboActivity {
 		}
 	}
 
+	private void showStartupFailureNotification() {
+		NotificationCompat.Builder b = new NotificationCompat.Builder(this);
+		b.setSmallIcon(android.R.drawable.stat_notify_error);
+		b.setContentTitle(getText(R.string.startup_failed_notification_title));
+		b.setContentText(getText(R.string.startup_failed_notification_text));
+		// Touch the notification to relaunch the app
+		Intent i = new Intent(this, HomeScreenActivity.class);
+		i.setFlags(FLAG_ACTIVITY_NEW_TASK);
+		b.setContentIntent(PendingIntent.getActivity(this, 0, i, 0));
+		Notification n = b.build();
+		Object o = getSystemService(Context.NOTIFICATION_SERVICE);
+		NotificationManager nm = (NotificationManager) o;
+		nm.notify(0, n);
+	}
+
 	private void showSpinner() {
 		LinearLayout layout = new LinearLayout(this);
 		layout.setLayoutParams(MATCH_MATCH);
 		layout.setGravity(CENTER);
+
 		ProgressBar progress = new ProgressBar(this);
 		progress.setIndeterminate(true);
 		layout.addView(progress);
+
 		setContentView(layout);
 	}
 
diff --git a/briar-android/src/org/briarproject/android/SetupActivity.java b/briar-android/src/org/briarproject/android/SetupActivity.java
index 5aef39a828c2c68476620b9207db96cd7a6922b4..bac901a07fbed9063dbd5d0b2f7516a682a9a291 100644
--- a/briar-android/src/org/briarproject/android/SetupActivity.java
+++ b/briar-android/src/org/briarproject/android/SetupActivity.java
@@ -247,7 +247,7 @@ public class SetupActivity extends RoboActivity implements OnClickListener {
 			public void run() {
 				Intent i = new Intent(SetupActivity.this,
 						HomeScreenActivity.class);
-				i.putExtra("org.briarproject.LOCAL_AUTHOR_HANDLE", handle);
+				i.putExtra("briar.LOCAL_AUTHOR_HANDLE", handle);
 				i.setFlags(FLAG_ACTIVITY_NEW_TASK);
 				startActivity(i);
 				finish();
diff --git a/briar-android/src/org/briarproject/android/contact/ContactListActivity.java b/briar-android/src/org/briarproject/android/contact/ContactListActivity.java
index b6852526fd025025da44a15ea00045f829d265e4..98478a8f806bd493cefd2f0b60ae0fb0f17786a6 100644
--- a/briar-android/src/org/briarproject/android/contact/ContactListActivity.java
+++ b/briar-android/src/org/briarproject/android/contact/ContactListActivity.java
@@ -212,10 +212,10 @@ ConnectionListener {
 		GroupId inbox = item.getInboxGroupId();
 		AuthorId localAuthorId = item.getContact().getLocalAuthorId();
 		Intent i = new Intent(this, ConversationActivity.class);
-		i.putExtra("org.briarproject.CONTACT_ID", contactId.getInt());
-		i.putExtra("org.briarproject.CONTACT_NAME", contactName);
-		i.putExtra("org.briarproject.GROUP_ID", inbox.getBytes());
-		i.putExtra("org.briarproject.LOCAL_AUTHOR_ID", localAuthorId.getBytes());
+		i.putExtra("briar.CONTACT_ID", contactId.getInt());
+		i.putExtra("briar.CONTACT_NAME", contactName);
+		i.putExtra("briar.GROUP_ID", inbox.getBytes());
+		i.putExtra("briar.LOCAL_AUTHOR_ID", localAuthorId.getBytes());
 		startActivity(i);
 	}
 
diff --git a/briar-android/src/org/briarproject/android/contact/ConversationActivity.java b/briar-android/src/org/briarproject/android/contact/ConversationActivity.java
index 08bf828b1a77375c33e11fb143244c4e633f3cf3..cb854dedf6d57f9d43de85a8e584f0504849c465 100644
--- a/briar-android/src/org/briarproject/android/contact/ConversationActivity.java
+++ b/briar-android/src/org/briarproject/android/contact/ConversationActivity.java
@@ -68,16 +68,16 @@ implements EventListener, OnClickListener, OnItemClickListener {
 		super.onCreate(state);
 
 		Intent i = getIntent();
-		int id = i.getIntExtra("org.briarproject.CONTACT_ID", -1);
+		int id = i.getIntExtra("briar.CONTACT_ID", -1);
 		if(id == -1) throw new IllegalStateException();
 		contactId = new ContactId(id);
-		contactName = i.getStringExtra("org.briarproject.CONTACT_NAME");
+		contactName = i.getStringExtra("briar.CONTACT_NAME");
 		if(contactName == null) throw new IllegalStateException();
 		setTitle(contactName);
-		byte[] b = i.getByteArrayExtra("org.briarproject.GROUP_ID");
+		byte[] b = i.getByteArrayExtra("briar.GROUP_ID");
 		if(b == null) throw new IllegalStateException();
 		groupId = new GroupId(b);
-		b = i.getByteArrayExtra("org.briarproject.LOCAL_AUTHOR_ID");
+		b = i.getByteArrayExtra("briar.LOCAL_AUTHOR_ID");
 		if(b == null) throw new IllegalStateException();
 		localAuthorId = new AuthorId(b);
 
@@ -221,9 +221,9 @@ implements EventListener, OnClickListener, OnItemClickListener {
 
 	public void onClick(View view) {
 		Intent i = new Intent(this, WritePrivateMessageActivity.class);
-		i.putExtra("org.briarproject.CONTACT_NAME", contactName);
-		i.putExtra("org.briarproject.GROUP_ID", groupId.getBytes());
-		i.putExtra("org.briarproject.LOCAL_AUTHOR_ID", localAuthorId.getBytes());
+		i.putExtra("briar.CONTACT_NAME", contactName);
+		i.putExtra("briar.GROUP_ID", groupId.getBytes());
+		i.putExtra("briar.LOCAL_AUTHOR_ID", localAuthorId.getBytes());
 		startActivity(i);
 	}
 
@@ -235,14 +235,14 @@ implements EventListener, OnClickListener, OnItemClickListener {
 	private void displayMessage(int position) {
 		MessageHeader header = adapter.getItem(position).getHeader();
 		Intent i = new Intent(this, ReadPrivateMessageActivity.class);
-		i.putExtra("org.briarproject.CONTACT_ID", contactId.getInt());
-		i.putExtra("org.briarproject.CONTACT_NAME", contactName);
-		i.putExtra("org.briarproject.GROUP_ID", groupId.getBytes());
-		i.putExtra("org.briarproject.LOCAL_AUTHOR_ID", localAuthorId.getBytes());
-		i.putExtra("org.briarproject.AUTHOR_NAME", header.getAuthor().getName());
-		i.putExtra("org.briarproject.MESSAGE_ID", header.getId().getBytes());
-		i.putExtra("org.briarproject.CONTENT_TYPE", header.getContentType());
-		i.putExtra("org.briarproject.TIMESTAMP", header.getTimestamp());
+		i.putExtra("briar.CONTACT_ID", contactId.getInt());
+		i.putExtra("briar.CONTACT_NAME", contactName);
+		i.putExtra("briar.GROUP_ID", groupId.getBytes());
+		i.putExtra("briar.LOCAL_AUTHOR_ID", localAuthorId.getBytes());
+		i.putExtra("briar.AUTHOR_NAME", header.getAuthor().getName());
+		i.putExtra("briar.MESSAGE_ID", header.getId().getBytes());
+		i.putExtra("briar.CONTENT_TYPE", header.getContentType());
+		i.putExtra("briar.TIMESTAMP", header.getTimestamp());
 		startActivityForResult(i, position);
 	}
 }
diff --git a/briar-android/src/org/briarproject/android/contact/ReadPrivateMessageActivity.java b/briar-android/src/org/briarproject/android/contact/ReadPrivateMessageActivity.java
index ee27a2d8500fd5fd1de48ff35bd90cec9590914b..932e6b49cf5dcd2f800609c5c5771ad445a553a9 100644
--- a/briar-android/src/org/briarproject/android/contact/ReadPrivateMessageActivity.java
+++ b/briar-android/src/org/briarproject/android/contact/ReadPrivateMessageActivity.java
@@ -72,30 +72,30 @@ implements OnClickListener {
 		super.onCreate(state);
 
 		Intent i = getIntent();
-		contactName = i.getStringExtra("org.briarproject.CONTACT_NAME");
+		contactName = i.getStringExtra("briar.CONTACT_NAME");
 		if(contactName == null) throw new IllegalStateException();
 		setTitle(contactName);
-		byte[] b = i.getByteArrayExtra("org.briarproject.LOCAL_AUTHOR_ID");
+		byte[] b = i.getByteArrayExtra("briar.LOCAL_AUTHOR_ID");
 		if(b == null) throw new IllegalStateException();
 		localAuthorId = new AuthorId(b);
-		String authorName = i.getStringExtra("org.briarproject.AUTHOR_NAME");
+		String authorName = i.getStringExtra("briar.AUTHOR_NAME");
 		if(authorName == null) throw new IllegalStateException();
-		b = i.getByteArrayExtra("org.briarproject.MESSAGE_ID");
+		b = i.getByteArrayExtra("briar.MESSAGE_ID");
 		if(b == null) throw new IllegalStateException();
 		messageId = new MessageId(b);
-		b = i.getByteArrayExtra("org.briarproject.GROUP_ID");
+		b = i.getByteArrayExtra("briar.GROUP_ID");
 		if(b == null) throw new IllegalStateException();
 		groupId = new GroupId(b);
-		String contentType = i.getStringExtra("org.briarproject.CONTENT_TYPE");
+		String contentType = i.getStringExtra("briar.CONTENT_TYPE");
 		if(contentType == null) throw new IllegalStateException();
-		timestamp = i.getLongExtra("org.briarproject.TIMESTAMP", -1);
+		timestamp = i.getLongExtra("briar.TIMESTAMP", -1);
 		if(timestamp == -1) throw new IllegalStateException();
 
 		if(state == null) {
 			read = false;
 			setReadInDatabase(true);
 		} else {
-			read = state.getBoolean("org.briarproject.READ");
+			read = state.getBoolean("briar.READ");
 		}
 
 		LinearLayout layout = new LinearLayout(this);
@@ -257,7 +257,7 @@ implements OnClickListener {
 	@Override
 	public void onSaveInstanceState(Bundle state) {
 		super.onSaveInstanceState(state);
-		state.putBoolean("org.briarproject.READ", read);
+		state.putBoolean("briar.READ", read);
 	}
 
 	public void onClick(View view) {
@@ -271,12 +271,12 @@ implements OnClickListener {
 			finish();
 		} else if(view == replyButton) {
 			Intent i = new Intent(this, WritePrivateMessageActivity.class);
-			i.putExtra("org.briarproject.CONTACT_NAME", contactName);
-			i.putExtra("org.briarproject.GROUP_ID", groupId.getBytes());
-			i.putExtra("org.briarproject.LOCAL_AUTHOR_ID",
+			i.putExtra("briar.CONTACT_NAME", contactName);
+			i.putExtra("briar.GROUP_ID", groupId.getBytes());
+			i.putExtra("briar.LOCAL_AUTHOR_ID",
 					localAuthorId.getBytes());
-			i.putExtra("org.briarproject.PARENT_ID", messageId.getBytes());
-			i.putExtra("org.briarproject.TIMESTAMP", timestamp);
+			i.putExtra("briar.PARENT_ID", messageId.getBytes());
+			i.putExtra("briar.TIMESTAMP", timestamp);
 			startActivity(i);
 			setResult(RESULT_REPLY);
 			finish();
diff --git a/briar-android/src/org/briarproject/android/contact/WritePrivateMessageActivity.java b/briar-android/src/org/briarproject/android/contact/WritePrivateMessageActivity.java
index 847a472ff041c42452987527b603a7f9d89c947b..1644cf5279f8a56f4eecd8f49082fcbb2e854711 100644
--- a/briar-android/src/org/briarproject/android/contact/WritePrivateMessageActivity.java
+++ b/briar-android/src/org/briarproject/android/contact/WritePrivateMessageActivity.java
@@ -78,17 +78,17 @@ implements OnClickListener {
 		super.onCreate(state);
 
 		Intent i = getIntent();
-		contactName = i.getStringExtra("org.briarproject.CONTACT_NAME");
+		contactName = i.getStringExtra("briar.CONTACT_NAME");
 		if(contactName == null) throw new IllegalStateException();
-		byte[] b = i.getByteArrayExtra("org.briarproject.GROUP_ID");
+		byte[] b = i.getByteArrayExtra("briar.GROUP_ID");
 		if(b == null) throw new IllegalStateException();
 		groupId = new GroupId(b);
-		b = i.getByteArrayExtra("org.briarproject.LOCAL_AUTHOR_ID");
+		b = i.getByteArrayExtra("briar.LOCAL_AUTHOR_ID");
 		if(b == null) throw new IllegalStateException();
 		localAuthorId = new AuthorId(b);
-		b = i.getByteArrayExtra("org.briarproject.PARENT_ID");
+		b = i.getByteArrayExtra("briar.PARENT_ID");
 		if(b != null) parentId = new MessageId(b);
-		timestamp = i.getLongExtra("org.briarproject.TIMESTAMP", -1);
+		timestamp = i.getLongExtra("briar.TIMESTAMP", -1);
 
 		LinearLayout layout = new LinearLayout(this);
 		layout.setLayoutParams(MATCH_WRAP);
diff --git a/briar-android/src/org/briarproject/android/groups/ConfigureGroupActivity.java b/briar-android/src/org/briarproject/android/groups/ConfigureGroupActivity.java
index 56b9cffc0628543ea60023383571f6ff53433aa9..91eb34ec26f11d60adcfb43971d5ca80f4be32c7 100644
--- a/briar-android/src/org/briarproject/android/groups/ConfigureGroupActivity.java
+++ b/briar-android/src/org/briarproject/android/groups/ConfigureGroupActivity.java
@@ -69,17 +69,17 @@ SelectContactsDialog.Listener {
 		super.onCreate(state);
 
 		Intent i = getIntent();
-		byte[] b = i.getByteArrayExtra("org.briarproject.GROUP_ID");
+		byte[] b = i.getByteArrayExtra("briar.GROUP_ID");
 		if(b == null) throw new IllegalStateException();
 		GroupId id = new GroupId(b);
-		String name = i.getStringExtra("org.briarproject.GROUP_NAME");
+		String name = i.getStringExtra("briar.GROUP_NAME");
 		if(name == null) throw new IllegalStateException();
 		setTitle(name);
-		b = i.getByteArrayExtra("org.briarproject.GROUP_SALT");
+		b = i.getByteArrayExtra("briar.GROUP_SALT");
 		if(b == null) throw new IllegalStateException();
 		group = new Group(id, name, b);
-		subscribed = i.getBooleanExtra("org.briarproject.SUBSCRIBED", false);
-		boolean all = i.getBooleanExtra("org.briarproject.VISIBLE_TO_ALL", false);
+		subscribed = i.getBooleanExtra("briar.SUBSCRIBED", false);
+		boolean all = i.getBooleanExtra("briar.VISIBLE_TO_ALL", false);
 
 		LinearLayout layout = new LinearLayout(this);
 		layout.setLayoutParams(MATCH_MATCH);
diff --git a/briar-android/src/org/briarproject/android/groups/GroupActivity.java b/briar-android/src/org/briarproject/android/groups/GroupActivity.java
index d37867ba2611d4a99057317d593f3f05a4e5f5ca..11ff72b1e30df43116e53cf54cf40da84b3e0c8d 100644
--- a/briar-android/src/org/briarproject/android/groups/GroupActivity.java
+++ b/briar-android/src/org/briarproject/android/groups/GroupActivity.java
@@ -67,10 +67,10 @@ OnClickListener, OnItemClickListener {
 		super.onCreate(state);
 
 		Intent i = getIntent();
-		byte[] b = i.getByteArrayExtra("org.briarproject.GROUP_ID");
+		byte[] b = i.getByteArrayExtra("briar.GROUP_ID");
 		if(b == null) throw new IllegalStateException();
 		groupId = new GroupId(b);
-		groupName = i.getStringExtra("org.briarproject.GROUP_NAME");
+		groupName = i.getStringExtra("briar.GROUP_NAME");
 		if(groupName == null) throw new IllegalStateException();
 		setTitle(groupName);
 
@@ -210,7 +210,7 @@ OnClickListener, OnItemClickListener {
 
 	public void onClick(View view) {
 		Intent i = new Intent(this, WriteGroupPostActivity.class);
-		i.putExtra("org.briarproject.GROUP_ID", groupId.getBytes());
+		i.putExtra("briar.GROUP_ID", groupId.getBytes());
 		startActivity(i);
 	}
 
@@ -222,16 +222,16 @@ OnClickListener, OnItemClickListener {
 	private void displayMessage(int position) {
 		MessageHeader item = adapter.getItem(position);
 		Intent i = new Intent(this, ReadGroupPostActivity.class);
-		i.putExtra("org.briarproject.GROUP_ID", groupId.getBytes());
-		i.putExtra("org.briarproject.GROUP_NAME", groupName);
-		i.putExtra("org.briarproject.MESSAGE_ID", item.getId().getBytes());
+		i.putExtra("briar.GROUP_ID", groupId.getBytes());
+		i.putExtra("briar.GROUP_NAME", groupName);
+		i.putExtra("briar.MESSAGE_ID", item.getId().getBytes());
 		Author author = item.getAuthor();
 		if(author != null) {
-			i.putExtra("org.briarproject.AUTHOR_ID", author.getId().getBytes());
-			i.putExtra("org.briarproject.AUTHOR_NAME", author.getName());
+			i.putExtra("briar.AUTHOR_ID", author.getId().getBytes());
+			i.putExtra("briar.AUTHOR_NAME", author.getName());
 		}
-		i.putExtra("org.briarproject.CONTENT_TYPE", item.getContentType());
-		i.putExtra("org.briarproject.TIMESTAMP", item.getTimestamp());
+		i.putExtra("briar.CONTENT_TYPE", item.getContentType());
+		i.putExtra("briar.TIMESTAMP", item.getTimestamp());
 		startActivityForResult(i, position);
 	}
 }
diff --git a/briar-android/src/org/briarproject/android/groups/GroupListActivity.java b/briar-android/src/org/briarproject/android/groups/GroupListActivity.java
index d61c4148e88c78f3d6904cbe10252aeaf02813f6..e1187e545102528a0946a88fc543143652ee2f20 100644
--- a/briar-android/src/org/briarproject/android/groups/GroupListActivity.java
+++ b/briar-android/src/org/briarproject/android/groups/GroupListActivity.java
@@ -347,8 +347,8 @@ implements EventListener, OnClickListener, OnItemClickListener {
 		} else {
 			Intent i = new Intent(this, GroupActivity.class);
 			Group g = item.getGroup();
-			i.putExtra("org.briarproject.GROUP_ID", g.getId().getBytes());
-			i.putExtra("org.briarproject.GROUP_NAME", g.getName());
+			i.putExtra("briar.GROUP_ID", g.getId().getBytes());
+			i.putExtra("briar.GROUP_NAME", g.getName());
 			startActivity(i);
 		}
 	}
diff --git a/briar-android/src/org/briarproject/android/groups/ManageGroupsActivity.java b/briar-android/src/org/briarproject/android/groups/ManageGroupsActivity.java
index dc27eec0dab011530543699dc5d44eb3668c81eb..9eca0b78a337d9380befffe319262ad127d95d0a 100644
--- a/briar-android/src/org/briarproject/android/groups/ManageGroupsActivity.java
+++ b/briar-android/src/org/briarproject/android/groups/ManageGroupsActivity.java
@@ -132,11 +132,11 @@ implements EventListener, OnItemClickListener {
 		GroupStatus s = item.getGroupStatus();
 		Group g = s.getGroup();
 		Intent i = new Intent(this, ConfigureGroupActivity.class);
-		i.putExtra("org.briarproject.GROUP_ID", g.getId().getBytes());
-		i.putExtra("org.briarproject.GROUP_NAME", g.getName());
-		i.putExtra("org.briarproject.GROUP_SALT", g.getSalt());
-		i.putExtra("org.briarproject.SUBSCRIBED", s.isSubscribed());
-		i.putExtra("org.briarproject.VISIBLE_TO_ALL", s.isVisibleToAll());
+		i.putExtra("briar.GROUP_ID", g.getId().getBytes());
+		i.putExtra("briar.GROUP_NAME", g.getName());
+		i.putExtra("briar.GROUP_SALT", g.getSalt());
+		i.putExtra("briar.SUBSCRIBED", s.isSubscribed());
+		i.putExtra("briar.VISIBLE_TO_ALL", s.isVisibleToAll());
 		startActivity(i);
 	}
 
diff --git a/briar-android/src/org/briarproject/android/groups/ReadGroupPostActivity.java b/briar-android/src/org/briarproject/android/groups/ReadGroupPostActivity.java
index 6284aab7283ae9d7082c683b03dadb12eaf3d446..83e97520dfa1561173af8151743a6318d2f6e144 100644
--- a/briar-android/src/org/briarproject/android/groups/ReadGroupPostActivity.java
+++ b/briar-android/src/org/briarproject/android/groups/ReadGroupPostActivity.java
@@ -69,26 +69,26 @@ implements OnClickListener {
 		super.onCreate(state);
 
 		Intent i = getIntent();
-		byte[] b = i.getByteArrayExtra("org.briarproject.GROUP_ID");
+		byte[] b = i.getByteArrayExtra("briar.GROUP_ID");
 		if(b == null) throw new IllegalStateException();
 		groupId = new GroupId(b);
-		String groupName = i.getStringExtra("org.briarproject.GROUP_NAME");
+		String groupName = i.getStringExtra("briar.GROUP_NAME");
 		if(groupName == null) throw new IllegalStateException();
 		setTitle(groupName);
-		b = i.getByteArrayExtra("org.briarproject.MESSAGE_ID");
+		b = i.getByteArrayExtra("briar.MESSAGE_ID");
 		if(b == null) throw new IllegalStateException();
 		messageId = new MessageId(b);
-		String contentType = i.getStringExtra("org.briarproject.CONTENT_TYPE");
+		String contentType = i.getStringExtra("briar.CONTENT_TYPE");
 		if(contentType == null) throw new IllegalStateException();
-		timestamp = i.getLongExtra("org.briarproject.TIMESTAMP", -1);
+		timestamp = i.getLongExtra("briar.TIMESTAMP", -1);
 		if(timestamp == -1) throw new IllegalStateException();
-		String authorName = i.getStringExtra("org.briarproject.AUTHOR_NAME");
+		String authorName = i.getStringExtra("briar.AUTHOR_NAME");
 
 		if(state == null) {
 			read = false;
 			setReadInDatabase(true);
 		} else {
-			read = state.getBoolean("org.briarproject.READ");
+			read = state.getBoolean("briar.READ");
 		}
 
 		LinearLayout layout = new LinearLayout(this);
@@ -255,7 +255,7 @@ implements OnClickListener {
 	@Override
 	public void onSaveInstanceState(Bundle state) {
 		super.onSaveInstanceState(state);
-		state.putBoolean("org.briarproject.READ", read);
+		state.putBoolean("briar.READ", read);
 	}
 
 	public void onClick(View view) {
@@ -269,9 +269,9 @@ implements OnClickListener {
 			finish();
 		} else if(view == replyButton) {
 			Intent i = new Intent(this, WriteGroupPostActivity.class);
-			i.putExtra("org.briarproject.GROUP_ID", groupId.getBytes());
-			i.putExtra("org.briarproject.PARENT_ID", messageId.getBytes());
-			i.putExtra("org.briarproject.TIMESTAMP", timestamp);
+			i.putExtra("briar.GROUP_ID", groupId.getBytes());
+			i.putExtra("briar.PARENT_ID", messageId.getBytes());
+			i.putExtra("briar.TIMESTAMP", timestamp);
 			startActivity(i);
 			setResult(RESULT_REPLY);
 			finish();
diff --git a/briar-android/src/org/briarproject/android/groups/WriteGroupPostActivity.java b/briar-android/src/org/briarproject/android/groups/WriteGroupPostActivity.java
index b46e488a3670267a187ef7849cc09d037680fca8..bedc77951a237909b948f7ef0eaa12e6e6466690 100644
--- a/briar-android/src/org/briarproject/android/groups/WriteGroupPostActivity.java
+++ b/briar-android/src/org/briarproject/android/groups/WriteGroupPostActivity.java
@@ -89,16 +89,16 @@ implements OnItemSelectedListener, OnClickListener {
 		super.onCreate(state);
 
 		Intent i = getIntent();
-		byte[] b = i.getByteArrayExtra("org.briarproject.GROUP_ID");
+		byte[] b = i.getByteArrayExtra("briar.GROUP_ID");
 		if(b == null) throw new IllegalStateException();
 		groupId = new GroupId(b);
 
-		b = i.getByteArrayExtra("org.briarproject.PARENT_ID");
+		b = i.getByteArrayExtra("briar.PARENT_ID");
 		if(b != null) parentId = new MessageId(b);
-		timestamp = i.getLongExtra("org.briarproject.TIMESTAMP", -1);
+		timestamp = i.getLongExtra("briar.TIMESTAMP", -1);
 
 		if(state != null) {
-			b = state.getByteArray("org.briarproject.LOCAL_AUTHOR_ID");
+			b = state.getByteArray("briar.LOCAL_AUTHOR_ID");
 			if(b != null) localAuthorId = new AuthorId(b);
 		}
 
@@ -214,7 +214,7 @@ implements OnItemSelectedListener, OnClickListener {
 		super.onSaveInstanceState(state);
 		if(localAuthorId != null) {
 			byte[] b =  localAuthorId.getBytes();
-			state.putByteArray("org.briarproject.LOCAL_AUTHOR_ID", b);
+			state.putByteArray("briar.LOCAL_AUTHOR_ID", b);
 		}
 	}
 
diff --git a/briar-android/src/org/briarproject/android/invitation/AddContactActivity.java b/briar-android/src/org/briarproject/android/invitation/AddContactActivity.java
index 6b3036d0d7f6c11e770d962ca07e967f8e577416..b767b18b28bf52e9d8060e7053ffef0f0f39c847 100644
--- a/briar-android/src/org/briarproject/android/invitation/AddContactActivity.java
+++ b/briar-android/src/org/briarproject/android/invitation/AddContactActivity.java
@@ -78,17 +78,17 @@ implements InvitationListener {
 			setView(new NetworkSetupView(this));
 		} else {
 			// Restore the activity's state
-			byte[] b = state.getByteArray("org.briarproject.LOCAL_AUTHOR_ID");
+			byte[] b = state.getByteArray("briar.LOCAL_AUTHOR_ID");
 			if(b != null) localAuthorId = new AuthorId(b);
-			taskHandle = state.getLong("org.briarproject.TASK_HANDLE", -1);
+			taskHandle = state.getLong("briar.TASK_HANDLE", -1);
 			task = referenceManager.getReference(taskHandle,
 					InvitationTask.class);
 			if(task == null) {
 				// No background task - we must be in an initial or final state
-				localInvitationCode = state.getInt("org.briarproject.LOCAL_CODE");
-				remoteInvitationCode = state.getInt("org.briarproject.REMOTE_CODE");
-				connectionFailed = state.getBoolean("org.briarproject.FAILED");
-				contactName = state.getString("org.briarproject.CONTACT_NAME");
+				localInvitationCode = state.getInt("briar.LOCAL_CODE");
+				remoteInvitationCode = state.getInt("briar.REMOTE_CODE");
+				connectionFailed = state.getBoolean("briar.FAILED");
+				contactName = state.getString("briar.CONTACT_NAME");
 				if(contactName != null) {
 					localCompared = remoteCompared = true;
 					localMatched = remoteMatched = true;
@@ -184,13 +184,13 @@ implements InvitationListener {
 		super.onSaveInstanceState(state);
 		if(localAuthorId != null) {
 			byte[] b = localAuthorId.getBytes();
-			state.putByteArray("org.briarproject.LOCAL_AUTHOR_ID", b);
+			state.putByteArray("briar.LOCAL_AUTHOR_ID", b);
 		}
-		state.putInt("org.briarproject.LOCAL_CODE", localInvitationCode);
-		state.putInt("org.briarproject.REMOTE_CODE", remoteInvitationCode);
-		state.putBoolean("org.briarproject.FAILED", connectionFailed);
-		state.putString("org.briarproject.CONTACT_NAME", contactName);
-		if(task != null) state.putLong("org.briarproject.TASK_HANDLE", taskHandle);
+		state.putInt("briar.LOCAL_CODE", localInvitationCode);
+		state.putInt("briar.REMOTE_CODE", remoteInvitationCode);
+		state.putBoolean("briar.FAILED", connectionFailed);
+		state.putString("briar.CONTACT_NAME", contactName);
+		if(task != null) state.putLong("briar.TASK_HANDLE", taskHandle);
 	}
 
 	@Override
diff --git a/briar-api/src/org/briarproject/api/db/DbSchemaException.java b/briar-api/src/org/briarproject/api/db/DbSchemaException.java
new file mode 100644
index 0000000000000000000000000000000000000000..79dc98533ea158d4ea9b36841f240f09021d145e
--- /dev/null
+++ b/briar-api/src/org/briarproject/api/db/DbSchemaException.java
@@ -0,0 +1,10 @@
+package org.briarproject.api.db;
+
+/**
+ * Thrown when the schema of the database differs from the schema expected by
+ * the code.
+ */
+public class DbSchemaException extends DbException {
+
+	private static final long serialVersionUID = -4444069279533651710L;
+}
diff --git a/briar-api/src/org/briarproject/api/lifecycle/LifecycleManager.java b/briar-api/src/org/briarproject/api/lifecycle/LifecycleManager.java
index f12e0f9627e4a04c3d3a49a2d4a83ba8829ffa69..1df494ddbc7f9490c99a9c7e584588ac8d8b97ec 100644
--- a/briar-api/src/org/briarproject/api/lifecycle/LifecycleManager.java
+++ b/briar-api/src/org/briarproject/api/lifecycle/LifecycleManager.java
@@ -13,8 +13,11 @@ public interface LifecycleManager {
 	 */
 	public void registerForShutdown(ExecutorService e);
 
-	/**  Starts any registered {@link Service}s. */
-	public void startServices();
+	/**
+	 * Starts any registered {@link Service}s and returns true if all services
+	 * started successfully.
+	 */
+	public boolean startServices();
 
 	/**
 	 * Stops any registered {@link Service}s and shuts down any registered
diff --git a/briar-core/src/org/briarproject/db/JdbcDatabase.java b/briar-core/src/org/briarproject/db/JdbcDatabase.java
index 3f63d08c1f8f450b6322b372b1674f98a7cd7e03..6f52733d756a733e5a6496c7d2fff7b90f32a4f5 100644
--- a/briar-core/src/org/briarproject/db/JdbcDatabase.java
+++ b/briar-core/src/org/briarproject/db/JdbcDatabase.java
@@ -34,6 +34,7 @@ import org.briarproject.api.TransportId;
 import org.briarproject.api.TransportProperties;
 import org.briarproject.api.db.DbClosedException;
 import org.briarproject.api.db.DbException;
+import org.briarproject.api.db.DbSchemaException;
 import org.briarproject.api.db.MessageHeader;
 import org.briarproject.api.messaging.Group;
 import org.briarproject.api.messaging.GroupId;
@@ -56,6 +57,14 @@ import org.briarproject.api.transport.TemporarySecret;
  */
 abstract class JdbcDatabase implements Database<Connection> {
 
+	private static final int SCHEMA_VERSION = 1;
+
+	private static final String CREATE_SETTINGS =
+			"CREATE TABLE settings"
+					+ " (key VARCHAR NOT NULL,"
+					+ " value VARCHAR NOT NULL,"
+					+ " PRIMARY KEY (key))";
+
 	// Locking: identity
 	// Dependents: contact, message, retention, subscription, transport, window
 	private static final String CREATE_LOCAL_AUTHORS =
@@ -356,7 +365,13 @@ abstract class JdbcDatabase implements Database<Connection> {
 		// Open the database and create the tables if necessary
 		Connection txn = startTransaction();
 		try {
-			if(!reopen) createTables(txn);
+			if(reopen) {
+				if(!checkSchemaVersion(txn))
+					throw new DbSchemaException();
+			} else {
+				createTables(txn);
+				setSchemaVersion(txn);
+			}
 			commitTransaction(txn);
 		} catch(DbException e) {
 			abortTransaction(txn);
@@ -364,10 +379,53 @@ abstract class JdbcDatabase implements Database<Connection> {
 		}
 	}
 
+	private boolean checkSchemaVersion(Connection txn) throws DbException {
+		PreparedStatement ps = null;
+		ResultSet rs = null;
+		String value;
+		try {
+			String sql = "SELECT value FROM settings WHERE key = ?";
+			ps = txn.prepareStatement(sql);
+			ps.setString(1, "schemaVersion");
+			rs = ps.executeQuery();
+			if(!rs.next()) throw new DbStateException();
+			value = rs.getString(1);
+			if(rs.next()) throw new DbStateException();
+			rs.close();
+			ps.close();
+		} catch(SQLException e) {
+			tryToClose(rs);
+			tryToClose(ps);
+			throw new DbException(e);
+		}
+		try {
+			return Integer.valueOf(value) == SCHEMA_VERSION;
+		} catch(NumberFormatException e) {
+			throw new DbException(e);
+		}
+	}
+
+	private void tryToClose(ResultSet rs) {
+		if(rs != null) try {
+			rs.close();
+		} catch(SQLException e) {
+			if(LOG.isLoggable(WARNING))LOG.log(WARNING, e.toString(), e);
+		}
+	}
+
+	private void tryToClose(Statement s) {
+		if(s != null) try {
+			s.close();
+		} catch(SQLException e) {
+			if(LOG.isLoggable(WARNING))LOG.log(WARNING, e.toString(), e);
+		}
+	}
+
 	private void createTables(Connection txn) throws DbException {
 		Statement s = null;
 		try {
 			s = txn.createStatement();
+			s.executeUpdate(insertTypeNames(CREATE_SETTINGS));
 			s.executeUpdate(insertTypeNames(CREATE_LOCAL_AUTHORS));
 			s.executeUpdate(insertTypeNames(CREATE_CONTACTS));
 			s.executeUpdate(insertTypeNames(CREATE_GROUPS));
@@ -405,19 +463,19 @@ abstract class JdbcDatabase implements Database<Connection> {
 		return s;
 	}
 
-	private void tryToClose(Statement s) {
-		if(s != null) try {
-			s.close();
-		} catch(SQLException e) {
-			if(LOG.isLoggable(WARNING))LOG.log(WARNING, e.toString(), e);
-		}
-	}
-
-	private void tryToClose(ResultSet rs) {
-		if(rs != null) try {
-			rs.close();
+	private void setSchemaVersion(Connection txn) throws DbException {
+		PreparedStatement ps = null;
+		try {
+			String sql = "INSERT INTO settings (key, value) VALUES (?, ?)";
+			ps = txn.prepareStatement(sql);
+			ps.setString(1, "schemaVersion");
+			ps.setString(2, String.valueOf(SCHEMA_VERSION));
+			int affected = ps.executeUpdate();
+			if(affected != 1) throw new DbStateException();
+			ps.close();
 		} catch(SQLException e) {
-			if(LOG.isLoggable(WARNING))LOG.log(WARNING, e.toString(), e);
+			tryToClose(ps);
+			throw new DbException(e);
 		}
 	}
 
diff --git a/briar-core/src/org/briarproject/lifecycle/LifecycleManagerImpl.java b/briar-core/src/org/briarproject/lifecycle/LifecycleManagerImpl.java
index 128191776286de926aad3ebffb52c7cdbefa3dc6..c76b397eadf42a6402fa5ae4d20800be949f7ec7 100644
--- a/briar-core/src/org/briarproject/lifecycle/LifecycleManagerImpl.java
+++ b/briar-core/src/org/briarproject/lifecycle/LifecycleManagerImpl.java
@@ -48,7 +48,7 @@ class LifecycleManagerImpl implements LifecycleManager {
 		executors.add(e);
 	}
 
-	public void startServices() {
+	public boolean startServices() {
 		try {
 			if(LOG.isLoggable(INFO)) LOG.info("Starting");
 			boolean reopened = db.open();
@@ -64,12 +64,16 @@ class LifecycleManagerImpl implements LifecycleManager {
 					if(started) LOG.info("Service started: " + name);
 					else LOG.info("Service failed to start: " + name);
 				}
+				if(!started) return false;
 			}
 			startupLatch.countDown();
+			return true;
 		} catch(DbException e) {
 			if(LOG.isLoggable(WARNING)) LOG.log(WARNING, e.toString(), e);
+			return false;
 		} catch(IOException e) {
 			if(LOG.isLoggable(WARNING)) LOG.log(WARNING, e.toString(), e);
+			return false;
 		}
 	}
 
diff --git a/briar-tests/src/org/briarproject/TestLifecycleModule.java b/briar-tests/src/org/briarproject/TestLifecycleModule.java
index 94a8811c41d246e7c28c72f9c61c61eacb0fb329..5f91d0ba6b58d58936043d8a52d5ef1fdf7be0cb 100644
--- a/briar-tests/src/org/briarproject/TestLifecycleModule.java
+++ b/briar-tests/src/org/briarproject/TestLifecycleModule.java
@@ -17,7 +17,7 @@ public class TestLifecycleModule extends AbstractModule {
 
 			public void registerForShutdown(ExecutorService e) {}
 
-			public void startServices() {}
+			public boolean startServices() { return true; }
 
 			public void stopServices() {}