lbry-desktop/src/ui/redux/actions/app.js

478 lines
12 KiB
JavaScript
Raw Normal View History

2019-02-22 06:01:59 +01:00
// @if TARGET='app'
import { execSync } from 'child_process';
import isDev from 'electron-is-dev';
import { ipcRenderer, remote } from 'electron';
2019-02-22 06:01:59 +01:00
// @endif
2019-03-05 05:46:57 +01:00
import path from 'path';
2018-11-07 23:44:38 +01:00
import * as ACTIONS from 'constants/action_types';
import * as MODALS from 'constants/modal_types';
2019-09-26 18:07:11 +02:00
import * as PAGES from 'constants/pages';
2019-08-14 05:04:08 +02:00
import {
Lbry,
doBalanceSubscribe,
doFetchFileInfosAndPublishedClaims,
doError,
makeSelectClaimForUri,
makeSelectClaimIsMine,
2019-09-26 21:31:27 +02:00
doPopulateSharedUserState,
2019-09-26 18:07:11 +02:00
doFetchChannelListMine,
2019-08-14 05:04:08 +02:00
} from 'lbry-redux';
2018-04-18 06:03:01 +02:00
import Native from 'native';
import { doFetchDaemonSettings } from 'redux/actions/settings';
2018-08-30 05:28:53 +02:00
import { doCheckSubscriptionsInit } from 'redux/actions/subscriptions';
2017-04-07 07:15:22 +02:00
import {
selectIsUpgradeSkipped,
2017-04-07 07:15:22 +02:00
selectUpdateUrl,
selectUpgradeDownloadItem,
selectUpgradeDownloadPath,
2017-07-29 01:31:10 +02:00
selectUpgradeFilename,
selectAutoUpdateDeclined,
selectRemoteVersion,
selectUpgradeTimer,
selectModal,
} from 'redux/selectors/app';
import { Lbryio, doAuthenticate, doGetSync } from 'lbryinc';
2018-09-24 05:44:42 +02:00
import { lbrySettings as config, version as appVersion } from 'package.json';
2019-04-05 00:23:12 +02:00
import { push } from 'connected-react-router';
2019-08-14 05:04:08 +02:00
import analytics from 'analytics';
2019-09-26 18:07:11 +02:00
import { deleteAuthToken } from 'util/saved-passwords';
import cookie from 'cookie';
2019-02-22 06:01:59 +01:00
// @if TARGET='app'
const { autoUpdater } = remote.require('electron-updater');
const { download } = remote.require('electron-dl');
const Fs = remote.require('fs');
2019-02-22 06:01:59 +01:00
// @endif
2017-12-13 22:36:30 +01:00
2017-11-16 22:39:10 +01:00
const CHECK_UPGRADE_INTERVAL = 10 * 60 * 1000;
2017-04-07 07:15:22 +02:00
export function doOpenModal(id, modalProps = {}) {
return {
type: ACTIONS.SHOW_MODAL,
data: {
id,
modalProps,
},
};
}
export function doHideModal() {
return {
type: ACTIONS.HIDE_MODAL,
};
}
2017-04-07 07:15:22 +02:00
export function doUpdateDownloadProgress(percent) {
return {
type: ACTIONS.UPGRADE_DOWNLOAD_PROGRESSED,
2017-04-07 07:15:22 +02:00
data: {
2017-12-13 22:36:30 +01:00
percent,
2017-06-06 23:19:12 +02:00
},
};
2017-04-07 07:15:22 +02:00
}
export function doSkipUpgrade() {
return {
type: ACTIONS.SKIP_UPGRADE,
2017-06-06 23:19:12 +02:00
};
2017-04-07 07:15:22 +02:00
}
export function doStartUpgrade() {
return (dispatch, getState) => {
2017-06-06 23:19:12 +02:00
const state = getState();
const upgradeDownloadPath = selectUpgradeDownloadPath(state);
2017-04-07 07:15:22 +02:00
ipcRenderer.send('upgrade', upgradeDownloadPath);
2017-06-06 23:19:12 +02:00
};
2017-04-07 07:15:22 +02:00
}
export function doDownloadUpgrade() {
return (dispatch, getState) => {
2019-02-22 06:01:59 +01:00
// @if TARGET='app'
2017-06-06 23:19:12 +02:00
const state = getState();
2017-04-07 07:15:22 +02:00
// Make a new directory within temp directory so the filename is guaranteed to be available
const dir = Fs.mkdtempSync(remote.app.getPath('temp') + path.sep);
const upgradeFilename = selectUpgradeFilename(state);
2017-04-07 07:15:22 +02:00
2017-12-13 22:36:30 +01:00
const options = {
2017-06-06 23:19:12 +02:00
onProgress: p => dispatch(doUpdateDownloadProgress(Math.round(p * 100))),
2017-04-07 07:15:22 +02:00
directory: dir,
};
download(remote.getCurrentWindow(), selectUpdateUrl(state), options).then(downloadItem => {
/**
* TODO: get the download path directly from the download object. It should just be
* downloadItem.getSavePath(), but the copy on the main process is being garbage collected
* too soon.
*/
2017-04-07 07:15:22 +02:00
dispatch({
type: ACTIONS.UPGRADE_DOWNLOAD_COMPLETED,
data: {
downloadItem,
path: path.join(dir, upgradeFilename),
},
});
});
2017-04-07 07:15:22 +02:00
dispatch({
type: ACTIONS.UPGRADE_DOWNLOAD_STARTED,
2017-06-06 23:19:12 +02:00
});
dispatch(doHideModal());
dispatch(doOpenModal(MODALS.DOWNLOADING));
2019-02-22 06:01:59 +01:00
// @endif
2017-06-06 23:19:12 +02:00
};
2017-04-07 07:15:22 +02:00
}
export function doDownloadUpgradeRequested() {
// This means the user requested an upgrade by clicking the "upgrade" button in the navbar.
// If on Mac and Windows, we do some new behavior for the auto-update system.
// This will probably be reorganized once we get auto-update going on Linux and remove
// the old logic.
return (dispatch, getState) => {
const state = getState();
const autoUpdateDeclined = selectAutoUpdateDeclined(state);
if (['win32', 'darwin'].includes(process.platform)) {
// electron-updater behavior
if (autoUpdateDeclined) {
// The user declined an update before, so show the "confirm" dialog
dispatch(doOpenModal(MODALS.AUTO_UPDATE_CONFIRM));
} else {
// The user was never shown the original update dialog (e.g. because they were
// watching a video). So show the inital "update downloaded" dialog.
dispatch(doOpenModal(MODALS.AUTO_UPDATE_DOWNLOADED));
}
} else {
// Old behavior for Linux
dispatch(doDownloadUpgrade());
}
};
}
export function doClearUpgradeTimer() {
return (dispatch, getState) => {
const state = getState();
if (selectUpgradeTimer(state)) {
clearInterval(selectUpgradeTimer(state));
dispatch({
type: ACTIONS.CLEAR_UPGRADE_TIMER,
});
}
};
}
export function doAutoUpdate() {
return dispatch => {
dispatch({
type: ACTIONS.AUTO_UPDATE_DOWNLOADED,
});
dispatch(doOpenModal(MODALS.AUTO_UPDATE_DOWNLOADED));
dispatch(doClearUpgradeTimer());
};
}
export function doAutoUpdateDeclined() {
return dispatch => {
dispatch(doClearUpgradeTimer());
dispatch({
type: ACTIONS.AUTO_UPDATE_DECLINED,
});
2018-02-24 01:24:00 +01:00
};
}
2017-04-07 07:15:22 +02:00
export function doCancelUpgrade() {
return (dispatch, getState) => {
2017-06-06 23:19:12 +02:00
const state = getState();
const upgradeDownloadItem = selectUpgradeDownloadItem(state);
2017-04-07 07:15:22 +02:00
if (upgradeDownloadItem) {
/*
* Right now the remote reference to the download item gets garbage collected as soon as the
* the download is over (maybe even earlier), so trying to cancel a finished download may
* throw an error.
*/
try {
upgradeDownloadItem.cancel();
} catch (err) {
2019-07-23 10:05:51 +02:00
console.error(err); // eslint-disable-line no-console
2017-04-07 07:15:22 +02:00
}
}
dispatch({ type: ACTIONS.UPGRADE_CANCELLED });
2017-06-06 23:19:12 +02:00
};
2017-04-07 07:15:22 +02:00
}
export function doCheckUpgradeAvailable() {
return (dispatch, getState) => {
2017-06-06 23:19:12 +02:00
const state = getState();
dispatch({
type: ACTIONS.CHECK_UPGRADE_START,
});
2017-04-07 07:15:22 +02:00
if (['win32', 'darwin'].includes(process.platform)) {
// On Windows and Mac, updates happen silently through
// electron-updater.
const autoUpdateDeclined = selectAutoUpdateDeclined(state);
if (!autoUpdateDeclined && !isDev) {
autoUpdater.checkForUpdates();
}
return;
}
const success = ({ remoteVersion, upgradeAvailable }) => {
dispatch({
type: ACTIONS.CHECK_UPGRADE_SUCCESS,
data: {
upgradeAvailable,
remoteVersion,
},
});
if (
upgradeAvailable &&
!selectModal(state) &&
(!selectIsUpgradeSkipped(state) || remoteVersion !== selectRemoteVersion(state))
) {
dispatch(doOpenModal(MODALS.UPGRADE));
}
};
const fail = () => {
dispatch({
type: ACTIONS.CHECK_UPGRADE_FAIL,
});
};
2018-04-18 06:03:01 +02:00
Native.getAppVersionInfo().then(success, fail);
};
}
/*
Initiate a timer that will check for an app upgrade every 10 minutes.
*/
2017-11-16 22:39:10 +01:00
export function doCheckUpgradeSubscribe() {
return dispatch => {
2019-05-07 23:38:29 +02:00
const checkUpgradeTimer = setInterval(() => dispatch(doCheckUpgradeAvailable()), CHECK_UPGRADE_INTERVAL);
dispatch({
type: ACTIONS.CHECK_UPGRADE_SUBSCRIBE,
data: { checkUpgradeTimer },
2017-04-07 07:15:22 +02:00
});
2017-06-06 23:19:12 +02:00
};
2017-04-07 07:15:22 +02:00
}
export function doCheckDaemonVersion() {
return dispatch => {
2019-02-22 06:01:59 +01:00
// @if TARGET='app'
Lbry.version().then(({ lbrynet_version: lbrynetVersion }) => {
// Avoid the incompatible daemon modal if running in dev mode
// Lets you run a different daemon than the one specified in package.json
if (config.lbrynetDaemonVersion === lbrynetVersion) {
return dispatch({
2018-05-11 01:06:41 +02:00
type: ACTIONS.DAEMON_VERSION_MATCH,
});
}
dispatch({
2018-05-11 01:06:41 +02:00
type: ACTIONS.DAEMON_VERSION_MISMATCH,
});
return dispatch(doOpenModal(MODALS.INCOMPATIBLE_DAEMON));
});
2019-02-22 06:01:59 +01:00
// @endif
// @if TARGET='web'
dispatch({
type: ACTIONS.DAEMON_VERSION_MATCH,
});
// @endif
};
}
2018-07-18 21:48:30 +02:00
export function doNotifyEncryptWallet() {
return dispatch => {
dispatch(doOpenModal(MODALS.WALLET_ENCRYPT));
2018-07-18 21:48:30 +02:00
};
}
export function doNotifyDecryptWallet() {
return dispatch => {
dispatch(doOpenModal(MODALS.WALLET_DECRYPT));
2018-07-18 21:48:30 +02:00
};
}
export function doNotifyUnlockWallet() {
return dispatch => {
dispatch(doOpenModal(MODALS.WALLET_UNLOCK));
2018-07-18 21:48:30 +02:00
};
}
export function doNotifyForgetPassword(props) {
2019-08-20 14:29:59 +02:00
return dispatch => {
dispatch(doOpenModal(MODALS.WALLET_PASSWORD_UNSAVE, props));
2019-08-20 14:29:59 +02:00
};
}
2017-04-07 07:15:22 +02:00
export function doAlertError(errorList) {
return dispatch => {
dispatch(doError(errorList));
2017-06-06 23:19:12 +02:00
};
2017-04-07 07:15:22 +02:00
}
2017-04-22 15:17:01 +02:00
export function doDaemonReady() {
return (dispatch, getState) => {
2017-11-16 22:39:10 +01:00
const state = getState();
2018-09-24 05:44:42 +02:00
dispatch(doAuthenticate(appVersion));
dispatch({ type: ACTIONS.DAEMON_READY });
2019-03-29 15:23:32 +01:00
2019-02-22 06:01:59 +01:00
// @if TARGET='app'
dispatch(doBalanceSubscribe());
2019-09-26 18:07:11 +02:00
dispatch(doFetchDaemonSettings());
2017-09-07 15:38:44 +02:00
dispatch(doFetchFileInfosAndPublishedClaims());
2017-11-16 22:39:10 +01:00
if (!selectIsUpgradeSkipped(state)) {
dispatch(doCheckUpgradeAvailable());
}
dispatch(doCheckUpgradeSubscribe());
dispatch(doCheckSubscriptionsInit());
2019-02-22 06:01:59 +01:00
// @endif
2017-06-08 02:56:52 +02:00
};
2017-04-22 15:17:01 +02:00
}
2017-05-23 09:21:21 +02:00
export function doClearCache() {
return () => {
2019-07-23 10:05:51 +02:00
// Need to update this to work with new version of redux-persist
// Leaving for now
// const reducersToClear = whiteListedReducers.filter(reducerKey => reducerKey !== 'tags');
// window.cacheStore.purge(reducersToClear);
window.localStorage.clear();
2019-07-23 10:05:51 +02:00
return window.persistor.purge();
};
}
export function doQuit() {
return () => {
2019-02-22 06:01:59 +01:00
// @if TARGET='app'
remote.app.quit();
2019-02-22 06:01:59 +01:00
// @endif
};
}
export function doQuitAnyDaemon() {
return dispatch => {
2019-02-22 06:01:59 +01:00
// @if TARGET='app'
2019-02-18 18:33:02 +01:00
Lbry.stop()
.catch(() => {
try {
if (process.platform === 'win32') {
execSync('taskkill /im lbrynet.exe /t /f');
} else {
execSync('pkill lbrynet');
}
} catch (error) {
dispatch(doAlertError(`Quitting daemon failed due to: ${error.message}`));
}
2019-02-22 06:01:59 +01:00
})
.finally(() => {
dispatch(doQuit());
2019-02-18 18:33:02 +01:00
});
2019-02-22 06:01:59 +01:00
// @endif
};
}
export function doChangeVolume(volume) {
return dispatch => {
dispatch({
type: ACTIONS.VOLUME_CHANGED,
data: {
volume,
},
});
};
}
2017-12-23 03:09:06 +01:00
export function doChangeMute(muted) {
return dispatch => {
dispatch({
type: ACTIONS.VOLUME_MUTED,
data: {
muted,
},
});
};
}
2018-11-07 23:44:38 +01:00
export function doClickCommentButton() {
return {
type: ACTIONS.ADD_COMMENT,
};
}
2017-12-23 03:09:06 +01:00
export function doConditionalAuthNavigate(newSession) {
return (dispatch, getState) => {
2017-12-23 03:09:06 +01:00
const state = getState();
const modal = selectModal(state);
2018-04-23 20:02:06 +02:00
if (newSession || (modal && modal.id !== MODALS.EMAIL_COLLECTION)) {
2019-09-26 18:07:11 +02:00
dispatch(push(`/$/${PAGES.AUTH}`));
2017-12-23 03:09:06 +01:00
}
};
}
2019-01-23 16:38:40 +01:00
export function doToggleSearchExpanded() {
return {
type: ACTIONS.TOGGLE_SEARCH_EXPANDED,
};
}
2019-08-14 05:04:08 +02:00
export function doAnalyticsView(uri, timeToStart) {
return (dispatch, getState) => {
const state = getState();
const { txid, nout, claim_id: claimId } = makeSelectClaimForUri(uri)(state);
const claimIsMine = makeSelectClaimIsMine(uri)(state);
const outpoint = `${txid}:${nout}`;
if (claimIsMine) {
2019-09-21 18:45:52 +02:00
return Promise.resolve();
2019-08-14 05:04:08 +02:00
}
return analytics.apiLogView(uri, outpoint, claimId, timeToStart);
2019-08-14 05:04:08 +02:00
};
}
2019-09-26 18:07:11 +02:00
export function doSignIn() {
return (dispatch, getState) => {
// The balance is subscribed to on launch for desktop
// @if TARGET='web'
const { auth_token: authToken } = cookie.parse(document.cookie);
Lbry.setApiHeader('X-Lbry-Auth-Token', authToken);
dispatch(doBalanceSubscribe());
dispatch(doFetchChannelListMine());
2019-10-01 06:53:33 +02:00
dispatch(doCheckSubscriptionsInit());
// @endif
2019-09-27 20:56:15 +02:00
2019-10-01 06:53:33 +02:00
// @if TARGET='app'
dispatch(doGetSync());
2019-09-26 18:07:11 +02:00
// @endif
Lbryio.call('user_settings', 'get').then(settings => {
2019-09-26 21:31:27 +02:00
dispatch(doPopulateSharedUserState(settings));
2019-09-26 18:07:11 +02:00
});
};
}
export function doSignOut() {
return dispatch => {
deleteAuthToken()
.then(window.persistor.purge)
2019-09-26 18:07:11 +02:00
.then(() => {
location.reload();
2019-09-26 18:07:11 +02:00
})
.catch(() => location.reload());
};
}