diff --git a/lib/actions/upload-actions.js b/lib/actions/upload-actions.js
--- a/lib/actions/upload-actions.js
+++ b/lib/actions/upload-actions.js
@@ -46,7 +46,7 @@
     const abortHandler = callbacks && callbacks.abortHandler;
     const uploadBlob = callbacks && callbacks.uploadBlob;
 
-    const stringExtras = {};
+    const stringExtras: { [string]: string } = {};
     if (extras.height !== null && extras.height !== undefined) {
       stringExtras.height = extras.height.toString();
     }
@@ -71,8 +71,8 @@
     const response = await callServerEndpoint(
       'upload_multimedia',
       {
-        multimedia: [multimedia],
         ...stringExtras,
+        multimedia: [multimedia],
       },
       {
         onProgress,
diff --git a/lib/actions/user-actions.js b/lib/actions/user-actions.js
--- a/lib/actions/user-actions.js
+++ b/lib/actions/user-actions.js
@@ -142,7 +142,7 @@
   };
 
 function mergeUserInfos(...userInfoArrays: UserInfo[][]): UserInfo[] {
-  const merged = {};
+  const merged: { [string]: UserInfo } = {};
   for (const userInfoArray of userInfoArrays) {
     for (const userInfo of userInfoArray) {
       merged[userInfo.id] = userInfo;
diff --git a/lib/components/chat-mention-provider.react.js b/lib/components/chat-mention-provider.react.js
--- a/lib/components/chat-mention-provider.react.js
+++ b/lib/components/chat-mention-provider.react.js
@@ -77,9 +77,9 @@
   chatMentionCandidatesObj: ChatMentionCandidatesObj,
   communityThreadIDForGenesisThreads: { +[id: string]: string },
 } {
-  const result = {};
+  const result: { [string]: { [string]: ResolvedThreadInfo } } = {};
   const visitedGenesisThreads = new Set();
-  const communityThreadIDForGenesisThreads = {};
+  const communityThreadIDForGenesisThreads: { [string]: string } = {};
   for (const currentThreadID in threadInfos) {
     const currentThreadInfo = threadInfos[currentThreadID];
     const { community: currentThreadCommunity } = currentThreadInfo;
@@ -90,9 +90,9 @@
       continue;
     }
     if (!result[currentThreadCommunity]) {
-      result[currentThreadCommunity] = {
-        [currentThreadCommunity]: threadInfos[currentThreadCommunity],
-      };
+      result[currentThreadCommunity] = {};
+      result[currentThreadCommunity][currentThreadCommunity] =
+        threadInfos[currentThreadCommunity];
     }
     // Handle GENESIS community case: mentioning inside GENESIS should only
     // show chats and threads inside the top level that is below GENESIS.
@@ -210,7 +210,7 @@
   +[id: string]: SentencePrefixSearchIndex,
 } {
   return React.useMemo(() => {
-    const result = {};
+    const result: { [string]: SentencePrefixSearchIndex } = {};
     for (const communityThreadID in chatMentionCandidatesObj) {
       const searchIndex = new SentencePrefixSearchIndex();
       const searchIndexEntries = [];
diff --git a/lib/permissions/thread-permissions.js b/lib/permissions/thread-permissions.js
--- a/lib/permissions/thread-permissions.js
+++ b/lib/permissions/thread-permissions.js
@@ -1,5 +1,7 @@
 // @flow
 
+import invariant from 'invariant';
+
 import {
   parseThreadPermissionString,
   includeThreadPermissionForThreadType,
@@ -35,19 +37,24 @@
   permissions: ?ThreadPermissionsBlob,
   threadID: string,
 ): ThreadPermissionsInfo {
-  const result = {};
+  const result: { [permission: ThreadPermission]: ThreadPermissionInfo } = {};
   for (const permissionName in threadPermissions) {
     const permissionKey = threadPermissions[permissionName];
     const permission = permissionLookup(permissions, permissionKey);
-    let source = null;
+    let entry: ThreadPermissionInfo = { value: false, source: null };
     if (permission) {
-      if (permissions && permissions[permissionKey]) {
-        source = permissions[permissionKey].source;
+      const blobEntry = permissions ? permissions[permissionKey] : null;
+      if (blobEntry) {
+        invariant(
+          blobEntry.value,
+          'permissionLookup returned true but blob had false permission!',
+        );
+        entry = { value: true, source: blobEntry.source };
       } else {
-        source = threadID;
+        entry = { value: true, source: threadID };
       }
     }
-    result[permissionKey] = { value: permission, source };
+    result[permissionKey] = entry;
   }
   return result;
 }
@@ -62,7 +69,7 @@
   threadID: string,
   threadType: ThreadType,
 ): ?ThreadPermissionsBlob {
-  let permissions = {};
+  let permissions: { [permission: string]: ThreadPermissionInfo } = {};
 
   if (permissionsFromParent) {
     for (const permissionKey in permissionsFromParent) {
@@ -116,7 +123,8 @@
   if (!permissions) {
     return null;
   }
-  const permissionsForChildren = {};
+  const permissionsForChildren: { [permission: string]: ThreadPermissionInfo } =
+    {};
   for (const permissionKey in permissions) {
     const permissionValue = permissions[permissionKey];
     const parsed = parseThreadPermissionString(permissionKey);
diff --git a/lib/reducers/draft-reducer.js b/lib/reducers/draft-reducer.js
--- a/lib/reducers/draft-reducer.js
+++ b/lib/reducers/draft-reducer.js
@@ -73,7 +73,7 @@
       },
     };
   } else if (action.type === setClientDBStoreActionType) {
-    const drafts = {};
+    const drafts: { [string]: string } = {};
     for (const dbDraftInfo of action.payload.drafts) {
       drafts[dbDraftInfo.key] = dbDraftInfo.text;
     }
diff --git a/lib/reducers/entry-reducer.js b/lib/reducers/entry-reducer.js
--- a/lib/reducers/entry-reducer.js
+++ b/lib/reducers/entry-reducer.js
@@ -92,7 +92,7 @@
   newEntryInfos: $ReadOnlyArray<RawEntryInfo>,
   threadInfos: { +[id: string]: RawThreadInfo },
 ) {
-  const mergedEntryInfos = {};
+  const mergedEntryInfos: { [string]: RawEntryInfo } = {};
   let someEntryUpdated = false;
 
   for (const rawEntryInfo of newEntryInfos) {
diff --git a/lib/reducers/invite-links-reducer.js b/lib/reducers/invite-links-reducer.js
--- a/lib/reducers/invite-links-reducer.js
+++ b/lib/reducers/invite-links-reducer.js
@@ -9,7 +9,7 @@
   deleteAccountActionTypes,
   logOutActionTypes,
 } from '../actions/user-actions.js';
-import type { InviteLinksStore } from '../types/link-types.js';
+import type { InviteLinksStore, CommunityLinks } from '../types/link-types.js';
 import type { BaseAction } from '../types/redux-types.js';
 import { setNewSessionActionType } from '../utils/action-utils.js';
 
@@ -18,7 +18,7 @@
   action: BaseAction,
 ): InviteLinksStore {
   if (action.type === fetchPrimaryInviteLinkActionTypes.success) {
-    const links = {};
+    const links: { [string]: CommunityLinks } = {};
     for (const link of action.payload.links) {
       links[link.communityID] = {
         ...state.links[link.communityID],
diff --git a/lib/reducers/message-reducer.js b/lib/reducers/message-reducer.js
--- a/lib/reducers/message-reducer.js
+++ b/lib/reducers/message-reducer.js
@@ -237,7 +237,7 @@
     pendingToRealizedThreadIDsSelector(threadInfos);
 
   const messageStoreOperations: MessageStoreOperation[] = [];
-  const messages = {};
+  const messages: { [string]: RawMessageInfo } = {};
   for (const storeMessageID in messageStore.messages) {
     const message = messageStore.messages[storeMessageID];
     const newThreadID = pendingToRealizedThreadIDs.get(message.threadID);
@@ -262,9 +262,9 @@
     messageStoreOperations.push(updateMsgOperation);
   }
 
-  const threads = {};
+  const threads: { [string]: ThreadMessageInfo } = {};
   const reassignedThreadIDs = [];
-  const updatedThreads = {};
+  const updatedThreads: { [string]: ThreadMessageInfo } = {};
   const threadsToRemove = [];
   for (const threadID in messageStore.threads) {
     const threadMessageInfo = messageStore.threads[threadID];
@@ -323,7 +323,7 @@
 function mergeNewMessages(
   oldMessageStore: MessageStore,
   newMessageInfos: $ReadOnlyArray<RawMessageInfo>,
-  truncationStatus: { [threadID: string]: MessageTruncationStatus },
+  truncationStatus: { +[threadID: string]: MessageTruncationStatus },
   threadInfos: { +[threadID: string]: RawThreadInfo },
   actionType: string,
 ): MergeNewMessagesResult {
@@ -480,8 +480,8 @@
   );
   const oldMessageInfosToCombine = [];
   const threadsThatNeedMessageIDsResorted = [];
-  const local = {};
-  const updatedThreads = {};
+  const local: { [string]: LocalMessageInfo } = {};
+  const updatedThreads: { [string]: ThreadMessageInfo } = {};
   const threads = _flow(
     _mapValuesWithKeys((messageIDs: string[], threadID: string) => {
       const oldThread = updatedMessageStore.threads[threadID];
@@ -665,7 +665,7 @@
   messageStoreOperations.push(...reassignMessagesOps);
   const watchedIDs = [...threadWatcher.getWatchedIDs(), ...reassignedThreadIDs];
 
-  const filteredThreads = {};
+  const filteredThreads: { [string]: ThreadMessageInfo } = {};
   const threadsToRemoveMessagesFrom = [];
   const messageIDsToRemove = [];
   for (const threadID in reassignedMessageStore.threads) {
@@ -679,7 +679,7 @@
     }
   }
 
-  const updatedThreads = {};
+  const updatedThreads: { [string]: ThreadMessageInfo } = {};
   for (const threadID in threadInfos) {
     const threadInfo = threadInfos[threadID];
     if (
@@ -905,7 +905,7 @@
       action.type,
     );
   } else if (action.type === registerActionTypes.success) {
-    const truncationStatuses = {};
+    const truncationStatuses: { [string]: MessageTruncationStatus } = {};
     for (const messageInfo of action.payload.rawMessageInfos) {
       truncationStatuses[messageInfo.threadID] =
         messageTruncationStatus.EXHAUSTIVE;
@@ -958,7 +958,7 @@
     );
   } else if (action.type === sendEditMessageActionTypes.success) {
     const { newMessageInfos } = action.payload;
-    const truncationStatuses = {};
+    const truncationStatuses: { [string]: MessageTruncationStatus } = {};
     for (const messageInfo of newMessageInfos) {
       truncationStatuses[messageInfo.threadID] =
         messageTruncationStatus.UNCHANGED;
@@ -1221,7 +1221,7 @@
       messageStore: newMessageStore,
     };
   } else if (action.type === saveMessagesActionType) {
-    const truncationStatuses = {};
+    const truncationStatuses: { [string]: MessageTruncationStatus } = {};
     for (const messageInfo of action.payload.rawMessageInfos) {
       truncationStatuses[messageInfo.threadID] =
         messageTruncationStatus.UNCHANGED;
@@ -1249,7 +1249,7 @@
     const now = Date.now();
     const messageIDsToPrune = [];
 
-    const updatedThreads = {};
+    const updatedThreads: { [string]: ThreadMessageInfo } = {};
     for (const threadID of action.payload.threadIDs) {
       let thread = messageStore.threads[threadID];
       if (!thread) {
@@ -1608,7 +1608,7 @@
         action.payload.messageStoreThreads ?? [],
       );
 
-    const newThreads = {};
+    const newThreads: { [string]: ThreadMessageInfo } = {};
     for (const threadID in actionPayloadMessageStoreThreads) {
       newThreads[threadID] = {
         ...actionPayloadMessageStoreThreads[threadID],
@@ -1646,7 +1646,7 @@
     // When starting the app on native, we filter out any local-only multimedia
     // messages because the relevant context is no longer available
     const messageIDsToBeRemoved = [];
-    const threadsToAdd = {};
+    const threadsToAdd: { [string]: ThreadMessageInfo } = {};
     for (const id in actionPayloadMessages) {
       const message = actionPayloadMessages[id];
       const { threadID } = message;
diff --git a/lib/reducers/thread-reducer.js b/lib/reducers/thread-reducer.js
--- a/lib/reducers/thread-reducer.js
+++ b/lib/reducers/thread-reducer.js
@@ -293,7 +293,7 @@
       threadStoreOperations,
     };
   } else if (action.type === updateActivityActionTypes.success) {
-    const updatedThreadInfos = {};
+    const updatedThreadInfos: { [string]: RawThreadInfo } = {};
     for (const setToUnread of action.payload.result.unfocusedToUnread) {
       const threadInfo = state.threadInfos[setToUnread];
       if (threadInfo && !threadInfo.currentUser.unread) {
diff --git a/lib/selectors/chat-selectors.js b/lib/selectors/chat-selectors.js
--- a/lib/selectors/chat-selectors.js
+++ b/lib/selectors/chat-selectors.js
@@ -463,7 +463,7 @@
     );
 
     const renderedReactions: ReactionInfo = (() => {
-      const result = {};
+      const result: { [string]: MessageReactionInfo } = {};
 
       let messageReactsMap;
       if (originalMessageInfo.id) {
diff --git a/lib/selectors/invite-links-selectors.js b/lib/selectors/invite-links-selectors.js
--- a/lib/selectors/invite-links-selectors.js
+++ b/lib/selectors/invite-links-selectors.js
@@ -10,7 +10,7 @@
 } = createSelector(
   (state: AppState) => state.inviteLinksStore.links,
   (links: InviteLinks) => {
-    const primaryLinks = {};
+    const primaryLinks: { [string]: InviteLink } = {};
     for (const communityID in links) {
       const communityLinks = links[communityID];
       if (communityLinks.primaryLink) {
diff --git a/lib/selectors/socket-selectors.js b/lib/selectors/socket-selectors.js
--- a/lib/selectors/socket-selectors.js
+++ b/lib/selectors/socket-selectors.js
@@ -107,7 +107,7 @@
         } else if (serverRequest.type === serverRequestTypes.CHECK_STATE) {
           const query = calendarQuery(calendarActive);
 
-          const hashResults = {};
+          const hashResults: { [string]: boolean } = {};
           for (const key in serverRequest.hashesToCheck) {
             const expectedHashValue = serverRequest.hashesToCheck[key];
             let hashValue;
diff --git a/lib/selectors/thread-selectors.js b/lib/selectors/thread-selectors.js
--- a/lib/selectors/thread-selectors.js
+++ b/lib/selectors/thread-selectors.js
@@ -156,7 +156,7 @@
     onScreen: $ReadOnlyArray<ThreadInfo>,
     includeDeleted: boolean,
   ) => {
-    const allDaysWithinRange = {},
+    const allDaysWithinRange: { [string]: string[] } = {},
       startDate = dateFromString(startDateString),
       endDate = dateFromString(endDateString);
     for (
@@ -186,7 +186,7 @@
 } = createSelector(
   threadInfoSelector,
   (threadInfos: { +[id: string]: ThreadInfo }) => {
-    const result = {};
+    const result: { [string]: ThreadInfo[] } = {};
     for (const id in threadInfos) {
       const threadInfo = threadInfos[id];
       const parentThreadID = threadInfo.parentThreadID;
@@ -194,7 +194,7 @@
         continue;
       }
       if (result[parentThreadID] === undefined) {
-        result[parentThreadID] = [];
+        result[parentThreadID] = ([]: ThreadInfo[]);
       }
       result[parentThreadID].push(threadInfo);
     }
@@ -207,7 +207,7 @@
 } = createSelector(
   threadInfoSelector,
   (threadInfos: { +[id: string]: ThreadInfo }) => {
-    const result = {};
+    const result: { [string]: ThreadInfo[] } = {};
     for (const id in threadInfos) {
       const threadInfo = threadInfos[id];
       const { containingThreadID } = threadInfo;
@@ -215,7 +215,7 @@
         continue;
       }
       if (result[containingThreadID] === undefined) {
-        result[containingThreadID] = [];
+        result[containingThreadID] = ([]: ThreadInfo[]);
       }
       result[containingThreadID].push(threadInfo);
     }
@@ -398,7 +398,7 @@
   ) => {
     const pendingToRealizedThreadIDs =
       pendingToRealizedThreadIDsSelector(rawThreadInfos);
-    const result = {};
+    const result: { [string]: ThreadInfo } = {};
     for (const realizedID of pendingToRealizedThreadIDs.values()) {
       const threadInfo = threadInfos[realizedID];
       if (threadInfo && threadInfo.sourceMessageID) {
diff --git a/lib/shared/entry-utils.js b/lib/shared/entry-utils.js
--- a/lib/shared/entry-utils.js
+++ b/lib/shared/entry-utils.js
@@ -102,7 +102,7 @@
   calendarQuery: CalendarQuery,
 ): { +[id: string]: RawEntryInfo } {
   let filtered = false;
-  const filteredRawEntryInfos = {};
+  const filteredRawEntryInfos: { [string]: RawEntryInfo } = {};
   for (const id in rawEntryInfos) {
     const rawEntryInfo = rawEntryInfos[id];
     if (!rawEntryInfoWithinCalendarQuery(rawEntryInfo, calendarQuery)) {
@@ -228,7 +228,7 @@
 function serverEntryInfosObject(
   array: $ReadOnlyArray<RawEntryInfo>,
 ): RawEntryInfos {
-  const obj = {};
+  const obj: { [string]: RawEntryInfo } = {};
   for (const rawEntryInfo of array) {
     const entryInfo = serverEntryInfo(rawEntryInfo);
     if (!entryInfo) {
diff --git a/lib/shared/messages/add-members-message-spec.js b/lib/shared/messages/add-members-message-spec.js
--- a/lib/shared/messages/add-members-message-spec.js
+++ b/lib/shared/messages/add-members-message-spec.js
@@ -118,7 +118,7 @@
     threadInfo: ThreadInfo,
     params: NotificationTextsParams,
   ): Promise<NotifTexts> {
-    const addedMembersObject = {};
+    const addedMembersObject: { [string]: RelativeUserInfo } = {};
     for (const messageInfo of messageInfos) {
       invariant(
         messageInfo.type === messageTypes.ADD_MEMBERS,
diff --git a/lib/shared/messages/change-role-message-spec.js b/lib/shared/messages/change-role-message-spec.js
--- a/lib/shared/messages/change-role-message-spec.js
+++ b/lib/shared/messages/change-role-message-spec.js
@@ -152,7 +152,7 @@
     threadInfo: ThreadInfo,
     params: NotificationTextsParams,
   ): Promise<NotifTexts> {
-    const membersObject = {};
+    const membersObject: { [string]: RelativeUserInfo } = {};
     for (const messageInfo of messageInfos) {
       invariant(
         messageInfo.type === messageTypes.CHANGE_ROLE,
diff --git a/lib/shared/messages/join-thread-message-spec.js b/lib/shared/messages/join-thread-message-spec.js
--- a/lib/shared/messages/join-thread-message-spec.js
+++ b/lib/shared/messages/join-thread-message-spec.js
@@ -91,7 +91,7 @@
     messageInfos: $ReadOnlyArray<MessageInfo>,
     threadInfo: ThreadInfo,
   ): Promise<NotifTexts> {
-    const joinerArray = {};
+    const joinerArray: { [string]: RelativeUserInfo } = {};
     for (const messageInfo of messageInfos) {
       invariant(
         messageInfo.type === messageTypes.JOIN_THREAD,
diff --git a/lib/shared/messages/leave-thread-message-spec.js b/lib/shared/messages/leave-thread-message-spec.js
--- a/lib/shared/messages/leave-thread-message-spec.js
+++ b/lib/shared/messages/leave-thread-message-spec.js
@@ -91,7 +91,7 @@
     messageInfos: $ReadOnlyArray<MessageInfo>,
     threadInfo: ThreadInfo,
   ): Promise<NotifTexts> {
-    const leaverBeavers = {};
+    const leaverBeavers: { [string]: RelativeUserInfo } = {};
     for (const messageInfo of messageInfos) {
       invariant(
         messageInfo.type === messageTypes.LEAVE_THREAD,
diff --git a/lib/shared/messages/remove-members-message-spec.js b/lib/shared/messages/remove-members-message-spec.js
--- a/lib/shared/messages/remove-members-message-spec.js
+++ b/lib/shared/messages/remove-members-message-spec.js
@@ -118,7 +118,7 @@
     threadInfo: ThreadInfo,
     params: NotificationTextsParams,
   ): Promise<NotifTexts> {
-    const removedMembersObject = {};
+    const removedMembersObject: { [string]: RelativeUserInfo } = {};
     for (const messageInfo of messageInfos) {
       invariant(
         messageInfo.type === messageTypes.REMOVE_MEMBERS,
diff --git a/lib/shared/thread-utils.js b/lib/shared/thread-utils.js
--- a/lib/shared/thread-utils.js
+++ b/lib/shared/thread-utils.js
@@ -378,7 +378,7 @@
     pinnedCount: 0,
   };
 
-  const userInfos = {};
+  const userInfos: { [string]: UserInfo } = {};
   for (const member of members) {
     const { id, username } = member;
     userInfos[id] = { id, username };
@@ -1559,7 +1559,7 @@
   threadInfo: ThreadInfo,
 ): RoleAndMemberCount {
   return React.useMemo(() => {
-    const roleIDsToNames = {};
+    const roleIDsToNames: { [string]: string } = {};
 
     Object.keys(threadInfo.roles).forEach(roleID => {
       roleIDsToNames[roleID] = threadInfo.roles[roleID].name;
@@ -1596,7 +1596,8 @@
   threadInfo: ThreadInfo,
 ): RoleUserSurfacedPermissions {
   return React.useMemo(() => {
-    const roleNamesToPermissions = {};
+    const roleNamesToPermissions: { [string]: Set<UserSurfacedPermission> } =
+      {};
 
     Object.keys(threadInfo.roles).forEach(roleID => {
       const roleName = threadInfo.roles[roleID].name;
diff --git a/lib/socket/socket.react.js b/lib/socket/socket.react.js
--- a/lib/socket/socket.react.js
+++ b/lib/socket/socket.react.js
@@ -163,7 +163,7 @@
     });
 
     const socket = this.props.openSocket();
-    const openObject = {};
+    const openObject: { initializeMessageSent?: true } = {};
     socket.onopen = () => {
       if (this.socket === socket) {
         this.initializeSocket();
@@ -546,7 +546,6 @@
       'inflightRequests falsey inside sendInitialMessage',
     );
     const messageID = this.nextClientMessageID++;
-    const promises = {};
 
     const shouldSendInitialPlatformDetails = !_isEqual(
       this.props.lastCommunicatedPlatformDetails,
@@ -560,13 +559,14 @@
       });
     }
 
+    let activityUpdatePromise;
     const { queuedActivityUpdates } = this.props.connection;
     if (queuedActivityUpdates.length > 0) {
       clientResponses.push({
         type: serverRequestTypes.INITIAL_ACTIVITY_UPDATES,
         activityUpdates: queuedActivityUpdates,
       });
-      promises.activityUpdateMessage = inflightRequests.fetchResponse(
+      activityUpdatePromise = inflightRequests.fetchResponse(
         messageID,
         serverSocketMessageTypes.ACTIVITY_UPDATE_RESPONSE,
       );
@@ -586,14 +586,17 @@
     this.initializedWithUserState = this.props.preRequestUserState;
     this.sendMessage(initialMessage);
 
-    promises.stateSyncMessage = inflightRequests.fetchResponse(
+    const stateSyncPromise = inflightRequests.fetchResponse(
       messageID,
       serverSocketMessageTypes.STATE_SYNC,
     );
 
-    const { stateSyncMessage, activityUpdateMessage } = await promiseAll(
-      promises,
-    );
+    // https://flow.org/try/#1N4Igxg9gdgZglgcxALlAJwKYEMwBcD6aArlLnALYYrgA2WAzvXGCADQgYAeOBARgJ74AJhhhYiNXClzEM7DFCLl602e0hQhcMtCw18ufgAcqyGUTkhyWIyaEHjp85cMn8CIljRCVZ2QF91CBFqBn4oMAACGBI8OGhIozQIcjh6DABBGhoAHgAVZEjgSIBtAGsMfkL6GTgoBABdQtTODCFI-wA+AAoAHShIyLqjIlxCgBIAJWwhAHkoGn58ztZ+gEpCgAVk1PSc8dneACsAWRt81kjXDAgYSPGsAHcsbU7Oov7BjRrEnbSMeiRAC8pQaAG5PpFvrhIhV+ICQYcjhg8AA6OH0brDUZrCEDaIQNCRbo0DAwuDAyIABjBQ0iOVhlXoqNJ9VwAAtaXAANTctYffFfaA-OGUjElODgyFCqA-JIpf6U7G4cqVKWC34K9LMkb0dndeW7DC4yH+SHQyKYegSXAIyJPF4w7ZajCovQ0A1-bUm-EWgQAOSwlAmSLORguV0ct3uDtelOA-jxgxghOJpPJlJpdIZGJZCgQHK5vP5wGlUOFMNFIPFkqTg0iAaDGFV-AalKtNvoEvVgzN+MwuCIaAGjcoeL7-TCEWisR0A3SmgAklBtHA9CcAfQsAgMN0S5D01cAbhnUa63A7n0QOyMNkIL0QKial5bQB1bT6h9lB9rfca3DHqeiogkB6SolaEA0AAbruzg+r2-SQrGMKGv8WQesUAE1KBGAdD6-hsCAMFoEw0DUFBVKogATAAjAAzKiVIgP4QA
+    // $FlowFixMe fixed in Flow 0.214
+    const { stateSyncMessage, activityUpdateMessage } = await promiseAll({
+      activityUpdateMessage: activityUpdatePromise,
+      stateSyncMessage: stateSyncPromise,
+    });
 
     if (shouldSendInitialPlatformDetails) {
       this.props.dispatch({
diff --git a/lib/utils/conversion-utils.js b/lib/utils/conversion-utils.js
--- a/lib/utils/conversion-utils.js
+++ b/lib/utils/conversion-utils.js
@@ -88,7 +88,7 @@
   }
   if (validator.meta.kind === 'interface' && typeof input === 'object') {
     const recastValidator: TInterface<typeof input> = (validator: any);
-    const result = {};
+    const result: { [string]: mixed } = {};
     for (const key in input) {
       const innerValidator = recastValidator.meta.props[key];
       result[key] = convertObject(
diff --git a/lib/utils/cookie-utils.js b/lib/utils/cookie-utils.js
--- a/lib/utils/cookie-utils.js
+++ b/lib/utils/cookie-utils.js
@@ -3,7 +3,7 @@
 function parseCookies(header: string): { +[string]: string } {
   const values = header.split(';').map(v => v.split('='));
 
-  const cookies = {};
+  const cookies: { [string]: string } = {};
   for (const [key, value] of values) {
     cookies[decodeURIComponent(key.trim())] = decodeURIComponent(value.trim());
   }
diff --git a/lib/utils/entity-helpers.js b/lib/utils/entity-helpers.js
--- a/lib/utils/entity-helpers.js
+++ b/lib/utils/entity-helpers.js
@@ -91,7 +91,7 @@
     options,
   );
   return React.useMemo(() => {
-    const obj = {};
+    const obj: { [string]: ResolvedThreadInfo } = {};
     for (const resolvedThreadInfo of resolvedThreadInfosArray) {
       obj[resolvedThreadInfo.id] = resolvedThreadInfo;
     }
diff --git a/lib/utils/promises.js b/lib/utils/promises.js
--- a/lib/utils/promises.js
+++ b/lib/utils/promises.js
@@ -2,7 +2,7 @@
 
 type Promisable<T> = Promise<T> | T;
 
-async function promiseAll<T: { [key: string]: Promisable<*> }>(
+async function promiseAll<T: { +[key: string]: Promisable<mixed> }>(
   input: T,
 ): Promise<$ObjMap<T, typeof $await>> {
   const promises = [];
@@ -13,7 +13,7 @@
     promises.push(promise);
   }
   const results = await Promise.all(promises);
-  const byName = {};
+  const byName: {[string]: mixed} = {};
   for (let i = 0; i < keys.length; i++) {
     const key = keys[i];
     byName[key] = results[i];
diff --git a/lib/utils/url-utils.js b/lib/utils/url-utils.js
--- a/lib/utils/url-utils.js
+++ b/lib/utils/url-utils.js
@@ -9,21 +9,23 @@
   pendingThreadIDRegex,
 } from './validation-utils.js';
 
-export type URLInfo = {
-  +year?: number,
-  +month?: number, // 1-indexed
-  +verify?: string,
-  +calendar?: boolean,
-  +chat?: boolean,
-  +thread?: string,
-  +settings?: 'account' | 'danger-zone',
-  +threadCreation?: boolean,
-  +selectedUserList?: $ReadOnlyArray<string>,
-  +inviteSecret?: string,
-  +qrCode?: boolean,
+type MutableURLInfo = {
+  year?: number,
+  month?: number, // 1-indexed
+  verify?: string,
+  calendar?: boolean,
+  chat?: boolean,
+  thread?: string,
+  settings?: 'account' | 'danger-zone',
+  threadCreation?: boolean,
+  selectedUserList?: $ReadOnlyArray<string>,
+  inviteSecret?: string,
+  qrCode?: boolean,
   ...
 };
 
+export type URLInfo = $ReadOnly<MutableURLInfo>;
+
 export const urlInfoValidator: TInterface<URLInfo> = tShape<URLInfo>({
   year: t.maybe(t.Number),
   month: t.maybe(t.Number),
@@ -76,7 +78,7 @@
   const inviteLinkMatches = inviteLinkRegex.exec(url);
   const qrCodeLoginMatches = qrCodeLoginRegex.exec(url);
 
-  const returnObj = {};
+  const returnObj: MutableURLInfo = {};
   if (yearMatches) {
     returnObj.year = parseInt(yearMatches[2], 10);
   }