An Online Marketplace Application
ThriftHub is a platform that allows users to post second-hand goods and connect with vendors interested in such items.
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.
-
configure:path
: ExecutesconfigurePath.js
, to ensure environment paths are correctly set up for the backend. -
configure:backend
: RunsconfigureBackend.js
, to prepare the Django backend environment by setting up required Python packages and configurations. -
setup:backend
: Sequentially runsconfigure:path
andconfigure:backend
to prepare the backend environment. -
start:backend
: Initiates the backend server by invoking the start script defined in the backend'spackage.json
. -
start:frontend
: Starts the frontend application by calling the start script defined in the frontend'spackage.json
. -
dev
: Utilizesconcurrently
to run both the backend and frontend servers simultaneously, allowing for concurrent development.
backend/package.json
Script
start
: Executesstart.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.
frontend/package.json
Script
start
: Launches the frontend application using theexpo start
command.
To set up the project as a monorepo:
- Ensure Yarn is installed on your system.
-
Clone the Repository:
git clone [email protected]:israelias/thrifthub.git cd thrifthub
-
Install Dependencies:
yarn install
This command installs all dependencies for both backend and frontend projects, leveraging Yarn Workspaces for efficient management.
-
Configure the backend
yarn setup:backend
This script prepares the backend environment by setting up required environment variables and configurations.
-
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.
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
infrontend/src/config.ts
to variably match the backend server's URL for development and production environments.
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
- 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.
- 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.
- 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.
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.
-
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.- 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.
- 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
andun-save
a product listing. (Registered User)
-
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 tosign up
,sign in
andsign out
in order to conditionally access existing features. A user is based on theuser
model, which requiresusername
,email
andpassword
for new users and only the latter two for existing users. For security, only usernames are stored inlocal storage
whiletokens
are stored in memory. To ensuresign out
across multiple open windows, a logout event triggers astorage event listener
which clears tokens and usernames in the memory of the current window.- 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.
- 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)
-
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.- 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.
- 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)
- The website should feature the ability to group any and all existing products into named collections aka
bulk deals
. A collection is based on thecollection
model, which is a name and a list of product listings. Each field is represented as either aform
field or aheader
followed by aul
depending on whether a collection is in edit or display mode. - Ability to create, edit and delete a named collection. (Registered Owner)
- Ability to add and remove listings to/from a collection. (Registered Owner)
search_text
indices are attached to the title and description fields of theproduct
cluster. The frontend UI of the search feature is designed to return product listings that match a query. Additionally, the ability to query bytags
andproduct_type
is enabled by filtering these existing fields in the database.- 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
orun-save
product listings from search. (Registered User) - Ability to perform
crud
operations on product listings from search. (Registered Owner)
- Please visit the frontend sub directory for details on ReactNative Typescript frameworks and libraries.
- Please visit the backend root directory for details on Python-Django frameworks and libraries.
- 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 andpush
to GitHub. - GitHub: GitHub is used to store the project's code and directory upon concurrent
push
es via Git. - Adobe InDesign: Adobe InDesign is used to mock wireframes.
-
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.
- The screen for
- User continues.
- User arrives at home page.
-
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.
- The sign-up form appears.
- User continues.
- Review:
- A user is able to securely create an account.
- User clicks
-
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.
- The sign-in form appears.
- User continues.
- User clicks
-
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.
- A form appears.
- User continues.
- Review:
- A user is able to successfully create a product listing.
- User clicks
-
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.
- User arrives at
-
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.
- A form appears.
- User continues.
- Review:
- A user is able to successfully edit their product listing.
- User clicks
-
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.
- A form appears.
- User continues.
- User clicks
-
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.
- The
- User is redirected to their
profile
page.
- A form appears.
- User continues.
- User clicks
-
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.
- A
- User continues.
- User arrives at
-
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.
- The
- User is redirected to their
profile
page. - The updated collection appears.
- A form appears.
- User continues.
- User clicks
-
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.
- A form appears.
- User continues.
- User clicks
-
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.
- A chat or message option appears with the buyer.
- User continues.
- User clicks
-
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.
- A payment gateway appears.
- User continues.
- User clicks
-
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.
- User clicks
-
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.
- User clicks
-
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.
- A payment gateway appears.
- User continues.
- User clicks
-
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
ortags
. - User clicks
Show All
to see most recent products. - Pagination buttons appear, and the user clicks
Next
for additional pages.
- User continues.
- User clicks
-
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.
- User clicks
-
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.
- A collection named
- User continues.
- User clicks
-
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.
- User clicks
-
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.
- User clicks
- 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 installingnpm 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.
- 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.
- 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
- The project frontend is written in Typescript ReactNative, bootstrapped with Expo and deployed with Snack. The backend is written in Python, developed with Django Rest Framework to serve a restful database via PostGres. Please visit the frontend root directory for details on deployment steps. Please visit the backend root directory for details on deployment steps.
- Product listings used to fill the database are random items gathered throughout the development of the project include url references in the product sources.
- ESlint Typescript with Prettier
- Typescript, ESLint & Prettier with Airbnb style guides on VSCode
- Airbnb Javascript style guide — Key takeaway -Config ESLint, Prettier in Typescript React App
MIT License Copyright (c) 2021 Joem Elias Sanez