/discord

Discord + ChatGPT + Google Trends + JavaScript Evaluation

Primary LanguageJavaScript

Discord

This is a powerful discord bot that can interact with ChatGPT, fetch data from Google, and execute JavaScript code safely!

Before you continue, please give credit and consider hitting the star button.

Running

Previously, node . was used to start the bot. After adding an express server, the best way to run the bot is using node server.js. The scripts.dev property in the package.json file has been set to node server.js to maintain consistency and programmnig principles.

Package Files

The main property in the package.json file determines the . file selector and defaults to index.js. If the main property is set to server.js, then node . will run the server.js file.

Events

This bot has an effective event handler. The basic events are listed below.

  • debug
  • error
  • interactionCreate
  • messageCreate
  • ready

Slash Commands

This bot mainly operates on slash commands. Currently, there are 3 categories with commands:

  • ai
    • chatgpt
  • bot
    • help
    • ping
  • fun
    • trivia
  • google
    • trend
  • utility
    • avatar
    • embed
    • profile
    • reactroles
    • reactembed

Data Storage

This bot stores reaction roles information by writing to a reactions.discord file which contains information about the message.id, roles, and emojis. The problem with storing data in files is that you will lose reaction roles between environments during hosting. The reactions.discord file will probably reset after every deployment.

Try to improve on the reaction roles by using PostgreSQL, MongoDB, or crud:api.

{crud:api} is a simple cloud database with a RESTful API based on the standard CRUD operations so you can add cloud storage to your project quickly and easily

Linear Commands

This bot can execute script commands that communicate directly with the server. Read the security section for more information.

Dashboard

This bot comes with a dashboard as well. The guide provides further information about OAuth2.

  1. Add redirects in the general OAuth2 tab https://discord.com/developers/applications/${client_id}/oauth2/general
  2. Configure the URL Generator by selecting scopes
  3. Adapt the code in the guide to your needs
// MUST BE ASYNCHRONOUS
app.get('/', async (req, res) => {
  const { code } = req.query
  if (code) {
    try {
      const params = new URLSearchParams({
        client_id: process.env.client, // CLIENT_ID
        client_secret: process.env.secret, // CLIENT_SECRET
        code,
        grant_type: 'authorization_code',
        redirect_uri: `http://localhost:${port}`, // REPLACE WITH YOUR REPLIT OR GITPOD URL
        scope: 'identify guilds guilds.join guilds.members.read', // separate scopes with spaces
      }).toString()
      console.log(params)
			const tokenResponseData = await request('https://discord.com/api/oauth2/token', {
				method: 'POST',
				body: params,
				headers: {
					'Content-Type': 'application/x-www-form-urlencoded',
				},
			})

			const oauthData = await tokenResponseData.body.json()
			console.log(oauthData)

      if (oauthData.error === 'invalid_grant') {
        console.log(`OAUTH ERROR: ${oauthData.error}`)
        route(req, res, { title: 'Discord', message: 'Hello, World!', auth: url })
      } else {
        const access = oauthData.access
        const refresh = oauthData.refresh
        const userResult = await request('https://discord.com/api/users/@me', {
          headers: {
            authorization: `${oauthData.token_type} ${access}`,
          },
        })
        const user = await userResult.body.json()
        console.log(user)
        route(req, res, { title: 'Discord', message: 'Hello, World!', auth: url, user }) // discord redirect route
      }
		} catch (error) {
			console.log(error);
		}
  } else {
    route(req, res, { title: 'Discord', message: 'Hello, World!', auth: url }) // default route
  }
})

Gitpod workspace URLs may change from time to time. The process.env.GITPOD_WORKSPACE_URL exposes the workspace URL that can be used to find the server URL. Ternary operators are recommended if you are thinking of hosting.

Security

It is important to note that new Function() constructors should always be used instead of eval(). Never pass discord interfaces directly into the function constructor. Make sure to read the Discord.js Documentation when adding features to the script command in the interactionCreate.js file.

Avoid ✕

The Client, Guild, and RoleManager classes can be abused easily.

(new Function('interaction', `interaciton.reply({ content: interaction.guild.client.token })`))(interaction)
// exposes MessageInteraction, Guild, and Client.token

