Skip to content

Multi-vendor thrift-store app built with React-Natiive via Expo-CLI on a Django-DRF backend

License

Notifications You must be signed in to change notification settings

israelias/thrifthub

Repository files navigation

ThriftHub

An Online Marketplace Application

ThriftHub is a platform that allows users to post second-hand goods and connect with vendors interested in such items.

Getting Started

This project is structured as a monorepo, managing both the frontend and backend applications. To streamline the setup process, several scripts are provided to configure and run the applications efficiently.

Root package.json Scripts

  • configure:path: Executes configurePath.js, to ensure environment paths are correctly set up for the backend.

  • configure:backend: Runs configureBackend.js, to prepare the Django backend environment by setting up required Python packages and configurations.

  • setup:backend: Sequentially runs configure:path and configure:backend to prepare the backend environment.

  • start:backend: Initiates the backend server by invoking the start script defined in the backend's package.json.

  • start:frontend: Starts the frontend application by calling the start script defined in the frontend's package.json.

  • dev: Utilizes concurrently to run both the backend and frontend servers simultaneously, allowing for concurrent development.

  • start: Executes start.sh, a shell script that initializes the backend server. This script typically activates the virtual environment, applies database migrations, and starts the Django development server.
  • start: Launches the frontend application using the expo start command.

Setup Instructions

To set up the project as a monorepo:

Prerequisites

  • Ensure Yarn is installed on your system.

Installation Steps

  1. Clone the Repository:

    git clone git@github.com:israelias/thrifthub.git
    cd thrifthub
  2. Install Dependencies:

    yarn install

    This command installs all dependencies for both backend and frontend projects, leveraging Yarn Workspaces for efficient management.

  3. Configure the backend

    yarn setup:backend

    This script prepares the backend environment by setting up required environment variables and configurations.

  4. Start the Development Servers:

    yarn dev

    This command runs both the backend and frontend servers concurrently, allowing you to develop and test the applications simultaneously.

TODO

The frontend web-build at thrifthub-frontend is currently still consuming a dead heroku backend. See frontend/src/config.ts for the current backend URL. The new production backend is hosted at thrifthub-backend and should be updated in the const API_URL variable.

  • Adjust const API_URL in frontend/src/config.ts to variably match the backend server's URL for development and production environments.

Configuration Scripts

To streamline the setup process, several scripts are provided to configure the project efficiently. These scripts are designed to prepare the backend environment, set up paths, and ensure the project is ready for development.

  • configurePath.js: This script sets up necessary path configurations for the backend, ensuring that all file and directory paths are correctly established.
  • configureBackend.js: This script prepares the backend environment by setting up required environment variables and configurations, ensuring the backend is ready for development and production.
  • start.sh: A shell script that initiates the backend server, handling any necessary pre-start tasks and ensuring the server runs with the appropriate settings.
    • Activates the virtual environment.
    • Applies database migrations.
    • Starts the Django development server.

By following these instructions, you can set up and run the project efficiently within a monorepo structure.


