/course-project-26

course-project-26 created by GitHub Classroom

Primary LanguageJava

UofT Secondhand Books MarketPlace

Hey there! Welcome to our project. Our purpose is to create a secondary hand book platform that connects students who wish to sell their book to potential buyers. Usually an E-commerce platform limits users to represent only one function - they are either the seller (store owner) or the buyer (customer). Our application creates a community that solves students' needs to buy affordable books for their new school term while also satisfying their needs to sell their old textbooks.

Features

Inventory and Shopping cart for Users

Users can add listings of the books they want to buy to their wishlist. This allows buyers to easily access the listings of the books they want to buy and monitor their status. Users can also add books to their inventory. This allows sellers to easily access the books they want to sell when creating listings.

Profile system

Users in our database system can now have access to their profile which includes their personal information as well as the login information. Also, users can now edit their first name, last name, and address. The information will be updated immediately. Moreover, if you don't want to be registered, you can still access our platform as a guest.

Database configuration (H2 Database)

H2 database is an open-source database written in Java programming language which supports querying data in standard SQL. It is very much lightweight and its JAR file is only 1.5 MB in size. It also provides a web interface called H2 Console to see the data.

The best thing about using an in-memory database is that we don’t have to do any configuration at all. We've' added a single property to the application file which enable the H2 console. Therefore, we can access the Web UI for our in-memory database via the following link: http://localhost:8080/h2-console. This URL is reachable only when our application is in a running mode. We will have a look at this console once we run our project.

H2 Database with Spring Boot

Java Spring Boot (Spring Boot) is a tool that makes developing web application and microservices with Spring Framework faster and easier. It provides a tool that allows you to set up a Spring-based application with minimal configuration and setup. With it, developers like us can get started quickly without losing time on preparing and configuring their Spring application.

So, how does H2 and Spring Boot combination work? Spring Boot is intelligent, it looks at the entities and creates the database and the tables. Spring Boot Application connect to the database h2 by Spring Boot Auto Configuration. It attempts to automatically configure your Spring application based on the jar dependencies that we've added.

Data Access Layer interface and the ConsoleUI

To access the database, we will be defining a simple JPA interface which provides its own helper functions. Once We run our application with a command-line runner, it will call the functions we defined from the JPA Repository interface and we will be able to see a simple output in our terminal.

It is important to note that the above UI console is accessible only for the time this application is running. Also, after the application has stopped, this console won’t be reachable anymore.

SOLID Principles

Single Responsibility Principle

Every class of our project follows the single responsibility principle such that they each have a single responsibility. We have isolated complex functionalities to different classes, so that each class only has one reason to change.

Open / Closed Principle

When we add new features to our code base, we extend upon the code we already have instead of modifying the software entities.

Liskov Substitution Principle

In our project, User and Admin classes are child classes of Account class. The objects User and Admin may be substituted without altering any of the desired properties of the program.

Dependency Inversion Principle

The Listingportal class in UI layer is used for getting input data and hand it into Service(Controller layer) classes. However, this action involves calling instances from Service classes, making the Listingportal class dependent on Service classes. High-level modules should not depend on low-level modules; instead, both should depend on abstractions. Therefore, we create Service interfaces. Then, we let Listingportal class depends on the Service interfaces, and make Services classes implement Service interfaces. Now, even though Listingportal class calls a method in the Service class, it is no longer depend on it.

Interface Segregation Principle

The use case Account manager is implemented as an interface to store the essential features when managing an account, including change username, change password etc. No realization class is forced to implement irrelevant methods of an interface.

The User Service interface is implemented as an interface to control users including getuserbyusername and deleteuserbyusername etc. No realization class is forced to implement irrelevant methods of an interface.

Clean Architecture

Our project is divided into different logical pieces, and they effectively communicate with each other. We aim to facilitate the development, deployment, operation and maintenance of the system by creating different layers of clean architecture that follow the dependency rule.

In clean architecture, entity classes are the highest level policies, meaning that they are the farthest from inputs and outputs. Our entity classes are "Account", "Admin", "Book", "Listing" and "User" classes. They are independent of each other. Our entity classes act as objects within our software system that embody sets of critical business rules (methods) operating on critical business data (variables).

For the usecase layer, we have various manager classes, each to manage a specific entity class.

Then, Service, Repository and Controller classes/interfaces are all in the third layer of clean architecture. Repository interfaces define how we retrieve and store data from the database, and are implemented by spring boot already. The ServiceImple classes implement Services interfaces and depend on Repository interfaces, so they interpret input data, and uses it to control the dance of the Entities. Ideally, the controller accesses entity classes through use case in order to maintain clean architecture - our controller uses service interface to obatain rules and data from entities.

Finally, consoleUI are in the fourth layer. They get the input data, then hand them in to Controller.

Overall, Our source code dependencies only points inwards. Nothing in the inner circle can know anything at all about something in an outer circle.

Design Patterns

We implemented the Builder design pattern in our project in Book and BookManager classes. As the builder, we used the Book class. And as the director, we used the BookManager class. The Book class knows the specifics about the book and it creates the necessary book with the given values. And when the user leaves some of the values blank and doesn’t put in each detail about the book like its size, its weight, etc., The BooksManager generates a Book without those details, so that the user can provide less details about the book when they want. This is necessary as a seller may not know every small detail about the book they are selling.

Another design pattern we implemented is Dependency Injection pattern by using autowired which is offered by springboot. For example, While constructing the UserController object, if there's no constructor or setter method to inject UserService, the container will use reflection to inject UserService into UserController.

Use of Github Features

We took advantage of a variety of features that Github offers for version control and collaborating as a team. We used features like branching, creating pull requests where we review each others code and resolve merge conflicts.

