UNPKG

react-sprucebot

Version:

React components for your Sprucebot Skill 💪🏼

373 lines (290 loc) 13 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); var _react = require('react'); var _react2 = _interopRequireDefault(_react); var _actions = require('../store/actions'); var actions = _interopRequireWildcard(_actions); var _cookies = require('cookies'); var _cookies2 = _interopRequireDefault(_cookies); var _jsCookies = require('js-cookies'); var _jsCookies2 = _interopRequireDefault(_jsCookies); var _index = require('../index'); var _index2 = _interopRequireDefault(_index); var _DevControls = require('../../components/DevControls/DevControls'); var _DevControls2 = _interopRequireDefault(_DevControls); var _Loader = require('../../components/Loader/Loader'); var _Loader2 = _interopRequireDefault(_Loader); var _qs = require('qs'); var _qs2 = _interopRequireDefault(_qs); var _lang = require('../helpers/lang'); var _lang2 = _interopRequireDefault(_lang); var _router = require('next/router'); var _router2 = _interopRequireDefault(_router); function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key]; } } newObj.default = obj; return newObj; } } function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in obj) { if (keys.indexOf(i) >= 0) continue; if (!Object.prototype.hasOwnProperty.call(obj, i)) continue; target[i] = obj[i]; } return target; } function _asyncToGenerator(fn) { return function () { var gen = fn.apply(this, arguments); return new Promise(function (resolve, reject) { function step(key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { return Promise.resolve(value).then(function (value) { step("next", value); }, function (err) { step("throw", err); }); } } return step("next"); }); }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } var debug = require('debug')('react-sprucebot'); var setCookie = function setCookie(named, value, req, res) { if (req && req.headers) { var cookies = new _cookies2.default(req, res, { secure: true }); return cookies.set(named, value); } else { return _jsCookies2.default.setItem(named, value); } }; var getCookie = function getCookie(named, req, res) { if (req && req.headers) { var cookies = new _cookies2.default(req, res, { secure: true }); return cookies.get(named); } else { return _jsCookies2.default.getItem(named); } }; var Page = function Page(Wrapped) { var ConnectedWrapped = (0, _router.withRouter)(Wrapped); return function (_Component) { _inherits(_class2, _Component); function _class2(props) { _classCallCheck(this, _class2); var _this = _possibleConstructorReturn(this, (_class2.__proto__ || Object.getPrototypeOf(_class2)).call(this, props)); _this.handleRouteChangStart = function () { // don't user skill off props, it is pulled server side and lacks all functions _index2.default.notifyOfRouteChangeStart(); }; _this.state = { attemptingReAuth: !!props.attemptingReAuth, isIframed: true }; _this.messageHandler = _this.messageHandler.bind(_this); return _this; } // Everything here is run server side _createClass(_class2, [{ key: 'messageHandler', value: function messageHandler(e) { if (e.data === 'Skill:NotReAuthing') { this.setState({ attemptingReAuth: false }); } } }, { key: 'componentDidMount', value: function () { var _ref = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee() { var WebFont; return regeneratorRuntime.wrap(function _callee$(_context) { while (1) { switch (_context.prev = _context.next) { case 0: window.addEventListener('message', this.messageHandler); if (window.self === window.top || window.__SBTEAMMATE__) { // make sure we are being loaded inside sb console.error('NOT LOADED FROM SPRUCEBOT!! BAIL BAIL BAIL'); this.setState({ attemptingReAuth: false, isIframed: !!window.__SBTEAMMATE__ }); } else if (this.props.attemptingReAuth) { _index2.default.forceAuth(); } // NOTE: Need to do this require here so that we can be sure the global window is defined WebFont = require('webfontloader'); //eslint-disable-line WebFont.load({ google: { families: ['Material Icons'] } }); _router2.default && _router2.default.router && _router2.default.router.events.on('routeChangeStart', this.handleRouteChangStart); case 5: case 'end': return _context.stop(); } } }, _callee, this); })); function componentDidMount() { return _ref.apply(this, arguments); } return componentDidMount; }() }, { key: 'componentWillUnmount', value: function componentWillUnmount() { window.removeEventListener('message', this.messageHandler); _router2.default && _router2.default.router && _router2.default.router.events.off('routeChangeStart', this.handleRouteChangStart); } }, { key: 'render', value: function render() { if (this.state.attemptingReAuth) { return _react2.default.createElement(_Loader2.default, null); } if (this.props.config.DEV_MODE) { return _react2.default.createElement( 'div', null, this.state.isIframed ? _react2.default.createElement( 'style', { jsx: true, global: true }, '\n\t\t\t\t\t\t\t\thtml,\n\t\t\t\t\t\t\t\tbody {\n\t\t\t\t\t\t\t\t\toverflow: hidden;\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t' ) : null, _react2.default.createElement(_DevControls2.default, { auth: this.props.auth }), _react2.default.createElement(ConnectedWrapped, _extends({}, this.props, { skill: _index2.default, lang: _lang2.default })) ); } return _react2.default.createElement( 'div', null, this.state.isIframed ? _react2.default.createElement( 'style', { jsx: true, global: true }, '\n\t\t\t\t\t\t\thtml,\n\t\t\t\t\t\t\tbody {\n\t\t\t\t\t\t\t\toverflow: hidden;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t' ) : null, _react2.default.createElement(ConnectedWrapped, _extends({}, this.props, { skill: _index2.default, lang: _lang2.default })) ); } }], [{ key: 'getInitialProps', value: function () { var _ref2 = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee2(_ref3) { var pathname = _ref3.pathname, query = _ref3.query, asPath = _ref3.asPath, store = _ref3.store, res = _ref3.res, req = _ref3.req; var props, jwt, state, args, redirect, role, firstPart, _jwt, rest, queryString, _args2 = arguments; return regeneratorRuntime.wrap(function _callee2$(_context2) { while (1) { switch (_context2.prev = _context2.next) { case 0: props = { pathname: pathname, query: query, asPath: asPath, skill: _index2.default }; jwt = query.jwt || getCookie('jwt', req, res); if (!jwt) { _context2.next = 17; break; } _context2.prev = 3; _context2.next = 6; return store.dispatch(actions.auth.go(jwt)); case 6: _context2.next = 8; return store.dispatch(actions.onboarding.didOnboarding()); case 8: // only save cookie if a new one has been passed if (query.jwt) { setCookie('jwt', query.jwt, req, res); } _context2.next = 15; break; case 11: _context2.prev = 11; _context2.t0 = _context2['catch'](3); debug(_context2.t0); debug('Error fetching user from jwt'); case 15: _context2.next = 18; break; case 17: debug('This looks pretty bad. You are missing a jwt and will probably be unauthorized'); case 18: state = store.getState(); if (state.auth && !state.auth.error) { state.auth.role = state.config.DEV_MODE && getCookie('devRole', req, res) || state.auth.role; } if (!ConnectedWrapped.getInitialProps) { _context2.next = 30; break; } args = Array.from(_args2); args[0] = _extends({}, args[0], state); _context2.t1 = _extends; _context2.t2 = {}; _context2.t3 = props; _context2.next = 28; return ConnectedWrapped.getInitialProps.apply(this, args); case 28: _context2.t4 = _context2.sent; props = (0, _context2.t1)(_context2.t2, _context2.t3, _context2.t4); case 30: redirect = props.redirect || false; if (query.back && query.jwt && (query.back.search('sprucebot.com') > 0 || query.back.search('bshop.io') > 0)) { // if there is a jwt, we are being authed redirect = query.back; } else if (!redirect && !props.public && (!state.auth || !state.auth.role || state.auth.error)) { // no redirect is set, we're not public, but auth failed redirect = '/unauthorized'; debug('AUTH FAILED', state); } else if (!redirect && !props.public) { // all things look good, lets just make sure we're in the right area (owner, teammate, or guest) role = state.auth.role; firstPart = props.pathname.split('/')[1]; _jwt = query.jwt, rest = _objectWithoutProperties(query, ['jwt']); queryString = _qs2.default.stringify(rest); // we are at '/' then redirect to the corresponding role's path if (props.pathname === '/') { redirect = '/' + role + '?' + queryString; } else if (role !== firstPart) { redirect = '/unauthorized'; } } if (!(redirect && res)) { _context2.next = 39; break; } res.writeHead(302, { Location: redirect }); res.end(); res.finished = true; return _context2.abrupt('return'); case 39: if (redirect) { window.location.href = redirect; } case 40: // if we are /unauthorized, don't have a cookie, but have NOT done cookie check if (props.pathname === '/unauthorized' && (!state.auth || !state.auth.role)) { props.attemptingReAuth = true; } // We can only return a plain object here because it is passed to the browser // No circular dependencies return _context2.abrupt('return', props); case 42: case 'end': return _context2.stop(); } } }, _callee2, this, [[3, 11]]); })); function getInitialProps(_x) { return _ref2.apply(this, arguments); } return getInitialProps; }() }]); return _class2; }(_react.Component); }; exports.default = function (Wrapped) { return Page(Wrapped); };