KRMA is a web application where users can place media (either photo or desired text) into KRMA in exchange for a karma key. Keys can be used to fetch an item out of the karmic cycle.
Access the live site here.
We aimed to provide an online experience that takes inspiration from Geocaching — an outdoor activity involving searching for hidden containers that are stashed around the world. In one of the variants of Geocaching, the containers house small trinkets that finders can take, but the convention is to leave something in its place. Although we didn't emulate the exploration aspect of geocaching, we developed an application that preserved the essence of a one-for-one exchange.
- Download the zip from github
- Open terminal
npm install
for Mongoose/Express/AWS dependenciescd frontend
and runnpm install
for React/Node dependenciescd ..
thennpm run dev
to start frontend and backend servers (note: you will need the proper keys_dev.js in config)- localhost3000 should auto-generate in your main browser
MongoDB is a popular document-based NoSQL database. For KRMA, we used the database to store collections of karma documents and user documents. As a setup decision, each krma document contains two references to two user documents: the owner of the karma and the uploader of the karma. Each karma also contains a link directing to an image hosted on S3.
Express is a web app framework for Node, and the means by which we were able to route our frontend components correctly.
React is used to create and render the front end of our application - the UI (user interface). It is an advanced, abstracted version of HTML written in Javascript, which creates a virtual DOM where changes take effect almost instantaneously without a hard refresh. It consists of components that hold mutable state, and allows for changes to render via the use of props (delivered by the container). Redux combines the dispatched actions, reducers, and store altogether to change the state of components in the UI. With these front-end libraries, we are able to provide the user a fast and seamless web browing experience.
Amazon Web Services provides many cloud-based services, including picture uploads. We utilized S3 as the main storage for the photo/gif media that were uploaded, and stored the resulting S3 link in Mongo.
router.get('/new/:id', (req, res) => {
Treasure.countDocuments({ ownerId: null }).exec(function (err, count) {
var rand = Math.floor(Math.random() * count)
Treasure.findOne({ownerId: null, reported: false}).skip(rand)
.then(treasure => res.json(treasure))
.then(() => {
User.findByIdAndUpdate(
{ _id: req.params.id },
{ $inc: { keyCount: -1 } },
{ new: true },
)
.catch(err => res.json(err))
})
.catch(err => res.json(err))
});
});
We needed to add functionality such that the user could only receive a random karma if the karma has never been pulled before. This is consistent with our original goal of mimicking GeoCaching in that once a karma is claimed only the recipient can access it. We implementented this logic in the backend, by first cycling through documents with no ownerId, then selecting a random document, and finally setting the the owerId to be equal the currentUserId that we pass through the params.
const Admin = ({ component: Component, path, loggedIn, user, ...rest }) => (
<Route {...rest} render={(props) => (
(loggedIn && (user.email === '[email protected]')) ? <Component {...props} /> : <Redirect to="/main" />)}
/>
);
In addition to the standard Protected and Authorized routes, we had to implemented an Admin route to hide certain functionality from an average user. We built this logic by checking across two criterias: 1) the current user is loggedin and 2) the current user has the approved Admin email.
To implement user auth securely, we used bcrypt to securely hash a user's password before storing it to Mongo and jsonwebtoken to generate/confirm a users' bearer token.
Inside the left portion of the Treasure Island page is a menu where you can upload a photo, quote and see your key count. Keys are used to open the treasure chest for various loot.
Admins can see all the flagged content
- We want to give the user the ability to pull a treasure based on a criteria, ie pull a random book request or a quote or story, etc. This will require a small refactor of the treasure document model.