Do ✓

The user is limited to the context which only contains a reply function.

(new Function('ctx', `ctx.reply({ content: 'Hello, World!' })`))({ reply(message) { interaction.reply(message) } })
// only replies to MessageInteraction

Nullifying the process object prevents hackers from touching your token!

(new Function('process', `return process.env.token`))(null)
// cannot read property 'env' of null

During debugging the client will log the bot's token unless you prevent it by using a conditional or by removing the console.log from the events/debug.js file completely.

Open AI

Open AI functions are stored in the openai folder which contains a chatgpt.js file. Most interactions with Open AI are asynchronous and can take over 3 seconds to complete. This means that Collector objects are required for longer completions. The openai package is available on NPM.

Make sure to grab your API key from Open AI and add it to your environment variables.

const { Configuration, OpenAIApi} = require('openai')
require('dotenv').config()

const configuration = new Configuration({
  apiKey: process.env.openai // get your key from the website
})

const openai = new OpenAIApi(configuration)

Completions Endpoint

Make sure to set the max_tokens property to increase the size of the response. Larger responses take more time.

async function say(instruction) {
  try {
    return await openai.createCompletion({
      model: "text-davinci-003",
      prompt: instruction,
      temperature: 0,
      max_tokens: 1024, // answer length is around 3 * max_tokens characters
    })
  } catch(error) {
    if (error.response) {
      console.error(error.response.status, error.response.data)
    } else {
      console.error(`Error with OpenAI API request: ${error.message}`)
    }
  }
}

The say function can be exported and used in a slash command. Collectors are important because they can last up to 15 minutes without a response.

const user = interaction.user
const prompt = interaction.options.get('prompt').value
const seconds = 5

const embed = new EmbedBuilder()
  .setTitle('ChatGPT')
  .setDescription(prompt)
  .setThumbnail('https://brandlogovector.com/wp-content/uploads/2023/01/ChatGPT-Icon-Logo-PNG.png')
  .setColor('#12a37f')

return interaction.reply({ content: `Generating your response in ${seconds} seconds!`, embeds: [embed] }).then(m => {
  const collector = m.createMessageComponentCollector({ time: seconds * 1000 })

  chatgpt.say(prompt).then(completion => {
    console.log(completion.data.choices)
    return interaction.editReply({ content: completion.data.choices[0].text })
  })

  collector.on('end', (collected, reason) => {
    if (reason === 'time') {
      console.log('timeout')
    }
  })
})

Hosting

Hosting can be a problem because discord ratelimits replit. After 24 hours your repl is bound to go offline despite Uptime Robot pinging the address of your repl. Other reliable options are usually paid but Int Fract has found free alternatives.

Render

Render provides most of the resources required to host a discord bot for free. Although, there are some things to remember about environment variables, uptime, and node versions.

  1. Create an account and log in with GitHub!
  2. Click New!
  • Select Web Service which is recommended for your discord bot!
  • Select Static Site for a discord bot dashboard!
  1. Connect to GitHub and select a repository!
  2. Open the advanced options dropdown and add environment variables!
  3. Configure your deployment!
  • Your build command which is usually npm install or npm run build is run once before the app goes online!
  • Your run command which is usually node . or npm run dev is used to start the app!

If you encounter an ERROR after the run command, read the error message carefully and troubleshoot.

  • The missing node:events module error suggests that the current version of node is too old!
  • The InvalidTokenError error suggests a few things.
    • You have provided the wrong token!
    • You have made a mistake in the .env file by using quotes!
    • You have not configured environment variables on your render web service!

The discord bot will go offline after a few minutes of inactivity unless it is equipped with an http or express server that can use the PORT environment variable.

const port = process.env.PORT || 3000 // uses render.com PORT if available or defaults to 3000
require('http').createServer((req, res) => res.end(process.version)).listen(port)

Fly

Fly.io offers limited features for free if you have a credit card. Fly only asks for a credit card to verify deployments. This can be solved by searching google for random credit cards with expiry dates. Honestly, it takes a bit of effort.

The ba.sh file contains some scripts for a fly deployment although the docs explain the process clearly. Most online development platforms like Gitpod and Replit use linux. The commands may vary between linux, windows, and mac. The following code is for Visual Studio Code Browser Gitpod users.