Directory Structure (tree)
.
├── backend
│   ├── account
│   │   ├── migrations
│   │   ├── tests
│   │   │   ├── test_models.py
│   │   │   ├── test_serializers.py
│   │   │   ├── test_urls.py
│   │   │   └── test_views.py
│   │   ├── admin.py
│   │   ├── apps.py
│   │   ├── models.py
│   │   ├── schema.py
│   │   ├── serializers.py
│   │   ├── urls.py
│   │   └── views.py
│   │   ├── database_schema.txt
│   │   ├── payments_schema.txt
│   │   └── screenshots
│   ├── order
│   │   ├── migrations
│   │   ├── tests
│   │   │   ├── test_models.py
│   │   │   ├── test_serializers.py
│   │   │   ├── test_urls.py
│   │   │   └── test_views.py
│   │   ├── admin.py
│   │   ├── apps.py
│   │   ├── cart.py
│   │   ├── models.py
│   │   ├── schema.py
│   │   ├── serializers.py
│   │   ├── types.py
│   │   ├── urls.py
│   │   └── views.py
│   ├── store
│   │   ├── migrations
│   │   ├── tests
│   │   │   ├── test_models.py
│   │   │   ├── test_serializers.py
│   │   │   ├── test_urls.py
│   │   │   └── test_views.py
│   │   ├── admin.py
│   │   ├── apps.py
│   │   ├── models.py
│   │   ├── schema.py
│   │   ├── serializers.py
│   │   ├── urls.py
│   │   └── views.py
│   └── vendor
│   │   ├── migrations
│   │   ├── tests
│   │   │   ├── test_models.py
│   │   │   ├── test_serializers.py
│   │   │   ├── test_urls.py
│   │   │   └── test_views.py
│   │   ├── admin.py
│   │   ├── apps.py
│   │   ├── models.py
│   │   ├── schema.py
│   │   ├── serializers.py
│   │   ├── urls.py
│   │   └── views.py
│   ├── core
│   │   ├── asgi.py
│   │   ├── schema.py
│   │   ├── settings.py
│   │   ├── storages.py
│   │   ├── urls.py
│   │   ├── wsgi.py
│   │   └── yasg.py
│   ├── documentation
│   ├── media
│   ├── static
│   ├── staticfiles
│   ├── manage.py
│   ├── requirements.txt
│   ├── Procfile
│   ├── package.json
│   ├── start.sh
│   └── README.md
├── frontend
│   ├── src
│   │   ├── components
│   │   │   ├── account
│   │   │   ├── products
│   │   │   ├── orders
│   │   │   └── common
│   │   ├── context
│   │   │   ├── user.context.tsx
│   │   │   ├── products.context.tsx
│   │   │   ├── vendor.context.tsx
│   │   │   └── authorization.context.tsx
│   │   ├── navigation
│   │   │   ├── accountStackNavigator.tsx
│   │   │   ├── productStackNavigator.tsx
│   │   │   ├── orderStackNavigator.tsx
│   │   │   └── coreNavigator.tsx
│   │   ├── screens
│   │   │   ├── account
│   │   │   ├── products
│   │   │   ├── orders
│   │   │   └── vendor
│   │   ├── hooks
│   │   ├── assets
│   │   ├── services
│   │   ├── utils
│   │   ├── constants
│   │   ├── types.ts
│   │   └──core.tsx
│   ├── native
│   │   └── AppLoader.tsx
│   ├── web-build
│   ├── assets
│   ├── babel.config.js
│   ├── globals.d.ts
│   ├── app.json
│   ├── tsconfig.json
│   ├── types.tsx
│   ├── App.tsx
│   ├── README.md
│   ├── package.json
│   └── yarn.lock
├── .gitignore
├── LICENSE
├── CONTRIBUTING.md
├── CODE_OF_CONDUCT.md
├── SECURITY.md
├── README.md
├── configureBackend.js
├── configurePath.js
├── package.json
└── yarn.lock


back to top

Contents

UX

User Stories

New Visitor Goals

  • As a new visitor, I want to have a clear understanding of what the platform offers.
  • As a new visitor, I want to be able to register for an account easily.

Returning Visitor Goals

  • As a returning visitor, I want to log in securely.
  • As a returning visitor, I want to create a listing for my second-hand goods.
  • As a returning visitor, I want to view my active and inactive listings.
  • As a returning visitor, I want to edit the details of my listings.
  • As a returning visitor, I want to delete listings that are no longer available.
  • As a returning visitor, I want to mark a listing as sold.
  • As a returning visitor, I want to connect with the buyer of my listing to coordinate the sale.
  • As a returning visitor, I want to process payments securely with buyers.
  • As a returning visitor, I want the option to cancel a purchase agreement.

