diff --git a/package.json b/package.json
index 3c33187..f428b08 100644
--- a/package.json
+++ b/package.json
@@ -3,18 +3,23 @@
"version": "0.1.0",
"private": true,
"dependencies": {
+ "babel-cli": "6.26.0",
"babel-preset-es2015": "^6.24.1",
"babel-preset-stage-0": "^6.24.1",
"body-parser": "^1.18.2",
"bootstrap": "^4.0.0",
"classnames": "^2.2.5",
"express": "^4.16.2",
+ "lodash": "4.17.10",
"mongoose": "^4.13.6",
"react": "^16.0.0",
"react-dom": "^16.0.0",
+ "react-redux": "^5.0.7",
"react-router-dom": "^4.2.2",
"react-scripts": "1.0.17",
"reactstrap": "^5.0.0-beta.3",
+ "redux": "^4.0.0",
+ "redux-thunk": "^2.2.0",
"uuid": "^3.1.0"
},
"devDependencies": {
diff --git a/src/client/components/AddToCart.js b/src/client/components/AddToCart.js
new file mode 100644
index 0000000..c262c6b
--- /dev/null
+++ b/src/client/components/AddToCart.js
@@ -0,0 +1,34 @@
+import React from "react";
+import { bindActionCreators } from "redux";
+import { connect } from "react-redux";
+import { Button } from "reactstrap";
+import * as actionCreators from "../redux/actions";
+
+class AddToCart extends React.Component {
+ render() {
+ const {
+ actions: { addItemsToCart },
+ cartItems: { isLoading },
+ product
+ } = this.props;
+ return (
+
+ );
+ }
+}
+
+const mapStateToProps = state => ({
+ cartItems: state.cartItems
+});
+
+const mapDispatchToProps = dispatch => ({
+ actions: bindActionCreators(actionCreators, dispatch)
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(AddToCart);
diff --git a/src/client/components/App.js b/src/client/components/App.js
new file mode 100644
index 0000000..6fb283b
--- /dev/null
+++ b/src/client/components/App.js
@@ -0,0 +1,27 @@
+import React from "react";
+import { Route, Switch } from "react-router-dom";
+import Home from "./Home";
+import Products from "./Products";
+import Product from "./Product";
+import Cart from "./Cart";
+import Header from "./Header";
+import { Container } from "reactstrap";
+
+export default class App extends React.Component {
+ render() {
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+ }
+}
diff --git a/src/client/components/Cart.js b/src/client/components/Cart.js
new file mode 100644
index 0000000..82fe003
--- /dev/null
+++ b/src/client/components/Cart.js
@@ -0,0 +1,39 @@
+import React from "react";
+import { ListGroup, ListGroupItem, Alert } from "reactstrap";
+import DeleteCartItem from "./DeleteCartItem";
+import { connect } from "react-redux";
+
+function CartItem({ cartItem }) {
+ return (
+
+
+
{cartItem.product.name}
+
+
+
+
+
+ );
+}
+
+export class Cart extends React.Component {
+ render() {
+ const { cartItems } = this.props;
+ if (!cartItems.ids.length) {
+ return Cart is empty;
+ }
+ return (
+
+ {Object.values(cartItems.byId).map(cartItem => (
+
+ ))}
+
+ );
+ }
+}
+
+const mapStateToProps = state => ({
+ cartItems: state.cartItems
+});
+
+export default connect(mapStateToProps)(Cart);
diff --git a/src/client/components/CartBadge.js b/src/client/components/CartBadge.js
new file mode 100644
index 0000000..3784d1e
--- /dev/null
+++ b/src/client/components/CartBadge.js
@@ -0,0 +1,32 @@
+import React from "react";
+import { bindActionCreators } from "redux";
+import { connect } from "react-redux";
+import * as actions from "../redux/actions";
+import { Badge } from "reactstrap";
+
+class CartBadge extends React.Component {
+ componentDidMount() {
+ const { actions } = this.props;
+ actions.getCartItems();
+ }
+
+ render() {
+ const { cartItems } = this.props;
+ if (!cartItems.ids.length) {
+ return null;
+ }
+ return cartItems.ids.length ? (
+ {cartItems.ids.length}
+ ) : null;
+ }
+}
+
+const mapStateToProps = state => ({
+ cartItems: state.cartItems
+});
+
+const mapDispatchToProps = dispatch => ({
+ actions: bindActionCreators(actions, dispatch)
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(CartBadge);
diff --git a/src/client/components/DeleteCartItem.js b/src/client/components/DeleteCartItem.js
new file mode 100644
index 0000000..eaf110a
--- /dev/null
+++ b/src/client/components/DeleteCartItem.js
@@ -0,0 +1,7 @@
+import React from "react";
+
+export default class DeleteCartItem extends React.Component {
+ render() {
+ return
;
+ }
+}
diff --git a/src/client/components/Header.js b/src/client/components/Header.js
new file mode 100644
index 0000000..bb1fc5b
--- /dev/null
+++ b/src/client/components/Header.js
@@ -0,0 +1,60 @@
+import React from "react";
+import { Navbar, NavbarBrand, Nav, NavItem } from "reactstrap";
+import { Link } from "react-router-dom";
+import classnames from "classnames";
+import CartBadge from "./CartBadge";
+
+export default class Header extends React.Component {
+ render() {
+ const { location: { pathname } } = this.props;
+ return (
+
+
+ Apollo Store
+
+
+
+ );
+ }
+}
diff --git a/src/client/components/Home.js b/src/client/components/Home.js
new file mode 100644
index 0000000..b4af15f
--- /dev/null
+++ b/src/client/components/Home.js
@@ -0,0 +1,7 @@
+import React from "react";
+
+export default class Home extends React.Component {
+ render() {
+ return Welcome to apollo store
;
+ }
+}
diff --git a/src/client/components/Price.js b/src/client/components/Price.js
new file mode 100644
index 0000000..51afe37
--- /dev/null
+++ b/src/client/components/Price.js
@@ -0,0 +1,5 @@
+import React from "react";
+
+export default function Price({ value }) {
+ return ₹ {value};
+}
diff --git a/src/client/components/Product.js b/src/client/components/Product.js
new file mode 100644
index 0000000..4166004
--- /dev/null
+++ b/src/client/components/Product.js
@@ -0,0 +1,51 @@
+import React from "react";
+import { bindActionCreators } from "redux";
+import { connect } from "react-redux";
+import { Media } from "reactstrap";
+import * as actionCreators from "../redux/actions";
+import AddToCart from "./AddToCart";
+import Price from "./Price";
+
+class Product extends React.Component {
+ componentDidMount() {
+ const { product, actions, match } = this.props;
+ if (!product) {
+ const productId = parseInt(match.params.id, 10);
+ actions.getProducts(productId);
+ }
+ }
+
+ render() {
+ const { products, match } = this.props;
+ const productId = parseInt(match.params.id, 10);
+ const product = products.ids.length && products.byId[productId];
+ if (!product) {
+ return null;
+ }
+ return (
+
+
+
+
+
+ {product.name}
+ {product.description}
+
+
+
+
+ );
+ }
+}
+
+const mapStateToProps = (state, props) => ({
+ products: state.products
+});
+
+const mapDisptachToProps = dispatch => ({
+ actions: bindActionCreators(actionCreators, dispatch)
+});
+
+export default connect(mapStateToProps, mapDisptachToProps)(Product);
diff --git a/src/client/components/ProductSelect.js b/src/client/components/ProductSelect.js
new file mode 100644
index 0000000..dbf9b72
--- /dev/null
+++ b/src/client/components/ProductSelect.js
@@ -0,0 +1,7 @@
+import React from "react";
+
+export default class ProductSelect extends React.Component {
+ render() {
+ return ;
+ }
+}
diff --git a/src/client/components/Products.js b/src/client/components/Products.js
new file mode 100644
index 0000000..243fd5e
--- /dev/null
+++ b/src/client/components/Products.js
@@ -0,0 +1,63 @@
+import React from "react";
+import { bindActionCreators } from "redux";
+import { connect } from "react-redux";
+import { Link } from "react-router-dom";
+import { ListGroup, ListGroupItem } from "reactstrap";
+import * as productActionCreators from "../redux/actions";
+import Price from "./Price";
+import ProductSelect from "./ProductSelect";
+
+function Product({ product }) {
+ return (
+
+
+
+ );
+}
+
+class Products extends React.Component {
+ componentDidMount() {
+ const { productActions, match } = this.props;
+ const brand = match.params.brand;
+ productActions.getProducts(brand);
+ }
+
+ componentWillReceiveProps(nextProps) {
+ const { match, productActions } = this.props;
+ if (nextProps.match.params.brand !== match.params.brand) {
+ productActions.getProducts(nextProps.match.params.brand);
+ }
+ }
+
+ render() {
+ const { products } = this.props;
+ return (
+
+ Products
+
+ {Object.values(products.byId).map(product => (
+
+ ))}
+
+
+ );
+ }
+}
+
+const mapStateToProps = state => ({
+ products: state.products
+});
+
+const mapDispatchToProps = dispatch => ({
+ productActions: bindActionCreators(productActionCreators, dispatch)
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(Products);
diff --git a/src/client/redux/actionTypes/index.js b/src/client/redux/actionTypes/index.js
new file mode 100644
index 0000000..b4ea067
--- /dev/null
+++ b/src/client/redux/actionTypes/index.js
@@ -0,0 +1,9 @@
+export const GET_PRODUCTS_REQUEST = "GET_PRODUCTS_REQUEST";
+export const GET_PRODUCTS_SUCCESS = "GET_PRODUCTS_SUCCESS";
+export const GET_PRODUCTS_FAILURE = "GET_PRODUCTS_SUCCESS";
+export const GET_CART_ITEMS_REQUEST = "GET_CART_ITEMS_REQUEST";
+export const GET_CART_ITEMS_SUCCESS = "GET_CART_ITEMS_SUCCESS";
+export const GET_CART_ITEMS_FAILURE = "GET_CART_ITEMS_FAILURE";
+export const ADD_ITEMS_TO_CART_REQUEST = "ADD_ITEMS_TO_CART_REQUEST";
+export const ADD_ITEMS_TO_CART_SUCCESS = "ADD_ITEMS_TO_CART_SUCCESS";
+export const ADD_ITEMS_TO_CART_FAILURE = "ADD_ITEMS_TO_CART_FAILURE";
diff --git a/src/client/redux/actions/index.js b/src/client/redux/actions/index.js
new file mode 100644
index 0000000..534c167
--- /dev/null
+++ b/src/client/redux/actions/index.js
@@ -0,0 +1,58 @@
+import * as actionTypes from "../actionTypes";
+import { transformProductsApi } from "../transformers/transformProductsApi";
+import { transformGetCartItemsApi } from "../transformers/transformGetCartItemsApi";
+
+export const getProducts = productId => {
+ const apiUrl = productId ? `/api/products/${productId}` : "/api/products";
+ return dispatch => {
+ return fetch(apiUrl).then(async response => {
+ const responseData = await response.json();
+ const data = transformProductsApi(responseData);
+ dispatch({
+ type: actionTypes.GET_PRODUCTS_SUCCESS,
+ payload: data
+ });
+ });
+ };
+};
+
+export const getCartItems = () => {
+ return dispatch => {
+ return fetch("/api/cart-items").then(async response => {
+ const responseData = await response.json();
+ dispatch({
+ type: actionTypes.GET_CART_ITEMS_SUCCESS,
+ payload: responseData
+ });
+ });
+ };
+};
+
+export const addItemsToCart = product => {
+ const data = {
+ productId: product.id
+ };
+ return dispatch => {
+ dispatch({ type: actionTypes.ADD_ITEMS_TO_CART_REQUEST });
+ return fetch("/api/cart-items", {
+ method: "POST",
+ body: JSON.stringify(data),
+ headers: {
+ "Content-Type": "application/json"
+ }
+ }).then(async response => {
+ const responseData = await response.json();
+ if (response.ok) {
+ dispatch({
+ type: actionTypes.ADD_ITEMS_TO_CART_SUCCESS,
+ payload: responseData
+ });
+ } else {
+ dispatch({
+ type: actionTypes.ADD_ITEMS_TO_CART_FAILURE,
+ payload: "Cannot add Products"
+ });
+ }
+ });
+ };
+};
diff --git a/src/client/redux/reducers/cartItems.js b/src/client/redux/reducers/cartItems.js
new file mode 100644
index 0000000..4f9dcf8
--- /dev/null
+++ b/src/client/redux/reducers/cartItems.js
@@ -0,0 +1,41 @@
+import * as actionTypes from "../actionTypes";
+
+const initialState = {
+ byId: {},
+ ids: [],
+ isLoading: false,
+ isError: false,
+ errorMsg: ""
+};
+
+export default function cartItemsReducer(state = initialState, action) {
+ switch (action.type) {
+ case actionTypes.ADD_ITEMS_TO_CART_REQUEST:
+ case actionTypes.GET_CART_ITEMS_REQUEST:
+ return {
+ ...state,
+ isLoading: true
+ };
+
+ case actionTypes.GET_CART_ITEMS_SUCCESS:
+ return {
+ ...state,
+ ...action.payload,
+ isLoading: false
+ };
+
+ case actionTypes.ADD_ITEMS_TO_CART_SUCCESS:
+ return {
+ ...state,
+ byId: {
+ ...state.byId,
+ [action.payload.id]: action.payload
+ },
+ ids: [...state.ids, action.payload.id],
+ isLoading: false
+ };
+
+ default:
+ return state;
+ }
+}
diff --git a/src/client/redux/reducers/index.js b/src/client/redux/reducers/index.js
new file mode 100644
index 0000000..2b6075b
--- /dev/null
+++ b/src/client/redux/reducers/index.js
@@ -0,0 +1,8 @@
+import { combineReducers } from "redux";
+import products from "./products";
+import cartItems from "./cartItems";
+
+export default combineReducers({
+ products,
+ cartItems
+});
diff --git a/src/client/redux/reducers/products.js b/src/client/redux/reducers/products.js
new file mode 100644
index 0000000..2ba4410
--- /dev/null
+++ b/src/client/redux/reducers/products.js
@@ -0,0 +1,41 @@
+import * as actionTypes from "../actionTypes";
+import merge from "lodash/merge";
+
+const initialState = {
+ byId: {},
+ ids: [],
+ isLoading: false,
+ isError: false,
+ errorMsg: ""
+};
+
+export default function productsReducer(state = initialState, action) {
+ switch (action.type) {
+ case actionTypes.GET_PRODUCTS_REQUEST:
+ return {
+ ...state,
+ byId: {},
+ ids: [],
+ isLoading: true
+ };
+
+ case actionTypes.GET_PRODUCTS_SUCCESS:
+ return {
+ ...state,
+ byId: merge({}, state.byId, action.payload.byId),
+ ids: [...state.ids, action.payload.ids],
+ isLoading: false
+ };
+
+ case actionTypes.GET_PRODUCTS_FAILURE:
+ return {
+ ...state,
+ isLoading: false,
+ isError: true,
+ errorMsg: action.payload
+ };
+
+ default:
+ return state;
+ }
+}
diff --git a/src/client/redux/transformers/transformGetCartItemsApi.js b/src/client/redux/transformers/transformGetCartItemsApi.js
new file mode 100644
index 0000000..03647f6
--- /dev/null
+++ b/src/client/redux/transformers/transformGetCartItemsApi.js
@@ -0,0 +1,10 @@
+export const transformGetCartItemsApi = data => ({
+ byId: data.reduce(
+ (obj, cartItem) => ({
+ ...obj,
+ [cartItem.id]: cartItem
+ }),
+ {}
+ ),
+ ids: data.map(cartItem => cartItem.id)
+});
diff --git a/src/client/redux/transformers/transformProductsApi.js b/src/client/redux/transformers/transformProductsApi.js
new file mode 100644
index 0000000..0e3980e
--- /dev/null
+++ b/src/client/redux/transformers/transformProductsApi.js
@@ -0,0 +1,10 @@
+export const transformProductsApi = data => ({
+ byId: data.reduce(
+ (obj, product) => ({
+ ...obj,
+ [product.id]: product
+ }),
+ {}
+ ),
+ ids: data.map(product => product.id)
+});
diff --git a/src/client/store.js b/src/client/store.js
new file mode 100644
index 0000000..581d1da
--- /dev/null
+++ b/src/client/store.js
@@ -0,0 +1,13 @@
+import { createStore, applyMiddleware, compose } from "redux";
+import rootReducer from "./redux/reducers";
+import thunk from "redux-thunk";
+
+export default function configureStore() {
+ const composeEnhancers =
+ window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ || compose;
+ const store = createStore(
+ rootReducer,
+ composeEnhancers(applyMiddleware(thunk))
+ );
+ return store;
+}
diff --git a/src/index.js b/src/index.js
new file mode 100644
index 0000000..ec1f0e8
--- /dev/null
+++ b/src/index.js
@@ -0,0 +1,16 @@
+import React from "react";
+import ReactDOM from "react-dom";
+import App from "./client/components/App";
+import "bootstrap/dist/css/bootstrap.min.css";
+import { BrowserRouter, Route } from "react-router-dom";
+import { Provider } from "react-redux";
+import configureStore from "./client/store";
+
+ReactDOM.render(
+
+
+
+
+ ,
+ document.getElementById("root")
+);
diff --git a/src/server/connectors/index.js b/src/server/connectors/index.js
index c9a4fce..d342667 100644
--- a/src/server/connectors/index.js
+++ b/src/server/connectors/index.js
@@ -2,6 +2,7 @@ let products = [
{
id: 1,
name: "Macbook",
+ brand: "Apple",
description: "Latest Macbook with 16GB ram and Quad core processor",
price: 65000,
url: "/img/macbook.jpeg"
@@ -9,6 +10,15 @@ let products = [
{
id: 2,
name: "Keyboard",
+ brand: "Apple",
+ description: "Ergonomic keyboard",
+ price: 3000,
+ url: "/img/keyboard.jpeg"
+ },
+ {
+ id: 3,
+ name: "Keyboard",
+ brand: "Samsung",
description: "Ergonomic keyboard",
price: 3000,
url: "/img/keyboard.jpeg"
@@ -29,8 +39,14 @@ export function getProducts() {
return products;
}
-export function getProduct(id) {
- return products.find(product => product.id === id);
+export function getProductById(id) {
+ return [products.find(product => product.id === id)];
+}
+
+export function getProductsByBrand(brand) {
+ return products.filter(
+ product => product.brand.toLowerCase() === brand.toLowerCase()
+ );
}
export function getCartItem(id) {
@@ -41,16 +57,20 @@ export function getCartItems() {
return cartItems;
}
-export function addToCart(args) {
- if (cartItems.find(c => c.productId === parseInt(args.productId, 10))) {
+export function addToCart({ productId }) {
+ if (cartItems.find(c => c.productId === parseInt(productId, 10))) {
throw new Error("Product already in cart");
}
const newCartItem = {
id: cartItems.length + 1,
- productId: parseInt(args.productId, 10)
+ product: products.find(p => p.id === productId)
};
cartItems.push(newCartItem);
- return newCartItem;
+ return new Promise((resolve, reject) => {
+ setTimeout(() => {
+ resolve(newCartItem);
+ }, 5000);
+ });
}
export function deleteCartItem(args) {
diff --git a/src/server/index.js b/src/server/index.js
index 65ac35b..7c32686 100644
--- a/src/server/index.js
+++ b/src/server/index.js
@@ -4,7 +4,8 @@ import morgan from "morgan";
import {
getUser,
getProducts,
- getProduct,
+ getProductById,
+ getProductsByBrand,
getCartItems,
getCartItem,
addToCart,
@@ -13,7 +14,7 @@ import {
const PORT = 8000;
const app = express();
-app.use(bodyParser.urlencoded({ extended: false }));
+app.use(bodyParser.urlencoded({ extended: true }));
app.use(bodyParser.json());
app.use(
@@ -31,9 +32,9 @@ app.use(
})
);
-app.use(function(req, res, next) {
- setTimeout(next, 500);
-});
+// app.use(function(req, res, next) {
+// setTimeout(next, 500);
+// });
app.get("/api/user", function(req, res) {
res.json(getUser());
@@ -43,9 +44,13 @@ app.get("/api/products", function(req, res) {
res.json(getProducts());
});
-app.get("/api/products/:id", function(req, res) {
+app.get("/api/products/:id(\\d+)/", function(req, res) {
const id = parseInt(req.params.id, 10);
- res.json(getProduct(id));
+ res.json(getProductById(id));
+});
+
+app.get("/api/products/:brand(\\w+)/", function(req, res) {
+ res.json(getProductsByBrand(req.params.brand));
});
app.get("/api/cart-items", function(req, res) {
@@ -58,7 +63,7 @@ app.get("/api/cart-items/:id", function(req, res) {
});
app.post("/api/cart-items", function(req, res) {
- res.json(addToCart(req.body));
+ addToCart(req.body).then(response => res.json(response));
});
app.post("/api/cart-items/:id", function(req, res) {
diff --git a/yarn.lock b/yarn.lock
index 824a933..8f79430 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -393,6 +393,27 @@ axobject-query@^0.1.0:
dependencies:
ast-types-flow "0.0.7"
+babel-cli@6.26.0:
+ version "6.26.0"
+ resolved "https://registry.yarnpkg.com/babel-cli/-/babel-cli-6.26.0.tgz#502ab54874d7db88ad00b887a06383ce03d002f1"
+ dependencies:
+ babel-core "^6.26.0"
+ babel-polyfill "^6.26.0"
+ babel-register "^6.26.0"
+ babel-runtime "^6.26.0"
+ commander "^2.11.0"
+ convert-source-map "^1.5.0"
+ fs-readdir-recursive "^1.0.0"
+ glob "^7.1.2"
+ lodash "^4.17.4"
+ output-file-sync "^1.1.2"
+ path-is-absolute "^1.0.1"
+ slash "^1.0.0"
+ source-map "^0.5.6"
+ v8flags "^2.1.1"
+ optionalDependencies:
+ chokidar "^1.6.1"
+
babel-code-frame@6.26.0, babel-code-frame@^6.11.0, babel-code-frame@^6.22.0, babel-code-frame@^6.26.0:
version "6.26.0"
resolved "https://registry.yarnpkg.com/babel-code-frame/-/babel-code-frame-6.26.0.tgz#63fd43f7dc1e3bb7ce35947db8fe369a3f58c74b"
@@ -984,6 +1005,14 @@ babel-plugin-transform-strict-mode@^6.24.1:
babel-runtime "^6.22.0"
babel-types "^6.24.1"
+babel-polyfill@^6.26.0:
+ version "6.26.0"
+ resolved "https://registry.yarnpkg.com/babel-polyfill/-/babel-polyfill-6.26.0.tgz#379937abc67d7895970adc621f284cd966cf2153"
+ dependencies:
+ babel-runtime "^6.26.0"
+ core-js "^2.5.0"
+ regenerator-runtime "^0.10.5"
+
babel-preset-env@1.6.1:
version "1.6.1"
resolved "https://registry.yarnpkg.com/babel-preset-env/-/babel-preset-env-1.6.1.tgz#a18b564cc9b9afdf4aae57ae3c1b0d99188e6f48"
@@ -1594,7 +1623,7 @@ chardet@^0.4.0:
version "0.4.2"
resolved "https://registry.yarnpkg.com/chardet/-/chardet-0.4.2.tgz#b5473b33dc97c424e5d98dc87d55d4d8a29c8bf2"
-chokidar@^1.6.0:
+chokidar@^1.6.0, chokidar@^1.6.1:
version "1.7.0"
resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-1.7.0.tgz#798e689778151c8076b4b360e5edd28cda2bb468"
dependencies:
@@ -3192,6 +3221,10 @@ fs-extra@^0.30.0:
path-is-absolute "^1.0.0"
rimraf "^2.2.8"
+fs-readdir-recursive@^1.0.0:
+ version "1.1.0"
+ resolved "https://registry.yarnpkg.com/fs-readdir-recursive/-/fs-readdir-recursive-1.1.0.tgz#e32fc030a2ccee44a6b5371308da54be0b397d27"
+
fs.realpath@^1.0.0:
version "1.0.0"
resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f"
@@ -3370,7 +3403,7 @@ got@^6.7.1:
unzip-response "^2.0.1"
url-parse-lax "^1.0.0"
-graceful-fs@^4.1.11, graceful-fs@^4.1.2, graceful-fs@^4.1.6, graceful-fs@^4.1.9:
+graceful-fs@^4.1.11, graceful-fs@^4.1.2, graceful-fs@^4.1.4, graceful-fs@^4.1.6, graceful-fs@^4.1.9:
version "4.1.11"
resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.1.11.tgz#0e8bdfe4d1ddb8854d64e04ea7c00e2a026e5658"
@@ -3543,7 +3576,7 @@ hoek@4.x.x:
version "4.2.1"
resolved "https://registry.yarnpkg.com/hoek/-/hoek-4.2.1.tgz#9634502aa12c445dd5a7c5734b572bb8738aacbb"
-hoist-non-react-statics@^2.3.0:
+hoist-non-react-statics@^2.3.0, hoist-non-react-statics@^2.5.0:
version "2.5.0"
resolved "https://registry.yarnpkg.com/hoist-non-react-statics/-/hoist-non-react-statics-2.5.0.tgz#d2ca2dfc19c5a91c5a6615ce8e564ef0347e2a40"
@@ -3800,7 +3833,7 @@ interpret@^1.0.0:
version "1.1.0"
resolved "https://registry.yarnpkg.com/interpret/-/interpret-1.1.0.tgz#7ed1b1410c6a0e0f78cf95d3b8440c63f78b8614"
-invariant@^2.2.1, invariant@^2.2.2:
+invariant@^2.0.0, invariant@^2.2.1, invariant@^2.2.2:
version "2.2.4"
resolved "https://registry.yarnpkg.com/invariant/-/invariant-2.2.4.tgz#610f3c92c9359ce1db616e538008d23ff35158e6"
dependencies:
@@ -4908,6 +4941,10 @@ locate-path@^2.0.0:
p-locate "^2.0.0"
path-exists "^3.0.0"
+lodash-es@^4.17.5:
+ version "4.17.10"
+ resolved "https://registry.yarnpkg.com/lodash-es/-/lodash-es-4.17.10.tgz#62cd7104cdf5dd87f235a837f0ede0e8e5117e05"
+
lodash._reinterpolate@~3.0.0:
version "3.0.0"
resolved "https://registry.yarnpkg.com/lodash._reinterpolate/-/lodash._reinterpolate-3.0.0.tgz#0ccf2d89166af03b3663c796538b75ac6e114d9d"
@@ -4965,6 +5002,10 @@ lodash.uniq@^4.5.0:
version "4.5.0"
resolved "https://registry.yarnpkg.com/lodash.uniq/-/lodash.uniq-4.5.0.tgz#d0225373aeb652adc1bc82e4945339a842754773"
+lodash@4.17.10:
+ version "4.17.10"
+ resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.10.tgz#1b7793cf7259ea38fb3661d4d38b3260af8ae4e7"
+
"lodash@>=3.5 <5", lodash@^4.13.1, lodash@^4.14.0, lodash@^4.15.0, lodash@^4.17.2, lodash@^4.17.3, lodash@^4.17.4, lodash@^4.17.5, lodash@^4.3.0:
version "4.17.5"
resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.5.tgz#99a92d65c0272debe8c96b6057bc8fbfa3bed511"
@@ -5693,6 +5734,14 @@ osenv@^0.1.4:
os-homedir "^1.0.0"
os-tmpdir "^1.0.0"
+output-file-sync@^1.1.2:
+ version "1.1.2"
+ resolved "https://registry.yarnpkg.com/output-file-sync/-/output-file-sync-1.1.2.tgz#d0a33eefe61a205facb90092e826598d5245ce76"
+ dependencies:
+ graceful-fs "^4.1.4"
+ mkdirp "^0.5.1"
+ object-assign "^4.1.0"
+
p-finally@^1.0.0:
version "1.0.0"
resolved "https://registry.yarnpkg.com/p-finally/-/p-finally-1.0.0.tgz#3fbcfb15b899a44123b34b6dcc18b724336a2cae"
@@ -6480,6 +6529,17 @@ react-portal@^4.1.2:
dependencies:
prop-types "^15.5.8"
+react-redux@^5.0.7:
+ version "5.0.7"
+ resolved "https://registry.yarnpkg.com/react-redux/-/react-redux-5.0.7.tgz#0dc1076d9afb4670f993ffaef44b8f8c1155a4c8"
+ dependencies:
+ hoist-non-react-statics "^2.5.0"
+ invariant "^2.0.0"
+ lodash "^4.17.5"
+ lodash-es "^4.17.5"
+ loose-envify "^1.1.0"
+ prop-types "^15.6.0"
+
react-router-dom@^4.2.2:
version "4.2.2"
resolved "https://registry.yarnpkg.com/react-router-dom/-/react-router-dom-4.2.2.tgz#c8a81df3adc58bba8a76782e946cbd4eae649b8d"
@@ -6685,10 +6745,25 @@ reduce-function-call@^1.0.1:
dependencies:
balanced-match "^0.4.2"
+redux-thunk@^2.2.0:
+ version "2.2.0"
+ resolved "https://registry.yarnpkg.com/redux-thunk/-/redux-thunk-2.2.0.tgz#e615a16e16b47a19a515766133d1e3e99b7852e5"
+
+redux@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/redux/-/redux-4.0.0.tgz#aa698a92b729315d22b34a0553d7e6533555cc03"
+ dependencies:
+ loose-envify "^1.1.0"
+ symbol-observable "^1.2.0"
+
regenerate@^1.2.1:
version "1.3.3"
resolved "https://registry.yarnpkg.com/regenerate/-/regenerate-1.3.3.tgz#0c336d3980553d755c39b586ae3b20aa49c82b7f"
+regenerator-runtime@^0.10.5:
+ version "0.10.5"
+ resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.10.5.tgz#336c3efc1220adcedda2c9fab67b5a7955a33658"
+
regenerator-runtime@^0.11.0:
version "0.11.1"
resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz#be05ad7f9bf7d22e056f9726cee5017fbf19e2e9"
@@ -7604,6 +7679,10 @@ symbol-observable@^0.2.2:
version "0.2.4"
resolved "https://registry.yarnpkg.com/symbol-observable/-/symbol-observable-0.2.4.tgz#95a83db26186d6af7e7a18dbd9760a2f86d08f40"
+symbol-observable@^1.2.0:
+ version "1.2.0"
+ resolved "https://registry.yarnpkg.com/symbol-observable/-/symbol-observable-1.2.0.tgz#c22688aed4eab3cdc2dfeacbb561660560a00804"
+
symbol-tree@^3.2.1, symbol-tree@^3.2.2:
version "3.2.2"
resolved "https://registry.yarnpkg.com/symbol-tree/-/symbol-tree-3.2.2.tgz#ae27db38f660a7ae2e1c3b7d1bc290819b8519e6"
@@ -7963,6 +8042,10 @@ use@^3.1.0:
dependencies:
kind-of "^6.0.2"
+user-home@^1.1.1:
+ version "1.1.1"
+ resolved "https://registry.yarnpkg.com/user-home/-/user-home-1.1.1.tgz#2b5be23a32b63a7c9deb8d0f28d485724a3df190"
+
util-deprecate@~1.0.1:
version "1.0.2"
resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf"
@@ -7993,6 +8076,12 @@ uuid@^3.0.0, uuid@^3.1.0:
version "3.2.1"
resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.2.1.tgz#12c528bb9d58d0b9265d9a2f6f0fe8be17ff1f14"
+v8flags@^2.1.1:
+ version "2.1.1"
+ resolved "https://registry.yarnpkg.com/v8flags/-/v8flags-2.1.1.tgz#aab1a1fa30d45f88dd321148875ac02c0b55e5b4"
+ dependencies:
+ user-home "^1.1.1"
+
validate-npm-package-license@^3.0.1:
version "3.0.3"
resolved "https://registry.yarnpkg.com/validate-npm-package-license/-/validate-npm-package-license-3.0.3.tgz#81643bcbef1bdfecd4623793dc4648948ba98338"