curl -L https://fly.io/install.sh | sh
# Wait for the previous command to finish and check for additional messages
export FLYCTL_INSTALL="/home/gitpod/.fly"
export PATH="$FLYCTL_INSTALL/bin:$PATH"

Deploying the bot with fly launch will create files called fly.toml and .dockerignore that have already been included in this repository. Make sure to remove the fly.toml and add a new one.

The bot can be launched with environment variables through the shell.

flyctl secrets set token=YOUR_BOT_TOKEN client=YOUR_CLIENT_ID

Express Server

This bot comes with an express server and a custom page renderer!

Enclosing embedded code in #{} will allow you to evaluate JavaScript locals from the server!

There is no need to use complex frameworks like ejs or pug to render your html views. A bleeding-edge router is available at server.js!

Rendering Example

You can also use static components using the include(file: string) function. Static components are any .html files that are not named index.html.

<!DOCTYPE html>
<html lang="en">
<head>
  <title>#{ title }</title>
  #{ include('head.html') }
</head>
<body>
  <main id="app">
    <h1>DISCORD BOT</h1>
    <p>Check out the <a href="/status">status</a> page!</p>
    <h2>DASHBOARD</h2>
    <p>Unlock the dashboard by <a href="#{ auth }">verifying yourself</a>!</p>
    <div class="discord">
      <span class="avatar"></span>
      <span class="user">#{ user.username + '#' + user.discriminator }</span>
    </div>
  </main>
</body>
</html>

Erros are suppressed and logged. Bad templates like an undefined user will simply return blank stirngs. Ternary operators can be used to adjust the behaviour of bad templates.

Pages

Do not place any .html files in the public directory. Pages are rendered from index.html files located in the views folder.

Debugging

Env

Environment variables can sometimes be a pain when deploying or using Gitpod. Firstly, make sure you are using the same workspace instead of creating new workspaces and cloning the same repository again. Your .gitignore file should and usually prevents environment variables from being exposed to other programmers. This means that you need to redeclare your variables during hosting.

If your environment variables are not updating, it is because you have set global environment variables through the Gitpod UI.

Replit allows direct access to environment variables, but editors like visual studio code require you to install the dotenv package.

require('dotenv').config() // add .env file variables to process.env
console.log(process.env)

OAuth2

The invalid_client and invalid_code are the msot common OAuth2 errors.

  1. Check your client id and secret at your developer application dashboard
  • Reset your client secret if necessary
  • Update the .env file and variables
  • Make sure you have added your redirect URLs for your development and hosting environment
  1. Check the client_id and client_secret parameters
  2. Use the latest code after connecting with discord

Multiple Gitpod redirect URLs may be necessary because Gitpod workspaces can change machines or regions. The subdomain is in the format ${port}-${user}-${repository}-${id}.ws-us${number} for western US. regions.

Currently, the bot is encountering the 401 error.

{ message: '401 Unauthorized', code: 0 }

Replit Bot Login

This message always appears, but it has something to do with fetch which is installed by replit. It is better to use v2 because it integrates with CJS modules.

(node:281) ExperimentalWarning: stream/web is an experimental feature. This feature could change at any time
(Use `node --trace-warnings ...` to show where the warning was created)

If the bot fails to log on then kill the container in the shell.

kill 1

The events/debug.js file might also help with this issue.

Embed Builder Errors

Try and catch blocks need to be used to prevent the bot from crashing due to faulty user input. This happens when users use poor formatting on their embed options. Invalid URLs and hex codes will trigger these erros as well.

try {
  embed[`set${x}`](v)
} catch (e) {
  console.log(e)
}

Wrapping functions also works. Once the erros are caught the embed will work fine but without the parts that generated the errors.

function safe(fn) {
  try {
    fn()
  } catch (e) {
    console.log(e)
  }
}

safe(() => embed[`set${x}`](v))

Visual Studio Code FJS Files

The .fjs file can be configured using file associations in the settings page of Visual Studio Code.

Pasting the code below into settings.json also works!

{
  "files.associations": {
    "*.fjs": "javascript"
  }
}