Frequent Visitor Goals

  • As a frequent visitor, I want to search for listings I have created.
  • As a frequent visitor, I want to explore listings created by other users.
  • As a frequent visitor, I want to save listings that interest me for future reference.
  • As a frequent visitor, I want to view my saved listings easily.
  • As a frequent visitor, I want to remove saved listings that are no longer relevant.
  • As a frequent visitor, I want to delete my account if I no longer need it.

Wireframes

Concept

The project can be understood as a social network with collections or arrangements with the ability to broadcast items for sale or guage prices for second-hand goods.


Store (items)



Categories (playlist)



Search (filter)



Membership/payments (e-commerce)




back to top

Features

Existing Features

Store

  • The Store features all product-related capabilities for vendors. This includes managing product listings, categorizing products, and uploading product images. Vendors can organize their store to showcase their items in an easy-to-navigate way.

    Capabilities:
    • Product Management: Vendors can add, edit, and delete products from their store.
    • Category Organization: Products can be grouped into categories for better browsing.
    • Image Upload: Vendors can upload and manage product images for each listing.
    Actions:
    • Ability to add, edit, and delete products. (Registered Vendor)
    • Ability to categorize products and create subcategories. (Registered Vendor)
    • Ability to upload and manage product images. (Registered Vendor)
    • Ability to save and un-save a product listing. (Registered User)

Vendor Registration

  • A Vendor is attached to each user in order to manage and track their sales. Vendors can also interact with their buyers, including managing product listings and fulfilling orders. The website features the ability to sign up, sign in and sign out in order to conditionally access existing features. A user is based on the user model, which requires username, email and password for new users and only the latter two for existing users. For security, only usernames are stored in local storage while tokens are stored in memory. To ensure sign out across multiple open windows, a logout event triggers a storage event listener which clears tokens and usernames in the memory of the current window.

    Capabilities:
    • Vendor Registration: Users can register as vendors to sell their products.
    • Account Management: Vendors can update their profiles, manage their contact information, and track their sales.
    • Order Fulfillment: Vendors can manage their orders, including processing and shipping.
    • Friendship: Vendors can add and manage connections with other vendors for networking and collaboration.
    Actions:
    • Ability to register as a vendor. (New User)
    • Ability to sign in to an account. (Registered Vendor)
    • Ability to sign out of an account. (Registered Vendor)
    • Ability to edit and update vendor account information. (Registered Vendor)
    • Ability to connect with other vendors by adding them as friends. (Registered Vendor)
    • Ability to view and manage orders. (Registered Vendor)
    • Ability to track and manage sales history. (Registered Vendor)

Order

  • The Order section is responsible for handling the purchasing process. Customers can view their orders, check out, and monitor the status of their orders. Vendors are notified when orders are placed and can update the status of these orders.

    Capabilities:
    • Order Creation: Customers can place orders for products listed by vendors.
    • Order Status: Vendors can update the status of the orders (e.g., processing, shipped, completed).
    • Order History: Customers and vendors can view past orders.
    Actions:
    • Ability to place orders for products. (Registered User)
    • Ability to update the status of orders. (Registered Vendor)
    • Ability to view order history. (Registered User, Registered Vendor)

Features Left to Implement

Collections

  • The website should feature the ability to group any and all existing products into named collections aka bulk deals. A collection is based on the collection model, which is a name and a list of product listings. Each field is represented as either a form field or a header followed by a ul depending on whether a collection is in edit or display mode.
  • Actions:
  • Ability to create, edit and delete a named collection. (Registered Owner)
  • Ability to add and remove listings to/from a collection. (Registered Owner)

Search

  • search_text indices are attached to the title and description fields of the product cluster. The frontend UI of the search feature is designed to return product listings that match a query. Additionally, the ability to query by tags and product_type is enabled by filtering these existing fields in the database.
  • Actions:
  • Ability to search product listings created by others. (any User)
  • Ability to filter product listings by type. (any User)
  • Ability to filter product listings by tags. (any User)
  • Ability to save or un-save product listings from search. (Registered User)
  • Ability to perform crud operations on product listings from search. (Registered Owner)

Technologies

