lbry-desktop/ui/index.jsx

348 lines
10 KiB
React
Raw Normal View History

import 'babel-polyfill';
2020-02-14 19:58:09 +01:00
import * as Sentry from '@sentry/browser';
import ErrorBoundary from 'component/errorBoundary';
import App from 'component/app';
import SnackBar from 'component/snackBar';
2019-02-22 06:01:59 +01:00
// @if TARGET='app'
2019-03-28 17:53:13 +01:00
import SplashScreen from 'component/splash';
import * as ACTIONS from 'constants/action_types';
import { changeZoomFactor } from 'util/zoomWindow';
2019-02-22 06:01:59 +01:00
// @endif
2019-07-29 16:34:20 +02:00
import { ipcRenderer, remote, shell } from 'electron';
import moment from 'moment';
import * as MODALS from 'constants/modal_types';
2019-08-27 16:43:42 +02:00
import React, { Fragment, useState, useEffect } from 'react';
import ReactDOM from 'react-dom';
import { Provider } from 'react-redux';
2021-09-27 05:06:34 +02:00
import { doLbryReady, doAutoUpdate, doOpenModal, doHideModal, doToggle3PAnalytics } from 'redux/actions/app';
2020-07-27 22:04:12 +02:00
import { Lbry, isURIValid, apiCall } from 'lbry-redux';
import { setSearchApi } from 'redux/actions/search';
import { doSetLanguage, doFetchLanguage, doUpdateIsNightAsync } from 'redux/actions/settings';
import { Lbryio, doBlackListedOutpointsSubscribe, doFilteredOutpointsSubscribe } from 'lbryinc';
import rewards from 'rewards';
2019-07-23 10:05:51 +02:00
import { store, persistor, history } from 'store';
2017-12-21 23:09:30 +01:00
import app from './app';
import doLogWarningConsoleMessage from './logWarningConsoleMessage';
2019-04-18 21:10:46 +02:00
import { ConnectedRouter, push } from 'connected-react-router';
2019-12-03 18:55:29 +01:00
import { formatLbryUrlForWeb, formatInAppUrl } from 'util/url';
2019-07-23 10:05:51 +02:00
import { PersistGate } from 'redux-persist/integration/react';
2019-10-02 20:20:25 +02:00
import analytics from 'analytics';
import { doToast } from 'redux/actions/notifications';
2021-09-27 05:06:34 +02:00
import keycloak from 'util/keycloak';
2019-04-18 21:10:46 +02:00
2021-09-27 05:06:34 +02:00
import { getAuthToken, setAuthToken, doAuthTokenRefresh, getTokens, deleteAuthToken } from 'util/saved-passwords';
import { X_LBRY_AUTH_TOKEN } from 'constants/token';
import { LBRY_WEB_API, DEFAULT_LANGUAGE, LBRY_API_URL, LBRY_WEB_PUBLISH_API, URL as SITE_URL } from 'config';
// Import 3rd-party styles before ours for the current way we are code-splitting.
import 'scss/third-party.scss';
2019-04-18 21:40:53 +02:00
// Import our app styles
// If a style is not necessary for the initial page load, it should be removed from `all.scss`
// and loaded dynamically in the component that consumes it
2021-07-28 22:10:19 +02:00
// @if TARGET='app'
2019-04-18 21:40:53 +02:00
import 'scss/all.scss';
2021-07-28 22:10:19 +02:00
// @endif
2021-07-29 17:11:13 +02:00
// @if TARGET='web'
import 'web/theme';
// @endif
2019-09-30 22:11:45 +02:00
// @if TARGET='web'
// These overrides can't live in web/ because they need to use the same instance of `Lbry`
import apiPublishCallViaWeb from 'web/setup/publish';
2020-02-05 04:46:00 +01:00
// Sentry error logging setup
// Will only work if you have a SENTRY_AUTH_TOKEN env
// We still add code in analytics.js to send the error to sentry manually
// If it's caught by componentDidCatch in component/errorBoundary, it will not bubble up to this error reporter
2020-02-14 19:58:09 +01:00
if (process.env.NODE_ENV === 'production') {
Sentry.init({
dsn: 'https://1f3c88e2e4b341328a638e138a60fb73@sentry.lbry.tech/2',
whitelistUrls: [/\/public\/ui.js/],
});
}
2020-02-05 04:46:00 +01:00
2020-03-24 19:43:51 +01:00
if (process.env.SDK_API_URL) {
console.warn('SDK_API_URL env var is deprecated. Use SDK_API_HOST instead'); // @eslint-disable-line
2020-03-24 19:43:51 +01:00
}
2020-05-21 17:38:28 +02:00
let sdkAPIHost = process.env.SDK_API_HOST || process.env.SDK_API_URL;
2021-07-29 17:49:01 +02:00
sdkAPIHost = LBRY_WEB_API;
2020-05-21 17:38:28 +02:00
2020-03-24 19:43:51 +01:00
export const SDK_API_PATH = `${sdkAPIHost}/api/v1`;
2021-07-29 18:11:49 +02:00
const proxyURL = `${SDK_API_PATH}/proxy`;
const publishURL = LBRY_WEB_PUBLISH_API; // || `${SDK_API_PATH}/proxy`;
2020-03-24 19:43:51 +01:00
Lbry.setDaemonConnectionString(proxyURL);
Lbry.setOverride(
'publish',
(params) =>
new Promise((resolve, reject) => {
apiPublishCallViaWeb(
2019-11-15 20:42:31 +01:00
apiCall,
2021-07-29 18:11:49 +02:00
publishURL,
Lbry.getApiRequestHeaders() && Object.keys(Lbry.getApiRequestHeaders()).includes(X_LBRY_AUTH_TOKEN)
? Lbry.getApiRequestHeaders()[X_LBRY_AUTH_TOKEN]
: '',
'publish',
params,
resolve,
reject
);
})
);
2019-09-30 22:11:45 +02:00
// @endif
2019-10-22 22:42:13 +02:00
const startTime = Date.now();
analytics.startupEvent();
2021-09-27 05:06:34 +02:00
const isDev = process.env.NODE_ENV !== 'production';
2019-10-22 22:42:13 +02:00
2019-02-22 06:01:59 +01:00
// @if TARGET='app'
const { autoUpdater } = remote.require('electron-updater');
2018-02-24 01:24:00 +01:00
autoUpdater.logger = remote.require('electron-log');
2019-02-22 06:01:59 +01:00
// @endif
2017-12-10 09:06:30 +01:00
2020-08-25 18:08:41 +02:00
if (LBRY_API_URL) {
Lbryio.setLocalApi(LBRY_API_URL);
2019-01-08 03:46:33 +01:00
}
2019-01-29 17:46:56 +01:00
if (process.env.SEARCH_API_URL) {
setSearchApi(process.env.SEARCH_API_URL);
}
2021-09-27 05:06:34 +02:00
if (getTokens().auth_token) {
doAuthTokenRefresh();
}
2018-09-24 05:44:42 +02:00
// We need to override Lbryio for getting/setting the authToken
2019-10-13 19:41:51 +02:00
// We interact with ipcRenderer to get the auth key from a users keyring
// We keep a local variable for authToken because `ipcRenderer.send` does not
// contain a response, so there is no way to know when it's been set
Lbryio.setOverride('setAuthToken', (authToken) => {
2021-09-27 05:06:34 +02:00
setAuthToken(authToken); // set the cookie to auth_token=
2020-09-04 17:02:30 +02:00
return authToken;
});
2021-09-27 05:06:34 +02:00
Lbryio.setOverride('deleteAuthToken', () => deleteAuthToken());
Lbryio.setOverride(
'getTokens',
() =>
new Promise((resolve) => {
resolve(getTokens());
})
);
2018-09-24 05:44:42 +02:00
Lbryio.setOverride(
'getAuthToken',
() =>
new Promise((resolve) => {
2021-09-27 05:06:34 +02:00
resolve(getAuthToken());
2018-09-24 05:44:42 +02:00
})
);
rewards.setCallback('claimFirstRewardSuccess', () => {
app.store.dispatch(doOpenModal(MODALS.FIRST_REWARD));
});
rewards.setCallback('claimRewardSuccess', (reward) => {
if (reward && reward.type === rewards.TYPE_REWARD_CODE) {
app.store.dispatch(doHideModal());
}
});
2019-02-22 06:01:59 +01:00
// @if TARGET='app'
ipcRenderer.on('open-uri-requested', (event, url, newSession) => {
function handleError() {
app.store.dispatch(
doToast({
message: __('Invalid LBRY URL requested'),
})
);
}
const path = url.slice('lbry://'.length);
if (path.startsWith('?')) {
2019-12-03 18:55:29 +01:00
const redirectUrl = formatInAppUrl(path);
return app.store.dispatch(push(redirectUrl));
2017-06-08 02:56:52 +02:00
}
if (isURIValid(url)) {
const formattedUrl = formatLbryUrlForWeb(url);
analytics.openUrlEvent(formattedUrl);
return app.store.dispatch(push(formattedUrl));
}
// If nothing redirected before here the url must be messed up
handleError();
});
2019-11-05 19:54:58 +01:00
ipcRenderer.on('language-set', (event, language) => {
app.store.dispatch(doSetLanguage(language));
});
ipcRenderer.on('open-menu', (event, uri) => {
if (uri && uri.startsWith('/help')) {
2019-04-18 21:10:46 +02:00
app.store.dispatch(push('/$/help'));
}
});
ipcRenderer.on('zoom-window', (event, action) => {
changeZoomFactor(action);
});
const { dock } = remote.app;
ipcRenderer.on('window-is-focused', () => {
if (!dock) return;
app.store.dispatch({ type: ACTIONS.WINDOW_FOCUSED });
dock.setBadge('');
});
ipcRenderer.on('devtools-is-opened', () => {
doLogWarningConsoleMessage();
});
2019-05-26 08:18:47 +02:00
// Force exit mode for html5 fullscreen api
// See: https://github.com/electron/electron/issues/18188
remote.getCurrentWindow().on('leave-full-screen', (event) => {
2019-05-26 08:18:47 +02:00
document.webkitExitFullscreen();
});
document.addEventListener('click', (event) => {
let { target } = event;
2018-11-07 23:44:38 +01:00
2017-06-08 02:56:52 +02:00
while (target && target !== document) {
if (target.matches('a[href^="http"]') || target.matches('a[href^="mailto"]')) {
2017-06-08 02:56:52 +02:00
event.preventDefault();
shell.openExternal(target.href);
return;
}
target = target.parentNode;
}
2017-05-21 18:15:41 +02:00
});
2020-08-21 17:49:13 +02:00
// @endif
document.addEventListener('dragover', (event) => {
2020-08-21 17:49:13 +02:00
event.preventDefault();
});
document.addEventListener('drop', (event) => {
2020-08-21 17:49:13 +02:00
event.preventDefault();
});
2017-05-21 18:15:41 +02:00
2019-07-23 10:05:51 +02:00
function AppWrapper() {
2019-11-18 19:30:15 +01:00
// Splash screen and sdk setup not needed on web
const [readyToLaunch, setReadyToLaunch] = useState(IS_WEB);
2019-11-18 19:30:15 +01:00
const [persistDone, setPersistDone] = useState(false);
2021-09-27 05:06:34 +02:00
const [keycloakReady, setKeycloakReady] = useState(false);
2018-10-14 19:47:18 +02:00
2021-09-27 05:06:34 +02:00
// init?
2019-07-23 10:05:51 +02:00
useEffect(() => {
2019-07-29 16:34:20 +02:00
// @if TARGET='app'
2019-07-23 10:05:51 +02:00
moment.locale(remote.app.getLocale());
autoUpdater.on('error', (error) => {
2019-07-23 10:05:51 +02:00
console.error(error.message); // eslint-disable-line no-console
});
2018-04-24 20:17:11 +02:00
if (['win32', 'darwin'].includes(process.platform) || !!process.env.APPIMAGE) {
2019-07-23 10:05:51 +02:00
autoUpdater.on('update-available', () => {
console.log('Update available'); // eslint-disable-line no-console
});
autoUpdater.on('update-not-available', () => {
console.log('Update not available'); // eslint-disable-line no-console
});
autoUpdater.on('update-downloaded', () => {
console.log('Update downloaded'); // eslint-disable-line no-console
app.store.dispatch(doAutoUpdate());
});
}
2019-07-29 16:34:20 +02:00
// @endif
2019-07-23 10:05:51 +02:00
}, []);
2019-02-28 00:50:00 +01:00
2021-09-27 05:06:34 +02:00
function initKeycloak() {
2021-09-27 06:52:08 +02:00
console.dir(keycloak);
keycloak
.init({
onLoad: 'check-sso',
2021-09-27 05:06:34 +02:00
silentCheckSsoFallback: false,
2021-09-27 06:52:08 +02:00
silentCheckSsoRedirectUri: window.location.origin + '/silent-check-sso.html',
redirectUri: isDev ? 'http://localhost:9090/' : `${SITE_URL}/`,
})
.then(function (authenticated) {
setKeycloakReady(true);
console.log('INIT: ', authenticated ? 'Authenticated' : 'Not Authenticated');
})
.catch(function () {
console.log('INIT: FAILED');
});
2021-09-27 05:06:34 +02:00
}
useEffect(() => {
2021-09-27 06:52:08 +02:00
console.log('KCR RENDER', keycloakReady);
2021-09-27 05:06:34 +02:00
if (!keycloakReady) {
initKeycloak();
}
}, [keycloakReady]);
// initKeycloak();
useEffect(() => {
if (persistDone) {
app.store.dispatch(doToggle3PAnalytics(null, true));
}
}, [persistDone]);
2021-09-27 05:06:34 +02:00
/**
* We have assured we have the latest browser persist,
* that daemon has started up,
* and we have checked with keycloak for a token
*/
2019-07-23 10:05:51 +02:00
useEffect(() => {
2021-09-27 06:52:08 +02:00
if (readyToLaunch && persistDone && keycloakReady) {
// keycloak ready
if (DEFAULT_LANGUAGE) {
app.store.dispatch(doFetchLanguage(DEFAULT_LANGUAGE));
}
2019-07-23 10:05:51 +02:00
app.store.dispatch(doUpdateIsNightAsync());
2021-09-27 05:06:34 +02:00
app.store.dispatch(doLbryReady());
2019-07-23 10:05:51 +02:00
app.store.dispatch(doBlackListedOutpointsSubscribe());
app.store.dispatch(doFilteredOutpointsSubscribe());
2019-10-02 20:20:25 +02:00
const appReadyTime = Date.now();
const timeToStart = appReadyTime - startTime;
analytics.readyEvent(timeToStart);
2019-07-23 10:05:51 +02:00
}
2021-09-27 05:06:34 +02:00
}, [readyToLaunch, persistDone, keycloakReady]);
useEffect(() => {
console.dir(keycloak);
}, [keycloak]);
2017-06-06 06:21:55 +02:00
2019-07-23 10:05:51 +02:00
return (
<Provider store={store}>
2019-11-18 19:30:15 +01:00
<PersistGate
persistor={persistor}
onBeforeLift={() => setPersistDone(true)}
loading={<div className="main--launching" />}
>
2019-08-27 16:43:42 +02:00
<Fragment>
2019-07-23 10:05:51 +02:00
{readyToLaunch ? (
<ConnectedRouter history={history}>
<ErrorBoundary>
<App />
<SnackBar />
</ErrorBoundary>
</ConnectedRouter>
) : (
2019-12-11 21:27:03 +01:00
<Fragment>
<SplashScreen onReadyToLaunch={() => setReadyToLaunch(true)} />
2019-12-12 21:18:13 +01:00
<SnackBar />
2019-12-11 21:27:03 +01:00
</Fragment>
2019-07-23 10:05:51 +02:00
)}
2019-08-27 16:43:42 +02:00
</Fragment>
2019-07-23 10:05:51 +02:00
</PersistGate>
</Provider>
);
}
2019-07-23 10:05:51 +02:00
ReactDOM.render(<AppWrapper />, document.getElementById('app'));