Added ToS verification, key system demo, embeds

This commit is contained in:
nsde 2023-07-25 03:39:17 +02:00
parent d2857910d3
commit b69d8bd67e
9 changed files with 165 additions and 34 deletions

View file

@ -1,3 +1,5 @@
# nova-discord
🤖 Discord bot for project Nova, built using [NextCord](https://github.com/nextcord/nextcord) for Python.
You can find more information about the bot and how it works on [https://nova-oss.com/novacord](https://nova-oss.com/novacord).

1
cord/__main__.py Normal file
View file

@ -0,0 +1 @@
import bot

View file

@ -1,8 +1,13 @@
# This example requires the 'members' and 'message_content' privileged intents to function.
import os
import openai
import asyncio
import nextcord
import requests
import keys
import chatbot
import embedder
from dotenv import load_dotenv
from nextcord.ext import commands
@ -28,45 +33,68 @@ async def on_message(message):
async def chat(interaction: nextcord.Interaction,
prompt: str = SlashOption(description='AI Prompt', required=True)
):
partial_message = await interaction.send('') # empty message
message = await partial_message.fetch()
await chatbot.respond(interaction, prompt)
openai.api_base = os.getenv('OPENAI_BASE', 'https://api.openai.com/v1')
openai.api_key = os.getenv('OPENAI_KEY')
@bot.slash_command(description='Sets your DMs up, so you can write the bot.')
async def dm(interaction: nextcord.Interaction):
try:
await interaction.user.create_dm()
await embedder.info(interaction.user.dm_channel, 'Hello!')
except nextcord.Forbidden:
await embedder.error(interaction, text='Please open this server\'s options, go to `Privacy Settings` and enable `Direct Messages` and `Message Requests`.')
else:
await embedder.ok(interaction, 'Great, DMs are set up successfully!')
model = os.getenv('OPENAI_MODEL')
@bot.slash_command(description='Get your secret NovaAI API key.')
async def key(interaction: nextcord.Interaction):
try:
resp = requests.post(
url='https://nova-oss.com/api/tos-verification',
timeout=5,
headers={'Content-Type': 'application/json', 'Authorization': os.getenv('TOS_VERIFICATION_KEY')}
).json()
except Exception as exc:
await embedder.error(interaction, """Sorry, the API server for the verification system is not functioning,
which means you can\'t create a new key right now. Please report this issue to the staff!""")
raise exc
async with interaction.channel.typing(): # show the "Typing..."
completion = openai.ChatCompletion.create(
model=model,
messages=[
{'role': 'system', 'content': f"""You are a helpful Discord AI bot based on OpenAI\'s {model} model called "Nova".
You were developed by NovaAI (website: nova-oss.com) in July of 2023, but your knowledge is limited to mid-2021.
Respond using Markdown. Keep things simple and short and directly do what the user says without any fluff.
For programming code, always make use formatted code blocks like this:
```py
print("Hello")
```
"""},
{'role': 'user', 'content': prompt}
],
temperature=0.6,
stream=True
tos_code = resp['code']
tos_emoji = resp['emoji']
tos_message = await embedder.warn(interaction, f"""You have to read the privacy policy and terms of service first.
In the latter, there is a hidden emoji which you'll have to send (NOT react!) in here.
https://nova-oss.com/legal/privacy
https://nova-oss.com/legal/terms?verify={tos_code}
I know it's annoying, but it really helps combat spam bots and abuse.
This message will be deleted and your code will run out **after about 10 minutes**
if you don't pass the verification, but **feel free to run this command again** at any time.
""", ephemeral=True)
def check(message): return interaction.user.id == message.author.id and message.content == tos_emoji
try:
answer = await bot.wait_for('message', timeout=666, check=check)
except asyncio.TimeoutError:
await tos_message.delete()
requests.delete(
url=f'https://nova-oss.com/api/tos-verification/{tos_code}',
timeout=5,
headers={'Content-Type': 'application/json', 'Authorization': os.getenv('TOS_VERIFICATION_KEY')}
)
text = ''
else:
await answer.delete()
api_key = await keys.create(interaction.user)
await embedder.ok(interaction, f"""This is your **secret** API key. Don't paste it on untrusted websites, apps or programs.
Store it securely using a `.env` file in the environment variables or use a secure password manager like *KeePass*, *ProtonPass* or *Bitwarden*.
We reserve the right to __disable your API key at any time__ if you violate our terms of service.
If you accept the terms of service and privacy policy, feel free to use the following API key:
for event in completion: # loop through word generation in real time
try:
new_text = event['choices'][0]['delta']['content'] # newly generated word
except KeyError: # end
break
## ||`{api_key}`||
text += new_text
if text:
await message.edit(content=text)
await message.add_reaction('')
""", ephemeral=True)
bot.run(os.getenv('DISCORD_TOKEN'))

54
cord/chatbot.py Normal file
View file

@ -0,0 +1,54 @@
import os
import openai
import embedder
from dotenv import load_dotenv
load_dotenv()
async def respond(interaction, prompt):
partial_message = await interaction.send('') # empty message
message = await partial_message.fetch()
openai.api_base = os.getenv('OPENAI_BASE', 'https://api.openai.com/v1')
openai.api_key = os.getenv('OPENAI_KEY')
model = os.getenv('OPENAI_MODEL')
async with interaction.channel.typing(): # show the "Typing..."
try:
completion = openai.ChatCompletion.create(
model=model,
messages=[
{'role': 'system', 'content': f"""You are a helpful Discord AI bot based on OpenAI\'s {model} model called "Nova".
You were developed by NovaAI (website: nova-oss.com) in July of 2023, but your knowledge is limited to mid-2021.
Respond using Markdown. Keep things simple and short and directly do what the user says without any fluff.
For programming code, always make use formatted code blocks like this:
```py
print("Hello")
```
"""},
{'role': 'user', 'content': prompt}
],
temperature=0.6,
stream=True
)
except Exception as exc:
await embedder.error(interaction, 'Could not generate an AI response.', ephemeral=True)
raise exc
text = ''
for event in completion: # loop through word generation in real time
try:
new_text = event['choices'][0]['delta']['content'] # newly generated word
except KeyError: # end
break
text += new_text
if text:
await message.edit(content=text)
await message.add_reaction('')

40
cord/embedder.py Normal file
View file

@ -0,0 +1,40 @@
import nextcord
import traceback
from typing import Union
async def send(
ctx,
title: str,
text: str,
ephemeral: bool = False,
color: nextcord.Color = nextcord.Color.blue()
):
embed = nextcord.Embed(
title=title,
description=text,
color=color
)
embed.set_footer(text='Powered by NovaAI', icon_url='https://i.ibb.co/LDyFcSh/fav-blurple.png')
embed.set_author(name='NovaCord', url='https://nova-oss.com/novacord')
if isinstance(ctx, nextcord.Message):
response = await ctx.reply(embed=embed)
elif isinstance(ctx, Union[nextcord.Interaction, nextcord.InteractionResponse]):
response = await ctx.send(embed=embed, ephemeral=ephemeral)
else:
response = await ctx.send(embed=embed)
return response
async def ok(ctx, text: str, title: str='Success', *args, **kwargs):
return await send(ctx, title, text, color=nextcord.Color.green(), *args, **kwargs)
async def info(ctx, text: str, title: str='Information', *args, **kwargs):
return await send(ctx, title, text, color=nextcord.Color.blue(), *args, **kwargs)
async def warn(ctx, text: str, title: str='Warning', *args, **kwargs):
return await send(ctx, title, text, color=nextcord.Color.orange(), *args, **kwargs)
async def error(ctx, text: str, title: str='Error - Command Failed', *args, **kwargs):
return await send(ctx, title, text, color=nextcord.Color.red(), *args, **kwargs)

4
cord/keys.py Normal file
View file

@ -0,0 +1,4 @@
import secrets
async def create(user):
return 'nv-' + secrets.token_urlsafe(32)

View file

1
emojis.txt Normal file
View file

@ -0,0 +1 @@
🥇🥈🥉📒📕📗📘📙💩👻💀👽👾🤖🎃💛💚💙💜💔🐶🐱🐭🐹🐰🦊🐻🐼🐨🐯🦁🐮🐷🐽🐸🐵🙊🙉🙊🐒🐔🐧🐦🐤🐣🐥🦆🦅🦉🦇🐺🐗🐴🦄🐝🐛🦋🐌🐚🐞🐢🐍🦎🦂🦀🦑🐙🦐🐠🐟🐡🐬🦈🐳🐋🐊🐆🐅🐂🐄🦌🐪🐫🐘🦏🦍🐎🐖🐐🐏🐑🐕🐩🐈🐓🦃🐇🐁🐀🐾🐉🐲🌵🎄🌲🌳🌴🌱🌿🍀🎍🎋🍃🍂🍁🍄🌾💐🌷🌹🥀🌻🌼🌸🌺🌎🌍🌏🌕🌖🌗🌘🌑🌒🌓🌔🌚🌝🌞🌛🌜🌙💫⭐️🌟✨🔥💥⛅️🌈⛄️💨🌪🌊💧💦🍏🍎🍐🍊🍋🍌🍉🍇🍓🍈🍒🍑🍍🥝🥑🍅🍆🥒🥕🌽🥔🍠🌰🥜🍯🥐🍞🥖🧀🥚🍳🥓🥞🍤🍗🍖🍕🌭🍔🍟🥙🌮🌯🥗🥘🍝🍜🍲🍥🍣🍱🍛🍚🍙🍘🍢🍡🍧🍨🍦🍰🎂🍮🍭🍬🍫🍿🍩🍪🥛🍼🍵🍶🍺🍻🥂🍷🥃🍸🍹🍾🥄🍴⚽️🏀🏈⚾️🎾🏐🏉🎱🏓🏸🥅🏒🏑🏏⛳️🏹🎣🥊🥋

1
screen.sh Executable file
View file

@ -0,0 +1 @@
screen -S nova-cord python cord