Frameworks and Libraries

  • Please visit the frontend sub directory for details on ReactNative Typescript frameworks and libraries.

Go to frontend

  • Please visit the backend root directory for details on Python-Django frameworks and libraries.

Go to backend

Programs and Software

  • VSCode: Visual Studiio Code 2020.3.2 by Microsoft is the IDE used to locally construct the project
  • Git: Git is used as the version control system and is utilized via the WebStorm terminal to commit to Git and push to GitHub.
  • GitHub: GitHub is used to store the project's code and directory upon concurrent pushes via Git.
  • Adobe InDesign: Adobe InDesign is used to mock wireframes.

back to top

Testing

User Testing

  • As a new visitor, I want to have a good understanding of what the website does

    • User arrives at home page.
      • The screen for Marketplace appears with a description of its functionality.
      • User reads description.
    • User continues.
  • As a new visitor, I want to be able to register for an account

    • User clicks Sign up.
      • The sign-up form appears.
        • The username field is in focus.
        • User types a username, email, and password.
          • If the username is taken, an error toast alert appears.
        • User modifies the username.
      • User is redirected to their profile page.
    • User continues.
    • Review:
    • A user is able to securely create an account.
  • As a Returning Visitor, I want to be able to log in securely

    • User clicks Sign in.
      • The sign-in form appears.
        • The email field is in focus.
        • User enters email and password.
        • User presses enter.
        • A success toast alert appears.
      • User is redirected to their profile page.
    • User continues.
  • As a Returning Visitor, I want to be able to create a product listing

    • User clicks Add New Product.
      • A form appears.
        • The title field is in focus.
        • User enters title, description, tags, and uploads images.
        • User clicks Submit.
          • A success toast alert appears.
      • User is redirected to their profile page.
    • User continues.
    • Review:
    • A user is able to successfully create a product listing.
  • As a Returning Visitor, I want to be able to see my product listing

    • User arrives at Profile page.
      • A product listing appears in the user's collection.
    • User continues.
    • Review:
    • A user is able to view their own product listings.
  • As a Returning Visitor, I want to be able to edit my product listing

    • User clicks Edit this product.
      • A form appears.
        • The title field is in focus.
        • User modifies the description, updates the image, and adds a source URL.
        • User adds additional tags.
        • User clicks Submit.
          • A success toast alert appears.
      • User is redirected to their profile page.
    • User continues.
    • Review:
    • A user is able to successfully edit their product listing.
  • As a Returning Visitor, I want to be able to delete my product listing(s)

    • User clicks Edit this product.
      • A form appears.
        • The title field is in focus.
        • User clicks Delete.
          • A confirmation modal appears.
          • The cancel button is in focus.
          • User confirms deletion.
          • The modal closes, and a success toast alert appears.
      • User is redirected to their profile page.
    • User continues.
  • As a Returning Visitor, I want to be able to create a collection of my product listings

    • User clicks Add New Collection.
      • A form appears.
        • The name field is in focus.
        • User enters a collection name and selects product listings to add.
        • User clicks Submit.
          • A success toast alert appears.
      • User is redirected to their profile page.
    • User continues.
  • As a Returning Visitor, I want to be able to see collections of my product listings

    • User arrives at Profile page.
      • A Collections card appears with product listings inside.
    • User continues.
  • As a Returning Visitor, I want to be able to edit collections of my product listings

    • User clicks Edit this Collection.
      • A form appears.
        • The name field is in focus.
        • User renames the collection and adds/removes products using select inputs.
        • User clicks Submit.
          • A success toast alert appears.
      • User is redirected to their profile page.
      • The updated collection appears.
    • User continues.
  • As a Returning Visitor, I want to be able to delete a collection of my product listing(s)

    • User clicks Edit this Collection.
      • A form appears.
        • The name field is in focus.
        • User clicks Delete.
          • A confirmation modal appears.
          • The cancel button is in focus.
          • User confirms deletion.
          • The modal closes, and a success toast alert appears.
      • User is redirected to their profile page.
    • User continues.
  • As a Returning Visitor, I want to be able to connect with someone interested in my product listing

    • User clicks Interested Buyer.
      • A chat or message option appears with the buyer.
        • User communicates with the interested buyer.
    • User continues.
  • As a Returning Visitor, I want to be able to collect payment from someone purchasing my product

    • User clicks View Transaction.
      • A payment gateway appears.
        • User processes the payment.
    • User continues.
  • As a Returning Visitor, I want to be able to cancel a transaction

    • User clicks Cancel Transaction.
      • A confirmation modal appears.
      • User confirms the cancellation.
    • User continues.
  • As a Returning Visitor, I want to be able to connect with someone with a product I am interested in purchasing

    • User clicks Contact Seller.
      • A chat or message option appears with the seller.
    • User continues.
  • As a Returning Visitor, I want to be able to make payment to someone I am purchasing a product from

    • User clicks Proceed to Payment.
      • A payment gateway appears.
        • User processes the payment.
    • User continues.
  • As a Frequent Visitor, I want to be able to search product listings I have created

    • User clicks Search Products.
      • A feed of all product listings appears.
      • A Product Search input field appears.
        • User types search text.
        • Listings return products matching the search text.
        • User filters by product_type or tags.
        • User clicks Show All to see most recent products.
        • Pagination buttons appear, and the user clicks Next for additional pages.
    • User continues.
  • As a Frequent Visitor, I want to be able to save product listings created by others

    • User clicks Save Product.
      • A success toast appears.
    • User continues.
  • As a Frequent Visitor, I want to be able to see my saved product listings created by others

    • User clicks Saved Listings.
      • A collection named Saves appears.
        • User sees all saved products.
    • User continues.
  • As a Frequent Visitor, I want to be able to un-save product listings created by others I have previously saved

    • User clicks Unsave Product.
      • A success toast appears.
    • User continues.
  • As a Frequent Visitor, I want to be able to delete my account

    • User clicks Delete My Account.
      • A confirmation modal appears.
      • User confirms the deletion.
      • The modal closes, and the user is redirected to the homepage.
      • The user is no longer able to log in with their credentials.
    • User continues.
    • Review:
    • A user is able to securely remove their account from the database.

