mirror of
https://github.com/BlueWallet/BlueWallet.git
synced 2025-01-19 05:45:15 +01:00
445 lines
15 KiB
JavaScript
445 lines
15 KiB
JavaScript
import AsyncStorage from '@react-native-async-storage/async-storage';
|
|
import PushNotificationIOS from '@react-native-community/push-notification-ios';
|
|
import { findNodeHandle, Platform } from 'react-native';
|
|
import { getApplicationName, getSystemName, getSystemVersion, getVersion, hasGmsSync, hasHmsSync } from 'react-native-device-info';
|
|
import { requestNotifications } from 'react-native-permissions';
|
|
import PushNotification from 'react-native-push-notification';
|
|
|
|
import loc from '../loc';
|
|
import ActionSheet from '../screen/ActionSheet';
|
|
import { groundControlUri } from './constants';
|
|
|
|
const PUSH_TOKEN = 'PUSH_TOKEN';
|
|
const GROUNDCONTROL_BASE_URI = 'GROUNDCONTROL_BASE_URI';
|
|
const NOTIFICATIONS_STORAGE = 'NOTIFICATIONS_STORAGE';
|
|
const NOTIFICATIONS_NO_AND_DONT_ASK_FLAG = 'NOTIFICATIONS_NO_AND_DONT_ASK_FLAG';
|
|
let alreadyConfigured = false;
|
|
let baseURI = groundControlUri;
|
|
|
|
function Notifications(props) {
|
|
async function _setPushToken(token) {
|
|
token = JSON.stringify(token);
|
|
return AsyncStorage.setItem(PUSH_TOKEN, token);
|
|
}
|
|
|
|
Notifications.getPushToken = async () => {
|
|
try {
|
|
let token = await AsyncStorage.getItem(PUSH_TOKEN);
|
|
token = JSON.parse(token);
|
|
return token;
|
|
} catch (_) {}
|
|
return false;
|
|
};
|
|
|
|
Notifications.isNotificationsCapable = hasGmsSync() || hasHmsSync() || Platform.OS !== 'android';
|
|
|
|
/**
|
|
* Calls `configure`, which tries to obtain push token, save it, and registers all associated with
|
|
* notifications callbacks
|
|
*
|
|
* @returns {Promise<boolean>} TRUE if acquired token, FALSE if not
|
|
*/
|
|
const configureNotifications = async function () {
|
|
return new Promise(function (resolve) {
|
|
requestNotifications(['alert', 'sound', 'badge']).then(({ status, _ }) => {
|
|
if (status === 'granted') {
|
|
PushNotification.configure({
|
|
// (optional) Called when Token is generated (iOS and Android)
|
|
onRegister: async function (token) {
|
|
console.debug('TOKEN:', token);
|
|
alreadyConfigured = true;
|
|
await _setPushToken(token);
|
|
resolve(true);
|
|
},
|
|
|
|
// (required) Called when a remote is received or opened, or local notification is opened
|
|
onNotification: async function (notification) {
|
|
// since we do not know whether we:
|
|
// 1) received notification while app is in background (and storage is not decrypted so wallets are not loaded)
|
|
// 2) opening this notification right now but storage is still unencrypted
|
|
// 3) any of the above but the storage is decrypted, and app wallets are loaded
|
|
//
|
|
// ...we save notification in internal notifications queue thats gona be processed later (on unsuspend with decrypted storage)
|
|
|
|
const payload = Object.assign({}, notification, notification.data);
|
|
if (notification.data && notification.data.data) Object.assign(payload, notification.data.data);
|
|
delete payload.data;
|
|
// ^^^ weird, but sometimes payload data is not in `data` but in root level
|
|
console.debug('got push notification', payload);
|
|
|
|
await Notifications.addNotification(payload);
|
|
|
|
// (required) Called when a remote is received or opened, or local notification is opened
|
|
notification.finish(PushNotificationIOS.FetchResult.NoData);
|
|
|
|
// if user is staring at the app when he receives the notification we process it instantly
|
|
// so app refetches related wallet
|
|
if (payload.foreground) props.onProcessNotifications();
|
|
},
|
|
|
|
// (optional) Called when Registered Action is pressed and invokeApp is false, if true onNotification will be called (Android)
|
|
onAction: function (notification) {
|
|
console.debug('ACTION:', notification.action);
|
|
console.debug('NOTIFICATION:', notification);
|
|
|
|
// process the action
|
|
},
|
|
|
|
// (optional) Called when the user fails to register for remote notifications. Typically occurs when APNS is having issues, or the device is a simulator. (iOS)
|
|
onRegistrationError: function (err) {
|
|
console.error(err.message, err);
|
|
resolve(false);
|
|
},
|
|
|
|
// IOS ONLY (optional): default: all - Permissions to register.
|
|
permissions: {
|
|
alert: true,
|
|
badge: true,
|
|
sound: true,
|
|
},
|
|
|
|
// Should the initial notification be popped automatically
|
|
// default: true
|
|
popInitialNotification: true,
|
|
|
|
/**
|
|
* (optional) default: true
|
|
* - Specified if permissions (ios) and token (android and ios) will requested or not,
|
|
* - if not, you must call PushNotificationsHandler.requestPermissions() later
|
|
* - if you are not using remote notification or do not have Firebase installed, use this:
|
|
* requestPermissions: Platform.OS === 'ios'
|
|
*/
|
|
requestPermissions: true,
|
|
});
|
|
}
|
|
});
|
|
});
|
|
// …
|
|
};
|
|
|
|
Notifications.cleanUserOptOutFlag = async function () {
|
|
return AsyncStorage.removeItem(NOTIFICATIONS_NO_AND_DONT_ASK_FLAG);
|
|
};
|
|
|
|
/**
|
|
* Should be called when user is most interested in receiving push notifications.
|
|
* If we dont have a token it will show alert asking whether
|
|
* user wants to receive notifications, and if yes - will configure push notifications.
|
|
* FYI, on Android permissions are acquired when app is installed, so basically we dont need to ask,
|
|
* we can just call `configure`. On iOS its different, and calling `configure` triggers system's dialog box.
|
|
*
|
|
* @returns {Promise<boolean>} TRUE if permissions were obtained, FALSE otherwise
|
|
*/
|
|
Notifications.tryToObtainPermissions = async function (anchor) {
|
|
if (!Notifications.isNotificationsCapable) return false;
|
|
if (await Notifications.getPushToken()) {
|
|
// we already have a token, no sense asking again, just configure pushes to register callbacks and we are done
|
|
if (!alreadyConfigured) configureNotifications(); // no await so it executes in background while we return TRUE and use token
|
|
return true;
|
|
}
|
|
|
|
if (await AsyncStorage.getItem(NOTIFICATIONS_NO_AND_DONT_ASK_FLAG)) {
|
|
// user doesn't want them
|
|
return false;
|
|
}
|
|
|
|
return new Promise(function (resolve) {
|
|
const options = [loc.notifications.no_and_dont_ask, loc.notifications.ask_me_later, loc._.ok];
|
|
|
|
ActionSheet.showActionSheetWithOptions(
|
|
{
|
|
title: loc.settings.notifications,
|
|
message: loc.notifications.would_you_like_to_receive_notifications,
|
|
options,
|
|
cancelButtonIndex: 0, // Assuming 'no and don't ask' is still treated as the cancel action
|
|
anchor: anchor ? findNodeHandle(anchor.current) : undefined,
|
|
},
|
|
buttonIndex => {
|
|
switch (buttonIndex) {
|
|
case 0:
|
|
AsyncStorage.setItem(NOTIFICATIONS_NO_AND_DONT_ASK_FLAG, '1').then(() => resolve(false));
|
|
break;
|
|
case 1:
|
|
resolve(false);
|
|
break;
|
|
case 2:
|
|
configureNotifications().then(resolve);
|
|
break;
|
|
}
|
|
},
|
|
);
|
|
});
|
|
};
|
|
|
|
function _getHeaders() {
|
|
return {
|
|
'Access-Control-Allow-Origin': '*',
|
|
'Content-Type': 'application/json',
|
|
};
|
|
}
|
|
|
|
async function _sleep(ms) {
|
|
return new Promise(resolve => setTimeout(resolve, ms));
|
|
}
|
|
|
|
/**
|
|
* Submits onchain bitcoin addresses and ln invoice preimage hashes to GroundControl server, so later we could
|
|
* be notified if they were paid
|
|
*
|
|
* @param addresses {string[]}
|
|
* @param hashes {string[]}
|
|
* @param txids {string[]}
|
|
* @returns {Promise<object>} Response object from API rest call
|
|
*/
|
|
Notifications.majorTomToGroundControl = async function (addresses, hashes, txids) {
|
|
if (!Array.isArray(addresses) || !Array.isArray(hashes) || !Array.isArray(txids))
|
|
throw new Error('no addresses or hashes or txids provided');
|
|
const pushToken = await Notifications.getPushToken();
|
|
if (!pushToken || !pushToken.token || !pushToken.os) return;
|
|
|
|
const response = await fetch(`${baseURI}/majorTomToGroundControl`, {
|
|
method: 'POST',
|
|
headers: _getHeaders(),
|
|
body: JSON.stringify({
|
|
addresses,
|
|
hashes,
|
|
txids,
|
|
token: pushToken.token,
|
|
os: pushToken.os,
|
|
}),
|
|
});
|
|
|
|
return response.json();
|
|
};
|
|
|
|
/**
|
|
* The opposite of `majorTomToGroundControl` call.
|
|
*
|
|
* @param addresses {string[]}
|
|
* @param hashes {string[]}
|
|
* @param txids {string[]}
|
|
* @returns {Promise<object>} Response object from API rest call
|
|
*/
|
|
Notifications.unsubscribe = async function (addresses, hashes, txids) {
|
|
if (!Array.isArray(addresses) || !Array.isArray(hashes) || !Array.isArray(txids))
|
|
throw new Error('no addresses or hashes or txids provided');
|
|
const pushToken = await Notifications.getPushToken();
|
|
if (!pushToken || !pushToken.token || !pushToken.os) return;
|
|
|
|
const response = await fetch(`${baseURI}/unsubscribe`, {
|
|
method: 'POST',
|
|
headers: _getHeaders(),
|
|
body: JSON.stringify({
|
|
addresses,
|
|
hashes,
|
|
txids,
|
|
token: pushToken.token,
|
|
os: pushToken.os,
|
|
}),
|
|
});
|
|
|
|
console.debug('Abandoning notifications Permissions...');
|
|
PushNotification.abandonPermissions();
|
|
console.debug('Abandoned notifications Permissions...');
|
|
return response.json();
|
|
};
|
|
|
|
Notifications.isNotificationsEnabled = async function () {
|
|
const levels = await getLevels();
|
|
return !!(await Notifications.getPushToken()) && !!levels.level_all;
|
|
};
|
|
|
|
Notifications.getDefaultUri = function () {
|
|
return groundControlUri;
|
|
};
|
|
|
|
Notifications.saveUri = async function (uri) {
|
|
baseURI = uri || groundControlUri; // setting the url to use currently. if not set - use default
|
|
return AsyncStorage.setItem(GROUNDCONTROL_BASE_URI, uri);
|
|
};
|
|
|
|
Notifications.getSavedUri = async function () {
|
|
return AsyncStorage.getItem(GROUNDCONTROL_BASE_URI);
|
|
};
|
|
|
|
/**
|
|
* Validates whether the provided GroundControl URI is valid by pinging it.
|
|
*
|
|
* @param uri {string}
|
|
* @returns {Promise<boolean>} TRUE if valid, FALSE otherwise
|
|
*/
|
|
Notifications.isGroundControlUriValid = async uri => {
|
|
let response;
|
|
try {
|
|
response = await Promise.race([fetch(`${uri}/ping`, { headers: _getHeaders() }), _sleep(2000)]);
|
|
} catch (_) {}
|
|
|
|
if (!response) return false;
|
|
|
|
const json = await response.json();
|
|
return !!json.description;
|
|
};
|
|
|
|
/**
|
|
* Returns a permissions object:
|
|
* alert: boolean
|
|
* badge: boolean
|
|
* sound: boolean
|
|
*
|
|
* @returns {Promise<Object>}
|
|
*/
|
|
Notifications.checkPermissions = async function () {
|
|
return new Promise(function (resolve) {
|
|
PushNotification.checkPermissions(result => {
|
|
resolve(result);
|
|
});
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Posts to groundcontrol info whether we want to opt in or out of specific notifications level
|
|
*
|
|
* @param levelAll {Boolean}
|
|
* @returns {Promise<*>}
|
|
*/
|
|
Notifications.setLevels = async function (levelAll) {
|
|
const pushToken = await Notifications.getPushToken();
|
|
if (!pushToken || !pushToken.token || !pushToken.os) return;
|
|
|
|
try {
|
|
await fetch(`${baseURI}/setTokenConfiguration`, {
|
|
method: 'POST',
|
|
headers: _getHeaders(),
|
|
body: JSON.stringify({
|
|
level_all: !!levelAll,
|
|
token: pushToken.token,
|
|
os: pushToken.os,
|
|
}),
|
|
});
|
|
console.debug('Abandoning notifications Permissions...');
|
|
PushNotification.abandonPermissions();
|
|
console.debug('Abandoned notifications Permissions...');
|
|
} catch (_) {}
|
|
};
|
|
|
|
/**
|
|
* Queries groundcontrol for token configuration, which contains subscriptions to notification levels
|
|
*
|
|
* @returns {Promise<{}|*>}
|
|
*/
|
|
const getLevels = async function () {
|
|
const pushToken = await Notifications.getPushToken();
|
|
if (!pushToken || !pushToken.token || !pushToken.os) return;
|
|
|
|
let response;
|
|
try {
|
|
response = await Promise.race([
|
|
fetch(`${baseURI}/getTokenConfiguration`, {
|
|
method: 'POST',
|
|
headers: _getHeaders(),
|
|
body: JSON.stringify({
|
|
token: pushToken.token,
|
|
os: pushToken.os,
|
|
}),
|
|
}),
|
|
_sleep(3000),
|
|
]);
|
|
} catch (_) {}
|
|
|
|
if (!response) return {};
|
|
|
|
return await response.json();
|
|
};
|
|
|
|
Notifications.getStoredNotifications = async function () {
|
|
let notifications = [];
|
|
try {
|
|
const stringified = await AsyncStorage.getItem(NOTIFICATIONS_STORAGE);
|
|
notifications = JSON.parse(stringified);
|
|
if (!Array.isArray(notifications)) notifications = [];
|
|
} catch (_) {}
|
|
|
|
return notifications;
|
|
};
|
|
|
|
Notifications.addNotification = async function (notification) {
|
|
let notifications = [];
|
|
try {
|
|
const stringified = await AsyncStorage.getItem(NOTIFICATIONS_STORAGE);
|
|
notifications = JSON.parse(stringified);
|
|
if (!Array.isArray(notifications)) notifications = [];
|
|
} catch (_) {}
|
|
|
|
notifications.push(notification);
|
|
await AsyncStorage.setItem(NOTIFICATIONS_STORAGE, JSON.stringify(notifications));
|
|
};
|
|
|
|
const postTokenConfig = async function () {
|
|
const pushToken = await Notifications.getPushToken();
|
|
if (!pushToken || !pushToken.token || !pushToken.os) return;
|
|
|
|
try {
|
|
const lang = (await AsyncStorage.getItem('lang')) || 'en';
|
|
const appVersion = getSystemName() + ' ' + getSystemVersion() + ';' + getApplicationName() + ' ' + getVersion();
|
|
|
|
await fetch(`${baseURI}/setTokenConfiguration`, {
|
|
method: 'POST',
|
|
headers: _getHeaders(),
|
|
body: JSON.stringify({
|
|
token: pushToken.token,
|
|
os: pushToken.os,
|
|
lang,
|
|
app_version: appVersion,
|
|
}),
|
|
});
|
|
} catch (_) {}
|
|
};
|
|
|
|
Notifications.clearStoredNotifications = async function () {
|
|
try {
|
|
await AsyncStorage.setItem(NOTIFICATIONS_STORAGE, JSON.stringify([]));
|
|
} catch (_) {}
|
|
};
|
|
|
|
Notifications.getDeliveredNotifications = () => {
|
|
return new Promise(resolve => {
|
|
PushNotification.getDeliveredNotifications(notifications => resolve(notifications));
|
|
});
|
|
};
|
|
|
|
Notifications.removeDeliveredNotifications = (identifiers = []) => {
|
|
PushNotification.removeDeliveredNotifications(identifiers);
|
|
};
|
|
|
|
Notifications.setApplicationIconBadgeNumber = function (badges) {
|
|
PushNotification.setApplicationIconBadgeNumber(badges);
|
|
};
|
|
|
|
Notifications.removeAllDeliveredNotifications = () => {
|
|
PushNotification.removeAllDeliveredNotifications();
|
|
};
|
|
|
|
// on app launch (load module):
|
|
(async () => {
|
|
// first, fetching to see if app uses custom GroundControl server, not the default one
|
|
try {
|
|
const baseUriStored = await AsyncStorage.getItem(GROUNDCONTROL_BASE_URI);
|
|
if (baseUriStored) {
|
|
baseURI = baseUriStored;
|
|
}
|
|
} catch (_) {}
|
|
|
|
// every launch should clear badges:
|
|
Notifications.setApplicationIconBadgeNumber(0);
|
|
|
|
if (!(await Notifications.getPushToken())) return;
|
|
// if we previously had token that means we already acquired permission from the user and it is safe to call
|
|
// `configure` to register callbacks etc
|
|
await configureNotifications();
|
|
await postTokenConfig();
|
|
})();
|
|
return null;
|
|
}
|
|
|
|
export default Notifications;
|