Branching

Each of us work on our own feature branch to develop new features to our project, providing us space to make changes without affecting the main branch. Whenever we make a completed change, we write concise but informative commit messages to help ourselves and future contributors to understand what changes the commit contains. After we commit our changes, we push our work to remote storage, so we can access our work from any device and our team can see each of our work, answer questions, and make suggestions.

Creating Pull Requests & Conducting Code Reviews

After we commit and push our feature branch to remote, we create a pull request by including a summary of the changes and what problem they solve. This feature allows us to ask each other for feedback on our changes before we merge into our main branch.

Pull request review is valuable for us in identifying potential issues in each other's code - we are able to comment on specific lines that cause problems, and everyone is able to see. It helps to address issues before merging to main, and it allows everyone to be aware of the added code and problems it may cause, so other members can avoid repeating the same problem with their features.

Resolving Merge Conflicts

Sometimes multiple members push changes to the same code which causes merge conflicts. We utilize the feature of resolving merge conflicts in Github to select the code we wish to keep before we merge into main. This feature gives us flexibility to create changes on any files in our project without the need to worry that the changes may conflict with another member's and will get lost during version control.

Refactoring and Code Smells

Shorten the methods in Listing portal by having helper methods.

Refactor some of the method's name and package's name.

< helper_sell_book > - a helper method for selling a book, which the seller will have to give the name, the desired price, and the author of the book.

< helper_listing_user > - a helper method for showing the listing portal of the books, and print out the seller's information, including the name, the email, and the address.

< helper_for_book_page_guest > - a helper method for showing the listing portal of the books for guests users which only includes information about the book

Reformat the code so that the code looks clean and easy to follow along with.

Testing

We have implemented many test cases for most of our Entity and Usecase classes. Also, we used the same packaging system in test files that we have in our actual code that made it easier to access and use the tests we have for our program. Some of these tests test basic functionalities of the program like getters and setters of entity classes while the other tests check more specific capabilities of the program such as removing books from a listing to check if the program gives an exception when an amount more than available is given and changes the Listing's status to "Sold out" when the number of books in the listing drops to zero.

Code Style and Documentation

For the code style, our code passes all the default code style formatting requirements that are in IntelliJ. We have used proper naming conventions for all of our classes, methods and variables. We also used proper indentation to increase the readability of our code. We have included documentation for all of our classes and most of the methods inside those classes throughout the program to be able to describe the function of each piece of code we have written in a more clear way. We used Javadoc to achieve this. This allowed us to also understand and check each other's code and build on them. These documentations include the purpose of the given code, its parameters and what it is meant to return or do.

Code Organization / Packaging Strategies

The packaging strategy we use for our UofT's online second-hand book store is packaging by the layers of the clean Architecture. We separate java files of different layer into different packages. For instance, we have the ConsoleUI, Controller, User cases and Entities packages that contain corresponding layer of java file. Since our program consists of many classes that belong to different categories but they also belong to the same layer in clean architecture, we also divided some layers by its categories. For example, even though controller and Gateway belong to the same layer, we separate them into its own packages, so that the files for gateways and controllers will not be that overcrowded and disturbing to find a certain file. Using the packaging by layer strategy not only contribute to our clean packages organization which could allow us to locate the class we want efficiently but also help us avoid some potential violations of the clean architecture that we might have along the way.

Accessibility

The 7 Principles of Universal Design

Equitable Use

We believed that our design is useful and marketable to people with diverse abilities, lIke what we mentioned in our specification. Our application creates a community that fulfills students' needs to buy affordable books for their new school term while also satisfying their needs to sell their old textbooks. We make our design appealing to all users (students.)

Simple and Intuitive Use

We believe that our design is easy to understand, regardless of the user's experience, knowledge, language skills, or current concentration level. The instructions given are detailed and well illustrated.

Tolerance for Error

Our second-hand book platform simply follow this principle by providing warnings of hazards and errors. The system will print error and show where the errors are when the given commands are wrong. This avoids Users from getting Error messages when they keep on using our system.

Low Physical Effort

Our second-hand book platform can be used efficiently and comfortably, without getting fatigued. It is simple and requires zero effort. It also allows users to maintain a neutral body position while using our system.

Size and Space for Approach and Use

In our design, we provide a clear line of sight to important elements by making them bold. Some important instructions and commands are also being highlighted, which provides convenience to users when using our system.

Flexibility in Use

In the future, we can provide different methods of use. Due to the fact that we currently use command line for our user interface, the flexibility of our application is limited.

Perceptible Information

We could improve a lot more on our User Interface in the future. We could provide adequate contrast between essential information and its surroundings. Furthermore, we could provide compatibility to people with sensory limitations by advancing the techniques.

Target Customer

Our Target customers will simply be the students. For now, it has been narrowed down to the UofT students who want to buy and sell second hand books. Our platform is different from other E-commerce platform which limits their users to represent either a seller or a buyer.

Our second-hand book platform is specifically designed for students who want to buy second-hand books or sell their old textbooks. Therefore, our system is less likely to be used for students who want new textbooks. Also, our system is less likely to be used for students who are in a hurry on getting their textbooks. It is an online platform, so you will need to wait for your books just like buying stuff online.

Major Design Decisions

Seller and Buyer classes are originally the child class for User Class. However, we believe that a user can either be a seller or a buyer, so User class and Admin class are now the child classes for Account class. The second major decision will be we choose to use Spring Boot and command line to present our system. Spring Boot is an open source Java-based framework used to create a micro Service.

These major design decisions lead us to have four strategy design pattern. The first one is MVC (model, view, controller). Then we have the builder. Next is the spring boot. Lastly, is the dependency injection (autowire) and interface (repository).