Supp is a user-friendly app that sorts users by distance and filters them by common interests/hobbies. Users connect with each other via private chats.
https://www.youtube.com/watch?v=yG3gEc1bC60&feature=youtu.be
- Clone repo.
- Install dependencies
npm install
. - Start your server
npm start
.
Github: Front-end https://github.com/AlbertCarreras/supp-client
React with Redux & Redux Thunk
- App Structure The app has 4 major subfolders: Adapters, Actions, Reducers, and Components.
- Adapters folder has files that act as reusable libraries.
- Actions & Reducers has subactions and subreducers files that work together in Redux.
- Components has all React components organized by major functional section (interests, user list, and conversations).
-
Redux and Redux Thunk are key for all the asynchronous requests happening at a time. Redux helps keep and handle all the state from a single place while Thunk is indispensable to allow the app to work.
-
Geolocation Web API for obtaining user’s current location coordinates
I used W3C Geolocation standard to request the browser the geolocation of the user. The request returns the latitude and longitude of the user which is persisted and used to georeference other nearby users.
I research the best approach to obtain the location, and decided to go use the Geolocation Web API after reading the following:
"Some browsers use IP addresses to detect a user's location. However, it may only provide a rough estimate of a user's location. The W3C >> approach is the easiest and most fully-supported so it should be prioritized over other geolocation methods." https://developers.google.com/maps/documentation/javascript/geolocation
-
Custom CSS, SASS and Semantic UI elements for front-end design App responsiveness but website first design. Most of the app is styled using custom CSS. Some elements such as the icons and the modals are Semantic UI elements. I decided to use Semantic UI elements because the library facilitated the front-end styling in previous projects. Moreover, I had in mind using modals and liked the examples from the library. However, I would refactor the modals, use custom CSS. The app uses SASS as CSS preprocessor. CSS code has been organized in different files and imported in a single app.scss file.
-
Bad-words (package) for filtering profane language
The search bar and the chat input bar do not allow to look for profane language. It is a very simple package but easily offers functionality to prevent some user behaviors.
- Clone repo.
- Install Gems
bundle install
. - Set up Database
rake db:create
, then runrake db:migrate
. - Seed Database
rake db:seed
(>10 Lines). - Create master.key file with 32 characters in it within config folder.
- Start your server
rails s
.
Github: Back-end https://github.com/AlbertCarreras/supp-server
Rails API with with serialization and Postgres
-
Active Storage for photo storage connected to AWS S3 in production
I used the Rails built-in solution -Active Storage- to attach files to your Active Record models. Active Record also facilitates uploading files to a cloud storage service like Amazon S3.
The server is hosted in Heroku. In order to deploy to Heroku and upload images, you need to create an AWS S3 account to store the files.
Great resources for implementing Active Storage:
-
Action Cable for live private chat feed updating and connected-user indicators
Action Cable is the built-in websocket implementation in Rails. In order to facilitate the websocket connection from the React front-end, I decided to use react-actioncable-provider (package).
Live private chats I initially implemented a simplified version of private chats that I posteriorly refactored into a more secure implementation. First, users were connecting to a general channel. New conversations were broadcasted to all users with serialized information about the 2 users to whom the conversation belongs to. In the front end, the received conversation was only displayed if one of the users was the logged-in user.
When implemented user identification on the websocket connection, each user started connecting to a dynamic private user channel based on each user's id. Now, new conversations are just broadcasted to both sender/receiver users. On the client side, we still check if the conversation's users match the actual logged-in user.
Connected users indicators (blue-green dots) When a user logs in, they subscribe to a presence channel changing their status to active in the database -reversely when disconnecting/unsubscribing. The change in status gets broadcasted to all subscribers.
Great resources for implementing websockets in React-Rails apps:
-
Knock for JSON Web Token authentication and cookies for websocket-connection user identification
I followed the 3.1.1 Connection Setup implementation guidelines from the Action Cable Overview documentation. However, I used JWT tokens for authorization. See Notes on technologies for further implementation details and thoughts.
-
Geokit (gem) for calculating surrounding users to connected user distance
Users are sorted by distance in the front-end. The server responds with an array of users including the distance to the logged-in user. The gem calculates the distance based on the coordinates of each user obtained via Geolocation Web API.
Database. The following image shows the database tables and relationships.
As I was building the app, I made some decisions that down the project happen to become some technical debt. I decided to install Knock because it was a light auth package. I considered Devise but I found it too comprehensive for the scope of the project.
Another decision was to use "React-actioncable-provider" to facilitate the implementation of Actioncable in the front end.
When implementing the websockets for live active-user indicator, I found out that Actioncable uses cookies to identify the user in the server (channels folder) and does not have access to the controllers folder. At first, I made the front-end create a cookie (https://developer.mozilla.org/en-US/docs/Web/API/Document/cookie)containing the token so it could be picked up in the server and identify the user which works in local/development. However, this solution does not work in production. I deployed front-end and backend in different domains and cookies don't work cross-domain.
Having used Devise, I could have scoped warden (https://www.sitepoint.com/create-a-chat-app-with-rails-5-actioncable-and-devise/).
Having not used "React-actioncable-provider", instead hard-coding all the action cable implementation, I could have passed some user information in the headers.
The quickest -and not optimal solution- was to pass the actual userId as a query parameter.
- Implementing some tests (Rspec in the back end and Mocha in the front end)
- Improving the algorithm for returning users by proximity and interests so it works with large user datasets.
- Pinned users in the central containers as a way of saving friends.
- Blocking users.
- Deleting/Archiving conversations.
- New notification indicator.
- Location refresher button.
ALBERTO CARRERAS
- acarrerasc @gmail.com
- https://github.com/AlbertCarreras
- https://medium.com/@a.carreras.c/
- https://www.linkedin.com/in/albertcarreras/en
I also want to thank Mike Chen () and Matt Lawford (https://www.linkedin.com/in/matthew-lawford-216b67b2/) for their technical support during the project development.
This project is licensed under the MIT License - see the LICENSE.md file for details
I'm a board game geek. I love playing Dominion, Carcassonne, Star Realms, 7 Wonders, and so forth. I landed in NYC 6 years ago now and have found that it takes a lot of effort to meet and/or "connect" with people to play board games: you have to go to board games stores, your friends live really far, and more.
At the same time, NYC has so many passionate people out there and is not difficult to find people who love the same things you love. However, despite having apps such as Meetup, I wanted to create a more genuine app to find people who share my same hobbies or passions and who actually live or are regularly nearby; start a conversation, and plan to meet to do those activities or experience them together.
Supp is an app to enhance the experience of meeting new people and making new friends.