-
-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(github): add github integration
- Loading branch information
1 parent
362aef2
commit d841a49
Showing
11 changed files
with
354 additions
and
56 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -7,3 +7,4 @@ praw==7.8.1 | |
py-cord==2.6.1 | ||
python-dotenv==1.0.1 | ||
requests==2.32.3 | ||
requests-oauthlib==2.0.0 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,69 @@ | ||
# standard imports | ||
import os | ||
|
||
# lib imports | ||
from cryptography import x509 | ||
from cryptography.hazmat.backends import default_backend | ||
from cryptography.hazmat.primitives import hashes | ||
from cryptography.hazmat.primitives.asymmetric import rsa | ||
from cryptography.hazmat.primitives.serialization import Encoding, PrivateFormat, NoEncryption | ||
from datetime import datetime, timedelta, UTC | ||
|
||
# local imports | ||
from src import common | ||
|
||
CERT_FILE = os.path.join(common.data_dir, "cert.pem") | ||
KEY_FILE = os.path.join(common.data_dir, "key.pem") | ||
|
||
|
||
def check_expiration(cert_path: str) -> int: | ||
with open(cert_path, "rb") as cert_file: | ||
cert_data = cert_file.read() | ||
cert = x509.load_pem_x509_certificate(cert_data, default_backend()) | ||
expiry_date = cert.not_valid_after_utc | ||
return (expiry_date - datetime.now(UTC)).days | ||
|
||
|
||
def generate_certificate(): | ||
private_key = rsa.generate_private_key( | ||
public_exponent=65537, | ||
key_size=4096, | ||
) | ||
subject = issuer = x509.Name([ | ||
x509.NameAttribute(x509.NameOID.COMMON_NAME, u"localhost"), | ||
]) | ||
cert = x509.CertificateBuilder().subject_name( | ||
subject | ||
).issuer_name( | ||
issuer | ||
).public_key( | ||
private_key.public_key() | ||
).serial_number( | ||
x509.random_serial_number() | ||
).not_valid_before( | ||
datetime.now(UTC) | ||
).not_valid_after( | ||
datetime.now(UTC) + timedelta(days=365) | ||
).sign(private_key, hashes.SHA256()) | ||
|
||
with open(KEY_FILE, "wb") as f: | ||
f.write(private_key.private_bytes( | ||
encoding=Encoding.PEM, | ||
format=PrivateFormat.TraditionalOpenSSL, | ||
encryption_algorithm=NoEncryption(), | ||
)) | ||
|
||
with open(CERT_FILE, "wb") as f: | ||
f.write(cert.public_bytes(Encoding.PEM)) | ||
|
||
|
||
def initialize_certificate() -> tuple[str, str]: | ||
print("Initializing SSL certificate") | ||
if os.path.exists(CERT_FILE) and os.path.exists(KEY_FILE): | ||
cert_expires_in = check_expiration(CERT_FILE) | ||
print(f"Certificate expires in {cert_expires_in} days.") | ||
if cert_expires_in >= 90: | ||
return CERT_FILE, KEY_FILE | ||
print("Generating new certificate") | ||
generate_certificate() | ||
return CERT_FILE, KEY_FILE | ||
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
# standard imports | ||
import shelve | ||
import threading | ||
|
||
|
||
class Database: | ||
def __init__(self, db_path): | ||
self.db_path = db_path | ||
self.lock = threading.Lock() | ||
|
||
def __enter__(self): | ||
self.lock.acquire() | ||
self.db = shelve.open(self.db_path, writeback=True) | ||
return self.db | ||
|
||
def __exit__(self, exc_type, exc_val, exc_tb): | ||
self.sync() | ||
self.db.close() | ||
self.lock.release() | ||
|
||
def sync(self): | ||
self.db.sync() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,123 @@ | ||
# standard imports | ||
import os | ||
|
||
# lib imports | ||
import discord | ||
import requests | ||
from requests_oauthlib import OAuth2Session | ||
|
||
|
||
class GitHubCommandsCog(discord.Cog): | ||
def __init__(self, bot): | ||
self.bot = bot | ||
self.token = os.getenv("GITHUB_TOKEN") | ||
self.org_name = os.getenv("GITHUB_ORG_NAME", "LizardByte") | ||
self.graphql_url = "https://api.github.com/graphql" | ||
self.headers = { | ||
"Authorization": f"Bearer {self.token}", | ||
"Content-Type": "application/json" | ||
} | ||
|
||
@discord.slash_command( | ||
name="get_sponsors", | ||
description="Get list of GitHub sponsors", | ||
default_member_permissions=discord.Permissions(manage_guild=True), | ||
) | ||
async def get_sponsors( | ||
self, | ||
ctx: discord.ApplicationContext, | ||
): | ||
""" | ||
Get list of GitHub sponsors. | ||
Parameters | ||
---------- | ||
ctx : discord.ApplicationContext | ||
Request message context. | ||
""" | ||
query = """ | ||
query { | ||
organization(login: "%s") { | ||
sponsorshipsAsMaintainer(first: 100) { | ||
edges { | ||
node { | ||
sponsorEntity { | ||
... on User { | ||
login | ||
name | ||
avatarUrl | ||
url | ||
} | ||
... on Organization { | ||
login | ||
name | ||
avatarUrl | ||
url | ||
} | ||
} | ||
tier { | ||
name | ||
monthlyPriceInDollars | ||
} | ||
} | ||
} | ||
} | ||
} | ||
} | ||
""" % self.org_name | ||
|
||
response = requests.post(self.graphql_url, json={'query': query}, headers=self.headers) | ||
data = response.json() | ||
|
||
if 'errors' in data: | ||
print(data['errors']) | ||
await ctx.respond("An error occurred while fetching sponsors.", ephemeral=True) | ||
return | ||
|
||
message = "List of GitHub sponsors" | ||
for edge in data['data']['organization']['sponsorshipsAsMaintainer']['edges']: | ||
sponsor = edge['node']['sponsorEntity'] | ||
tier = edge['node'].get('tier', {}) | ||
tier_info = f" - Tier: {tier.get('name', 'N/A')} (${tier.get('monthlyPriceInDollars', 'N/A')}/month)" | ||
message += f"\n* [{sponsor['login']}]({sponsor['url']}){tier_info}" | ||
|
||
embed = discord.Embed(title="GitHub Sponsors", color=0x00ff00, description=message) | ||
|
||
await ctx.respond(embed=embed, ephemeral=True) | ||
|
||
@discord.slash_command( | ||
name="link_github", | ||
description="Validate GitHub sponsor status" | ||
) | ||
async def link_github(self, ctx: discord.ApplicationContext): | ||
""" | ||
Link Discord account with GitHub account, by validating Discord user's "GitHub" connected account status. | ||
User to login to Discord via OAuth2, and check if their connected GitHub account is a sponsor of the project. | ||
Parameters | ||
---------- | ||
ctx : discord.ApplicationContext | ||
Request message context. | ||
""" | ||
discord_oauth = OAuth2Session( | ||
os.environ['DISCORD_CLIENT_ID'], | ||
redirect_uri=os.environ['DISCORD_REDIRECT_URI'], | ||
scope=[ | ||
"identify", | ||
"connections", | ||
], | ||
) | ||
authorization_url, state = discord_oauth.authorization_url("https://discord.com/oauth2/authorize") | ||
|
||
with self.bot.db as db: | ||
db['oauth_states'] = db.get('oauth_states', {}) | ||
db['oauth_states'][str(ctx.author.id)] = state | ||
db.sync() | ||
|
||
# Store the state in the user's session or database | ||
await ctx.respond(f"Please authorize the application by clicking [here]({authorization_url}).", ephemeral=True) | ||
|
||
|
||
def setup(bot: discord.Bot): | ||
bot.add_cog(GitHubCommandsCog(bot=bot)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,2 @@ | ||
DISCORD_BOT = None | ||
REDDIT_BOT = None | ||
This file was deleted.
Oops, something went wrong.
Oops, something went wrong.