Skip to content

Latest commit

 

History

History
510 lines (471 loc) · 14.4 KB

README.md

File metadata and controls

510 lines (471 loc) · 14.4 KB

Description

This is the back-end API for our Frugal Foods application. For for detailed information about the application please visit the frugal foods front-end repository.

This app includes a PostgreSQL database that stores users, stores, items, store items, user store items, and user stores information for the app. Store and product information is provided by consumption of both the Target and [Kroger] (https://developer.kroger.com/reference/) API's. This app consolidates the data provided by the external API's to be consumed by our front end and served to the end user. This prototype currently has two zipcodes available for demo: '80108' and '80206'.

Links

Frugal Foods Live
Frugal Foods BE deployment for Postman use
frugal foods front-end repository

Technologies Used

Ruby 2.7.2
Rails 5.2.6
Apollo GraphQL Ruby Rails Postgres CircleCI Postman
Fly.io for deployment

Schema:

The schema of this project includes users, stores, and items. And many joins tables between them.

Local Setup Directions

  1. Clone down the repo.
git clone [email protected]:Frugal-Foods/Frugal-Foods-BE.git
  1. Switch your working directory to the root directory.
cd Frugal-Foods-BE/

(If you have not installed Ruby before, you can follow this tutorial (for Mac computers).
3) Install bundler if you haven't before.

gem install bundler
  1. Install the project's gem dependencies.
bundle install
  1. Set up the database.
rails db:{create,migrate}
  1. Seed the DB (prototype version only).
rails json_load:all
  1. Run the Rails server.
rails s

Follow the setup instructions for the frugal foods front-end repository in order to make queries to the frugal foods API.

You can also access queries through Postman by sending a POST request to localhost:3000/graphql and sending the query (shown below) in the body of your request.

Future goals

  • Due to rate limiting for free usage of both the Target and Kroger API's, our prototype only includes data from two zipcodes. Our goal moving forward to make this application more dynamic is to make live API calls to return data for any user in any zipcode.
  • Implement authentication, allowing a user to create an account and have their popular stores and items remembered
  • Implement 'shop by department' feature that allows user to browse all products for a store along with the ability to search for a specific item.
  • Collect data for other popular grocery stores to allow the user more options when choosing stores near them.
  • Implement 'cheapest in one stop' feature that allows a user to search all items on their grocery list and see which store would have the cheapest total if they only wanted to stop at one store.
  • Implement a map feature that creates a route for the user depending on which stores were chosen.

Endpoints Available

Base URL: https://frugal-foods-be.fly.dev/ All endpoints below can be accessed with POST request to https://frugal-foods-be.fly.dev/graphql and the query (below) sent in the request body.

Endpoints

Get User

This endpoint requires no parameter.

Request

{
  user {
    id
  }
}

Get Stores

This endpoint requires a zipcode parameter.
Note: This prototype currently has two zipcodes available for demo: '80108' and '80206'. It is returning all Kroger and Target stores within 10 miles of the zipcode.

Request

{
  user {
    id
  }
}

Expected Response

{
    "data": {
        "stores": [
            {
                "id": "84",
                "name": "King Soopers - CAPITOL HILL",
                "address": "1155 E 9Th Ave, Denver, CO, 80218"
            },
            {
                "id": "85",
                "name": "King Soopers - MAYFAIR",
                "address": "1355 Krameria St, Denver, CO, 80220"
            },
            {
                "id": "86",
                "name": "King Soopers - Leetsdale Cherry",
                "address": "4600 Leetsdale Dr, Glendale, CO, 80246"
            }
        ]
    }
}

Expected Response

{
    "data": {
        "stores": [
            {
                "id": "84",
                "name": "King Soopers - CAPITOL HILL",
                "address": "1155 E 9Th Ave, Denver, CO, 80218"
            },
            {
                "id": "85",
                "name": "King Soopers - MAYFAIR",
                "address": "1355 Krameria St, Denver, CO, 80220"
            },
            {
                "id": "86",
                "name": "King Soopers - Leetsdale Cherry",
                "address": "4600 Leetsdale Dr, Glendale, CO, 80246"
            }
        ]
    }
}

Get UserStores

This endpoint requires a userId parameter and returns all stores that are associated with that user.

Request

query {
  userStores(userId: 5) {
    id
    storeId
    userId
  }
}

Expected Response

{
    "data": {
        "userStores": [
            {
                "id": "207",
                "storeId": "84",
                "userId": 5
            },
            {
                "id": "210",
                "storeId": "82",
                "userId": 5
            },
            {
                "id": "211",
                "storeId": "83",
                "userId": 5
            },
            {
                "id": "212",
                "storeId": "88",
                "userId": 5
            }
        ]
    }
}

Get StoreItems

This endpoint requires a search parameter and a userId parameter. It returns all items that match the search parameter from stores that are associated with a specific User.

Request

query storeItems {
  items(search: "banana", userId: 5) {
    	userId
      itemName
      itemId
      photoUrl
      storeName
      price
      storeItemId
      storeId
  }
}

Expected Response

{
    "data": {
        "items": [
            {
                "userId": 5,
                "itemName": "Organic Banana",
                "itemId": "1774",
                "photoUrl": "https://www.kroger.com/product/images/small/front/0000000094011",
                "storeName": "King Soopers - Castle Rock",
                "price": 0.69,
                "storeItemId": "21382",
                "storeId": "82"
            },
            {
                "userId": 5,
                "itemName": "Organic Banana",
                "itemId": "1774",
                "photoUrl": "https://www.kroger.com/product/images/small/front/0000000094011",
                "storeName": "King Soopers - Castle Rock",
                "price": 0.69,
                "storeItemId": "21432",
                "storeId": "83"
            },
            {
                "userId": 5,
                "itemName": "Organic Banana",
                "itemId": "1774",
                "photoUrl": "https://www.kroger.com/product/images/small/front/0000000094011",
                "storeName": "King Soopers - CAPITOL HILL",
                "price": 0.69,
                "storeItemId": "21482",
                "storeId": "84"
            },
            {
                "userId": 5,
                "itemName": "Banana - each",
                "itemId": "1592",
                "photoUrl": "https://target.scene7.com/is/image/Target/GUEST_f5d0cfc3-9d02-4ee0-a6c6-ed5dc09971d1",
                "storeName": "Target",
                "price": 0.25,
                "storeItemId": "20456",
                "storeId": "88"
            },
            {
                "userId": 5,
                "itemName": "Organic Bananas - 2lb",
                "itemId": "1602",
                "photoUrl": "https://target.scene7.com/is/image/Target/GUEST_90259dc9-363e-43e3-9570-93655723ec61",
                "storeName": "Target",
                "price": 1.49,
                "storeItemId": "20516",
                "storeId": "88"
            }
        ]
    }
}

Get UserStoreItems

This endpoint requires a userId parameter. It returns all items associated with a User, organized by store. This can be used for a final shopping list.

Request

  { userStoreItems(userId: 5) {
    storeId
    name
    address
    storeTotalPrice
    listItems {
      id
      itemName
      itemPhotoUrl
      price
      quantity
      itemTotal
    }
  }
  }

Expected Response

{
    "data": {
        "userStoreItems": [
            {
                "storeId": 84,
                "name": "King Soopers - CAPITOL HILL",
                "address": "1155 E 9Th Ave, Denver, CO, 80218",
                "storeTotalPrice": 0.0,
                "listItems": []
            },
            {
                "storeId": 82,
                "name": "King Soopers - Castle Rock",
                "address": "7284 Lagae Rd, Castle Pines, CO, 80108",
                "storeTotalPrice": 0.0,
                "listItems": []
            },
            {
                "storeId": 83,
                "name": "King Soopers - Castle Rock",
                "address": "750 N Ridge Rd, Castle Rock, CO, 80104",
                "storeTotalPrice": 1.38,
                "listItems": null
            },
            {
                "storeId": 88,
                "name": "Target",
                "address": "11150 S Twenty Mile Rd, Parker, CO, 80134",
                "storeTotalPrice": 4.19,
                "listItems": null
            }
        ]
    },
    "errors": [
        {
            "message": "Cannot return null for non-nullable field UserStoreItem.id"
        },
        {
            "message": "Cannot return null for non-nullable field UserStoreItem.id"
        }
    ]
}

Create UserStore

This mutation requires userId and storeId parameters. It creates an object in the database that creates an association between a User and a Store.

Request

mutation {
  createUserStore(input: {userId: 5, storeId: 91}) {
    userStore {
      id
      userId
      storeId
    }
    errors
  }
}

Expected Response

{
    "data": {
        "createUserStore": {
            "userStore": {
                "id": "213",
                "userId": 5,
                "storeId": 91
            },
            "errors": []
        }
    }
}

Create UserStoreItem

This mutation requires userId, storeItemId, and quantity parameters. This mutation creates an association between a User and a StoreItem (for example: used when adding items from a specific store to a cart).

Request

mutation {
  createUserStoreItem(input: {userId: 5, storeItemId: 19615, quantity: 4}) {
    userStoreItem {
      id
      userId
      storeItemId
      quantity
    }
    errors
  }
}

Expected Response

{
    "data": {
        "createUserStoreItem": {
            "userStoreItem": {
                "id": "175",
                "userId": 5,
                "storeItemId": 19615,
                "quantity": 4
            },
            "errors": []
        }
    }
}

Delete UserStore

This mutation requires an id (UserStore) parameter. It destroys the association between a User and a Store.

Request

mutation{
	destroyUserStore(input:
  { id:"213"}) {
	  id
  }
}

Expected Response

{
    "data": {
        "destroyUserStore": {
            "id": "213"
        }
    }
}

Delete UserStoreItem

This mutation requires an id (UserStoreItem) parameter. It destroys the association between a User and a StoreItem (removes an item from a cart).

Request

mutation{
	destroyUserStoreItem(input:
  { id:"13"}) {
	  id
  }
}

Expected Response

{
    "data": {
        "destroyUserStore": {
            "id": "213"
        }
    }
}

Delete ALL UserStores

This endpoint requires a userId parameter and destroys ALL associations between a User and a Store. It will also destroy all associations between a User and a StoreItem (clear the shopping cart). This is used for restarting a shopping session.

Request

mutation {
  destroyAllUserStores(input: {userId: "5"}) {
    userId
  }
}

Expected Response

{
    "data": {
        "destroyAllUserStores": {
            "userId": "[]"
        }
    }
}

Update UserStoreItem

This endpoint requires id (UserStoreItem) and quantity parameters. This endpoint patches / updates the quantity of a UserStoreItem (can be used to update the quantity of items in a cart).

Request

mutation{
  updateUserStoreItem(input:{
    id: "175",
    quantity: 11
      }) {
        id
        quantity
    }
  }

Expected Response

{
    "data": {
        "updateUserStoreItem": {
            "id": "175",
            "quantity": 11
        }
    }
}

Contributors

Mentor

Project Manager