back to top

Code Testing

Frontend

Performance, Accessibility, Best Practices, SEO, PWA

View Latest Results

  • Lighthouse via Vercel is used to test performace, which produces unique results on every git push. lighthouse-badges is used to generate new badges for every deployment by installing npm i -g lighthouse-badges and pushing the new hashed url to the array of urls:
    lighthouse-badges 
    -o docs/badges -r 
    -u https://thrifthub.vercel.app/ [... all other urls]
                       
    # Output to docs/badges
    # Badges will contain the respective
      average score(s) of all the urls 
      supplied, combined
    
  • Lighthouse's metrics, namely Accessibility and Performance generate specific flags on each audit. Adjustments are made on each push that specifically address any issues.
Accessibility Testing
  • ChromeVox Extension was used to ensure that screen-reader accessibility standards are met. This was done by walking through the entire project with the screen-reader plugin enabled. Various adjustments were made following these tests. Notably, the tab-index order of nav elements, and changing refining HTML5 semantic elements for role clarity.
Browser Testing
  • Throughout the development of the project, in-browser dev tools were used to test for consistency across browsers. The browsers themselves were equally used for general use-case testing. The following browsers' per-device applications were accessed with an iPhone 11 Pro, MacBook Pro 15" and iPad Pro 12.9":
  • Chrome Version: 83
  • Firefox 82
  • Opera 72
  • Safari 14

back to top

Deployment

Credits

Content and Media

  • Product listings used to fill the database are random items gathered throughout the development of the project include url references in the product sources.

Acknowledgments

ESLint and Typescript Configuration

ReactJS and Typescript References

DRF with React/NextJS frontend References

License

MIT License Copyright (c) 2021 Joem Elias Sanez

back to top