Fauna is at the peak of serverless databases and providing scalable and secure infrastructure to organizations and individuals with easy integration with various frameworks. This article will build a Twitter counter with Fauna, Netlify, and the Nuxt framework. I used Fauna as my serverless database of choice because of its high scalability and consistency.
Audience
This article will be easier to follow if you have intermediate Vue.js/Nuxt.js knowledge. You should also know how to deploy a Nuxt project on Netlify.
Goals
After reading this article, you will know how to use a serverless database like Fauna with a Nuxt jamstack site to create a twitter counter .
What We Need
The purpose of this tutorial is to implement a kind of Twitter reaction counter in a blog post. I will be using the personal portfolio website I quickly built as a reference during this tutorial. In the next paragraph, we will set up Netlify and Fauna. We will be using the Nuxt content module as our content management system to write our blog posts and the blog slug to identify the blog post and properly route each article. Slug will be of great use as we implement the reaction system.
Set-up Project
First, we will set up a Fauna account. To start, navigate to their official site, create an account with them (here), and then log into the account and create our database for the project.
We need a way to store the reaction counter for the articles in our blogs, so we add a collection in our new database to do this.
Create an index that allows for easy organization and retrieval of documents by attributes according to their queried way
From the image above, we create an index named โreaction_countโ under the reactions collection created to use as the source collection. In the terms field, we have added data.slug as one of the terms that can be searched.
Using our Fauna configuration, we generate our Fauna API key that we can use in our Nuxt application to query our reaction_counter_database
successfully. To do this, we go to the dashboard and click on the security section to generate the API.
When we generate the API, we have to store it somewhere safe, like our environment variable file as YOUR_FAUNA_SECRET_KEY
, so that it cannot be accessed by a third party.
Installing Fauna in our Nuxt App
Now we can set up Fauna in our Nuxt application, by installing the fauna package.
npm install --save faunadb
When setting up my Nuxt project, i had already selected the @nuxtjs/axios
plugin in making API request, so keep this in mind. You can manually install it by running:
npm install @nuxtjs/axios
Also, in your Nuxt config file you can add in the modules
export default {
modules: ['@nuxtjs/axios']
}
In the .env
file created at the root of our application, we set our fauna API key
YOUR_FAUNA_SECRET_KEY = THE_KEY
We want to be able to to test our app in development so we set configuration for the local and development deployment link like this in our Nuxt config file
publicRuntimeConfig: {
axios: {
baseURL: process.env.NODE_ENV === 'production' ? process.env.BASE_URL || 'http://localhost:8000/' : 'http://localhost:8000/',
}
},
Now if we run the app locally we have it running at port 8000
, this is to avoid any errors when running our serverless function
Securing Fauna's Secret Key on Netlify
We want to hide our Fauna API secret from prying eyes. Use of Netlify functions to protect your Fauna API secret. Netlify functions, based on AWS lambda, give you powerful abilities when building serverless functions allowing you to auto-scale your projects. In simple terms, they are functions that run on the server. To do this, you can visit Netlify and add the environmental variables we defined earlier.
We store our Fauna secret key here to query our Fauna database when we deploy successfully. After setting up Netlify, Fauna, and Nuxt, we can dive into writing code. Yayy!!
Writing our Fauna Functions
When our page loads, we want to fetch all the reactions on our blog and want to be able to increase the reaction count. The Fauna functions are what we will focus on implementing for now.
Write a function that allows us to fetch our reaction counter, but first, create a functions folder at the root of our Nuxt project, and in this folder, we create a fetch_reactions.js file to write our fetch reaction functions.
Note: Once deployed, Netlify automatically locates the functions folder where all our serverless functions are located.
const faunadb = require('faunadb');
exports.handler = async (event) => {
const q = faunadb.query;
const client = new faunadb.Client({
secret: process.env.YOUR_FAUNA_SECRET_KEY,
});
const { slug } = event.queryStringParameters;
if (!slug) {
return {
statusCode: 400,
body: JSON.stringify({
message: 'Blog slug not available',
}),
};
}
const doesDocExist = await client.query(
q.Exists(q.Match(q.Index('reaction_count'), slug))
);
if (!doesDocExist) {
await client.query(
q.Create(q.Collection('reactions'), {
data: { slug: slug, reactions: 1 },
})
);
}
const document = await client.query(
q.Get(q.Match(q.Index('reaction_count'), slug))
);
return {
statusCode: 200,
body: JSON.stringify({
reactions: document.data.reactions,
}),
};
};
Several things are going on above; we initialized Fauna and set up the Fauna client with our secret password. We check if the bog slug is provided as a query parameter, If it isn't, we return a status code of 400. After doing this, we check if the Fauna document that we created exists. If it does not exist, we create a new document with our reaction collection, and set the initial reaction count to 1
using our slug as a unique identifier. We send a query for the reaction_count index we created and retrieve our reactions and return the reaction count of the blog.
Increase Reaction Count Function
The primary purpose of this function is to increment the reaction count in our blog post. As we will see below, it is almost the same as the fetch reaction function. The difference is that we are increasing the reaction count.
Note: This function is the function that will be called when a reader on the blog post clicks the reaction button.
To do this, we create a new file in our functions folder called increment_reactions.js and paste the code.
const faunadb = require('faunadb');
exports.handler = async (event) => {
const q = faunadb.query;
const client = new faunadb.Client({
secret: process.env.YOUR_FAUNA_SECRET_KEY,
});
const { slug } = event.queryStringParameters;
if (!slug) {
return {
statusCode: 400,
body: JSON.stringify({
message: 'Blog slug not available',
}),
};
}
const doesDocExist = await client.query(
q.Exists(q.Match(q.Index('reaction_count'), slug))
);
if (!doesDocExist) {
await client.query(
q.Create(q.Collection('reactions'), {
data: { slug: slug, reactions: 1 },
})
);
}
const document = await client.query(
q.Get(q.Match(q.Index('reaction_count'), slug))
);
await client.query(
q.Update(document.ref, {
data: {
reactions: document.data.reactions + 1,
},
})
);
const updatedDocument = await client.query(
q.Get(q.Match(q.Index('reaction_count'), slug))
);
return {
statusCode: 200,
body: JSON.stringify({
reactions: updatedDocument.data.reactions,
}),
};
};
After we query our database, we increment the reaction counter by 1
by running an update in our database.
Using Our Functions In Our Nuxt App
Using the functions we created earlier, we will create a Twitter love reaction button using SVG and assign a click event. When we click on the love reaction, a function runs that accesses our serverless functions that fetch and increment the love reaction.
Note: When we deploy our application, our functions can be located at .netlify/functions/OUR_FUNCTION_NAME
In our component folder in our application, we will create a TwitterReaction.vue component where we will house our love reaction and button.
// TwitterReaction.vue
<template>
<div>
<button @click="addReaction" class="focus:outline-none">
{{ initialReaction }}
<svg
class="w-6 h-6"
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
xmlns="http://www.w3.org/2000/svg"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M4.318 6.318a4.5 4.5 0 000 6.364L12 20.364l7.682-7.682a4.5 4.5 0 00-6.364-6.364L12 7.636l-1.318-1.318a4.5 4.5 0 00-6.364 0z"
></path>
</svg>
</button>
</div>
</template>
We can see above that we have addReaction function that runs when we click the button. The following function below runs in our script:
// TwitterReaction.vue
<script>
export default {
data() {
return {
initialReaction: null
}
},
async fetch() {
const { data } = await this.$axios.get(
`/.netlify/functions/fetch_reactions?slug=${this.$route.params.slug}`
)
this.initialReaction = data.reactions
},
fetchOnServer: false,
methods: {
addReaction() {
this.initialReaction++
this.incrementLikes()
},
async incrementLikes() {
await this.$axios.post(
`/.netlify/functions/increment_reactions?slug=${this.$route.params.slug}`
)
}
}
}
</script>
What happens above is we set up a reactive reaction counter initialReaction
, and using the fetch_reactions functions we fetch the reactions and set it to our initialReaction
. Since we are using static mode, we set fetchOnServer
to false
to ensure the fetch hook is called whenever our component is mounted.
Then whenever a reader clicks on the reaction button, we run the addReaction
method, which increments our reaction count.
Conclusion
Yayy! We have finally come to the end of the tutorial. In this tutorial, we learned how to use a serverless database in a jamstack site, we also looked at setting up Netlify and hiding API keys. In the end, we created a โTwitter-likeโ reaction counter.
If you have any questions, feel free to send me a message on Twitter.
Resources
- Read more about Netlify Functions
- Indexing in Fauna
- Github Repo
Top comments (0)