import React from 'react'; import { Text, Keyboard, StyleSheet, View, BackHandler, Image } from 'react-native'; import { connect } from 'react-redux'; import { Base64 } from 'js-base64'; import parse from 'url-parse'; import { Q } from '@nozbe/watermelondb'; import { TouchableOpacity } from 'react-native-gesture-handler'; import Orientation from 'react-native-orientation-locker'; import { StackNavigationProp } from '@react-navigation/stack'; import { Dispatch } from 'redux'; import Model from '@nozbe/watermelondb/Model'; import UserPreferences from '../../lib/userPreferences'; import EventEmitter from '../../utils/events'; import { selectServerRequest, serverRequest, serverFinishAdd as serverFinishAddAction } from '../../actions/server'; import { inviteLinksClear as inviteLinksClearAction } from '../../actions/inviteLinks'; import sharedStyles from '../Styles'; import Button from '../../containers/Button'; import OrSeparator from '../../containers/OrSeparator'; import FormContainer, { FormContainerInner } from '../../containers/FormContainer'; import I18n from '../../i18n'; import { themes } from '../../constants/colors'; import { events, logEvent } from '../../utils/log'; import { withTheme } from '../../theme'; import { BASIC_AUTH_KEY, setBasicAuth } from '../../utils/fetch'; import * as HeaderButton from '../../containers/HeaderButton'; import { showConfirmationAlert } from '../../utils/info'; import database from '../../lib/database'; import { sanitizeLikeString } from '../../lib/database/utils'; import SSLPinning from '../../utils/sslPinning'; import RocketChat from '../../lib/rocketchat'; import { isTablet } from '../../utils/deviceInfo'; import { verticalScale, moderateScale } from '../../utils/scaling'; import { withDimensions } from '../../dimensions'; import ServerInput from './ServerInput'; const styles = StyleSheet.create({ onboardingImage: { alignSelf: 'center', resizeMode: 'contain' }, title: { ...sharedStyles.textBold, letterSpacing: 0, alignSelf: 'center' }, subtitle: { ...sharedStyles.textRegular, alignSelf: 'center' }, certificatePicker: { alignItems: 'center', justifyContent: 'flex-end' }, chooseCertificateTitle: { ...sharedStyles.textRegular }, chooseCertificate: { ...sharedStyles.textSemibold }, description: { ...sharedStyles.textRegular, textAlign: 'center' }, connectButton: { marginBottom: 0 } }); export interface IServer extends Model { url: string; username: string; } interface INewServerView { navigation: StackNavigationProp; theme: string; connecting: boolean; connectServer(server: string, username?: string, fromServerHistory?: boolean): void; selectServer(server: string): void; previousServer: string; inviteLinksClear(): void; serverFinishAdd(): void; width: number; height: number; } interface IState { text: string; connectingOpen: boolean; certificate: any; serversHistory: IServer[]; } interface ISubmitParams { fromServerHistory?: boolean; username?: string; } class NewServerView extends React.Component { constructor(props: INewServerView) { super(props); if (!isTablet) { Orientation.lockToPortrait(); } this.setHeader(); this.state = { text: '', connectingOpen: false, certificate: null, serversHistory: [] }; EventEmitter.addEventListener('NewServer', this.handleNewServerEvent); BackHandler.addEventListener('hardwareBackPress', this.handleBackPress); } componentDidMount() { this.queryServerHistory(); } componentWillUnmount() { EventEmitter.removeListener('NewServer', this.handleNewServerEvent); BackHandler.removeEventListener('hardwareBackPress', this.handleBackPress); const { previousServer, serverFinishAdd } = this.props; if (previousServer) { serverFinishAdd(); } } setHeader = () => { const { previousServer, navigation } = this.props; if (previousServer) { return navigation.setOptions({ headerTitle: I18n.t('Workspaces'), headerLeft: () => }); } return navigation.setOptions({ headerShown: false }); }; handleBackPress = () => { const { navigation, previousServer } = this.props; if (navigation.isFocused() && previousServer) { this.close(); return true; } return false; }; onChangeText = (text: string) => { this.setState({ text }); this.queryServerHistory(text); }; queryServerHistory = async (text?: string) => { const db = database.servers; try { const serversHistoryCollection = db.get('servers_history'); let whereClause = [Q.where('username', Q.notEq(null)), Q.experimentalSortBy('updated_at', Q.desc), Q.experimentalTake(3)]; if (text) { const likeString = sanitizeLikeString(text); whereClause = [...whereClause, Q.where('url', Q.like(`%${likeString}%`))]; } const serversHistory = (await serversHistoryCollection.query(...whereClause).fetch()) as IServer[]; this.setState({ serversHistory }); } catch { // Do nothing } }; close = () => { const { selectServer, previousServer, inviteLinksClear } = this.props; inviteLinksClear(); selectServer(previousServer); }; handleNewServerEvent = (event: { server: string }) => { let { server } = event; if (!server) { return; } const { connectServer } = this.props; this.setState({ text: server }); server = this.completeUrl(server); connectServer(server); }; onPressServerHistory = (serverHistory: IServer) => { this.setState({ text: serverHistory.url }, () => this.submit({ fromServerHistory: true, username: serverHistory?.username })); }; submit = async ({ fromServerHistory = false, username }: ISubmitParams = {}) => { logEvent(events.NS_CONNECT_TO_WORKSPACE); const { text, certificate } = this.state; const { connectServer } = this.props; this.setState({ connectingOpen: false }); if (text) { Keyboard.dismiss(); const server = this.completeUrl(text); // Save info - SSL Pinning await UserPreferences.setStringAsync(`${RocketChat.CERTIFICATE_KEY}-${server}`, certificate); // Save info - HTTP Basic Authentication await this.basicAuth(server, text); if (fromServerHistory) { connectServer(server, username, true); } else { connectServer(server); } } }; connectOpen = () => { logEvent(events.NS_JOIN_OPEN_WORKSPACE); this.setState({ connectingOpen: true }); const { connectServer } = this.props; connectServer('https://open.rocket.chat'); }; basicAuth = async (server: string, text: string) => { try { const parsedUrl = parse(text, true); if (parsedUrl.auth.length) { const credentials = Base64.encode(parsedUrl.auth); await UserPreferences.setStringAsync(`${BASIC_AUTH_KEY}-${server}`, credentials); setBasicAuth(credentials); } } catch { // do nothing } }; chooseCertificate = async () => { try { const certificate = await SSLPinning?.pickCertificate(); this.setState({ certificate }); } catch { // Do nothing } }; completeUrl = (url: string) => { const parsedUrl = parse(url, true); if (parsedUrl.auth.length) { url = parsedUrl.origin; } url = url && url.replace(/\s/g, ''); if (/^(\w|[0-9-_]){3,}$/.test(url) && /^(htt(ps?)?)|(loca((l)?|(lh)?|(lho)?|(lhos)?|(lhost:?\d*)?)$)/.test(url) === false) { url = `${url}.rocket.chat`; } if (/^(https?:\/\/)?(((\w|[0-9-_])+(\.(\w|[0-9-_])+)+)|localhost)(:\d+)?$/.test(url)) { if (/^localhost(:\d+)?/.test(url)) { url = `http://${url}`; } else if (/^https?:\/\//.test(url) === false) { url = `https://${url}`; } } return url.replace(/\/+$/, '').replace(/\\/g, '/'); }; uriToPath = (uri: string) => uri.replace('file://', ''); handleRemove = () => { // TODO: Remove ts-ignore when migrate the showConfirmationAlert // @ts-ignore showConfirmationAlert({ message: I18n.t('You_will_unset_a_certificate_for_this_server'), confirmationText: I18n.t('Remove'), onPress: this.setState({ certificate: null }) // We not need delete file from DocumentPicker because it is a temp file }); }; deleteServerHistory = async (item: IServer) => { const db = database.servers; try { await db.write(async () => { await item.destroyPermanently(); }); this.setState((prevstate: IState) => ({ serversHistory: prevstate.serversHistory.filter((server: IServer) => server.id !== item.id) })); } catch { // Nothing } }; renderCertificatePicker = () => { const { certificate } = this.state; const { theme, width, height, previousServer } = this.props; return ( {certificate ? I18n.t('Your_certificate') : I18n.t('Do_you_have_a_certificate')} {certificate ?? I18n.t('Apply_Your_Certificate')} ); }; render() { const { connecting, theme, previousServer, width, height } = this.props; const { text, connectingOpen, serversHistory } = this.state; const marginTop = previousServer ? 0 : 35; return ( Rocket.Chat {I18n.t('Onboarding_subtitle')}