Compare commits
2 commits
50dce8da96
...
a8c0969a71
Author | SHA1 | Date | |
---|---|---|---|
a8c0969a71 | |||
6b6c047b68 |
11 changed files with 181 additions and 14 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -2512,6 +2512,7 @@ dependencies = [
|
||||||
"rocket_codegen 0.5.0-rc.3",
|
"rocket_codegen 0.5.0-rc.3",
|
||||||
"rocket_http 0.5.0-rc.3",
|
"rocket_http 0.5.0-rc.3",
|
||||||
"serde",
|
"serde",
|
||||||
|
"serde_json",
|
||||||
"state 0.5.3",
|
"state 0.5.3",
|
||||||
"tempfile",
|
"tempfile",
|
||||||
"time 0.3.21",
|
"time 0.3.21",
|
||||||
|
|
|
@ -14,7 +14,7 @@ dotenv = "0.15.0"
|
||||||
gettext = "0.4.0"
|
gettext = "0.4.0"
|
||||||
poise = "0.5.5"
|
poise = "0.5.5"
|
||||||
reqwest = "0.11.18"
|
reqwest = "0.11.18"
|
||||||
rocket = { version = "=0.5.0-rc.3", features = ["secrets"] }
|
rocket = { version = "=0.5.0-rc.3", features = ["secrets", "json"] }
|
||||||
rocket_contrib = { version = "0.4.11", features = ["templates"] }
|
rocket_contrib = { version = "0.4.11", features = ["templates"] }
|
||||||
rocket_dyn_templates = { version = "0.1.0-rc.3", features = ["tera"] }
|
rocket_dyn_templates = { version = "0.1.0-rc.3", features = ["tera"] }
|
||||||
sass-rocket-fairing = "0.2.0"
|
sass-rocket-fairing = "0.2.0"
|
||||||
|
|
|
@ -9,3 +9,4 @@ pub mod user {
|
||||||
pub const USER_AVATAR_COOKIE: &str = "user_avatar";
|
pub const USER_AVATAR_COOKIE: &str = "user_avatar";
|
||||||
}
|
}
|
||||||
pub const LANG_COOKIE: &str = "lang";
|
pub const LANG_COOKIE: &str = "lang";
|
||||||
|
pub const WELCOMED_COOKIE: &str = "welcomed";
|
|
@ -42,7 +42,7 @@ async fn rocket() -> _ {
|
||||||
.manage(http)
|
.manage(http)
|
||||||
.mount(
|
.mount(
|
||||||
"/",
|
"/",
|
||||||
routes![get_challenge, login, post_login, success, logout, testing],
|
routes![get_challenge, get_guilds, login, post_login, success, logout, testing],
|
||||||
)
|
)
|
||||||
.mount("/css", FileServer::from(relative!("styles/css")))
|
.mount("/css", FileServer::from(relative!("styles/css")))
|
||||||
.mount("/", FileServer::from(relative!("static")).rank(1))
|
.mount("/", FileServer::from(relative!("static")).rank(1))
|
||||||
|
|
34
src/routes/get_guilds.rs
Normal file
34
src/routes/get_guilds.rs
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
use poise::serenity_prelude::Http;
|
||||||
|
use rocket::{http::CookieJar, State};
|
||||||
|
use rocket::response::stream::{Event, EventStream};
|
||||||
|
|
||||||
|
use crate::{cookies::user::USER_ID_COOKIE, models::Settings};
|
||||||
|
|
||||||
|
// TODO: Incrementally send guilds
|
||||||
|
#[get("/get_guilds")]
|
||||||
|
pub async fn get_guilds<'a>(
|
||||||
|
cookies: &'a CookieJar<'_>,
|
||||||
|
settings: &'a State<Settings>,
|
||||||
|
http: &'a State<Http>,
|
||||||
|
) -> EventStream![Event + 'a] {
|
||||||
|
// EventStream![] is shorthand for EventStream[Event],
|
||||||
|
// but we need to pass in a lifetime parameter.
|
||||||
|
EventStream! {
|
||||||
|
// TODO: Proper error handling
|
||||||
|
let user_id: u64 = match cookies.get_private(USER_ID_COOKIE) {
|
||||||
|
Some(id) => match id.value().parse() {
|
||||||
|
Ok(id) => id,
|
||||||
|
Err(_) => panic!("BadRequest"),
|
||||||
|
// Err(_) => return Err(Status::BadRequest),
|
||||||
|
},
|
||||||
|
None => panic!("Unauthorized"),
|
||||||
|
// None => return Err(Status::Unauthorized),
|
||||||
|
};
|
||||||
|
for guild in &settings.guilds {
|
||||||
|
if guild.hidden {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
yield Event::data(format!("{},{}", guild.id, http.get_member(guild.id, user_id).await.is_ok()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,7 +6,7 @@ use rocket::response::Redirect;
|
||||||
pub fn login() -> Redirect {
|
pub fn login() -> Redirect {
|
||||||
Redirect::to(format!(
|
Redirect::to(format!(
|
||||||
// Switch from response_type=code to response_type=token from URL generator
|
// Switch from response_type=code to response_type=token from URL generator
|
||||||
"https://discord.com/api/oauth2/authorize?client_id={client_id}&redirect_uri={redirect_uri}&response_type=token&scope=identify%20guilds.join%20guilds",
|
"https://discord.com/api/oauth2/authorize?client_id={client_id}&redirect_uri={redirect_uri}&response_type=token&scope=identify",
|
||||||
client_id = env::var("CLIENT_ID").unwrap(),
|
client_id = env::var("CLIENT_ID").unwrap(),
|
||||||
redirect_uri = format!("{}success", env::var("DOMAIN").unwrap()),
|
redirect_uri = format!("{}success", env::var("DOMAIN").unwrap()),
|
||||||
))
|
))
|
||||||
|
|
|
@ -2,6 +2,10 @@
|
||||||
mod _get_challenge;
|
mod _get_challenge;
|
||||||
pub use _get_challenge::get_challenge;
|
pub use _get_challenge::get_challenge;
|
||||||
|
|
||||||
|
#[path = "get_guilds.rs"]
|
||||||
|
mod _get_guilds;
|
||||||
|
pub use _get_guilds::get_guilds;
|
||||||
|
|
||||||
#[path = "login.rs"]
|
#[path = "login.rs"]
|
||||||
mod _login;
|
mod _login;
|
||||||
pub use _login::login;
|
pub use _login::login;
|
||||||
|
|
|
@ -5,7 +5,7 @@ use rocket::{
|
||||||
response::Redirect,
|
response::Redirect,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{cookies::token::TOKEN_EXPIRE_COOKIE, models::User};
|
use crate::{cookies::{token::TOKEN_EXPIRE_COOKIE, WELCOMED_COOKIE}, models::User};
|
||||||
|
|
||||||
#[derive(FromForm)]
|
#[derive(FromForm)]
|
||||||
pub struct Login<'r> {
|
pub struct Login<'r> {
|
||||||
|
@ -17,7 +17,7 @@ pub struct Login<'r> {
|
||||||
|
|
||||||
#[post("/login", data = "<login>")]
|
#[post("/login", data = "<login>")]
|
||||||
pub async fn post_login(login: Form<Login<'_>>, cookies: &CookieJar<'_>) -> Redirect {
|
pub async fn post_login(login: Form<Login<'_>>, cookies: &CookieJar<'_>) -> Redirect {
|
||||||
if (login.token_type != "Bearer" || login.scope != "guilds.join+identify+guilds")
|
if (login.token_type != "Bearer" || login.scope.split("+").any(|scope| scope == "identify"))
|
||||||
&& User::init(login.access_token, cookies).await.is_ok()
|
&& User::init(login.access_token, cookies).await.is_ok()
|
||||||
{
|
{
|
||||||
cookies.add(Cookie::new(
|
cookies.add(Cookie::new(
|
||||||
|
@ -26,6 +26,7 @@ pub async fn post_login(login: Form<Login<'_>>, cookies: &CookieJar<'_>) -> Redi
|
||||||
.timestamp()
|
.timestamp()
|
||||||
.to_string(),
|
.to_string(),
|
||||||
));
|
));
|
||||||
|
cookies.remove(Cookie::named(WELCOMED_COOKIE));
|
||||||
}
|
}
|
||||||
Redirect::to("/")
|
Redirect::to("/")
|
||||||
}
|
}
|
||||||
|
|
|
@ -164,14 +164,20 @@ a {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
a {
|
.joinButton {
|
||||||
|
display: inline;
|
||||||
background: seagreen;
|
background: seagreen;
|
||||||
color: white;
|
color: white;
|
||||||
|
font: inherit;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
border: none;
|
border: none;
|
||||||
outline: none;
|
outline: none;
|
||||||
padding: 0.25em 0.5em 0.25em 0.5em;
|
padding: 0.25em 0.5em 0.25em 0.5em;
|
||||||
border-radius: 2px;
|
border-radius: 2px;
|
||||||
|
|
||||||
|
&:disabled {
|
||||||
|
background: gray;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.recommended {
|
.recommended {
|
||||||
|
@ -194,4 +200,9 @@ a {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon {
|
||||||
|
height: 1.5em;
|
||||||
|
vertical-align: middle;
|
||||||
}
|
}
|
|
@ -9,7 +9,9 @@
|
||||||
<link href="/css/style.css" rel="stylesheet">
|
<link href="/css/style.css" rel="stylesheet">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
|
{% if user %}
|
||||||
{% include "modal" %}
|
{% include "modal" %}
|
||||||
|
{% endif %}
|
||||||
<div id="content">
|
<div id="content">
|
||||||
<nav>
|
<nav>
|
||||||
<a class="link" href="/{{ challenge - 1 }}"><svg class="svg-inline" aria-hidden="true" focusable="false" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 384 512"><path fill="currentColor" d="M41.4 233.4c-12.5 12.5-12.5 32.8 0 45.3l192 192c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3L109.3 256 278.6 86.6c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0l-192 192z"></path></svg></a>
|
<a class="link" href="/{{ challenge - 1 }}"><svg class="svg-inline" aria-hidden="true" focusable="false" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 384 512"><path fill="currentColor" d="M41.4 233.4c-12.5 12.5-12.5 32.8 0 45.3l192 192c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3L109.3 256 278.6 86.6c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0l-192 192z"></path></svg></a>
|
||||||
|
@ -26,8 +28,21 @@
|
||||||
<span>{{ user.username }}</span> <img src="https://cdn.discordapp.com/avatars/{{ user.id }}/{{ user.avatar }}.webp?size=1024">
|
<span>{{ user.username }}</span> <img src="https://cdn.discordapp.com/avatars/{{ user.id }}/{{ user.avatar }}.webp?size=1024">
|
||||||
</a>
|
</a>
|
||||||
<nav class="dropdown-content">
|
<nav class="dropdown-content">
|
||||||
|
<span class="link" onclick="showServers()">
|
||||||
|
<span>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="icon">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" d="M18 18.72a9.094 9.094 0 003.741-.479 3 3 0 00-4.682-2.72m.94 3.198l.001.031c0 .225-.012.447-.037.666A11.944 11.944 0 0112 21c-2.17 0-4.207-.576-5.963-1.584A6.062 6.062 0 016 18.719m12 0a5.971 5.971 0 00-.941-3.197m0 0A5.995 5.995 0 0012 12.75a5.995 5.995 0 00-5.058 2.772m0 0a3 3 0 00-4.681 2.72 8.986 8.986 0 003.74.477m.94-3.197a5.971 5.971 0 00-.94 3.197M15 6.75a3 3 0 11-6 0 3 3 0 016 0zm6 3a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0zm-13.5 0a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0z" />
|
||||||
|
</svg>
|
||||||
|
Join servers
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
<a href="/logout" class="link">
|
<a href="/logout" class="link">
|
||||||
<span>Log out</span>
|
<span>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="icon">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" d="M15.75 9V5.25A2.25 2.25 0 0013.5 3h-6a2.25 2.25 0 00-2.25 2.25v13.5A2.25 2.25 0 007.5 21h6a2.25 2.25 0 002.25-2.25V15M12 9l-3 3m0 0l3 3m-3-3h12.75" />
|
||||||
|
</svg>
|
||||||
|
Log out
|
||||||
|
</span>
|
||||||
</a>
|
</a>
|
||||||
</nav>
|
</nav>
|
||||||
</div>
|
</div>
|
||||||
|
@ -36,7 +51,7 @@
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div class="dropdown">
|
<div class="dropdown">
|
||||||
<span class="link">
|
<span class="link">
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-6 h-6" style="height: 1.5em">
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="icon">
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" d="M10.5 21l5.25-11.25L21 21m-9-3h7.5M3 5.621a48.474 48.474 0 016-.371m0 0c1.12 0 2.233.038 3.334.114M9 5.25V3m3.334 2.364C11.176 10.658 7.69 15.08 3 17.502m9.334-12.138c.896.061 1.785.147 2.666.257m-4.589 8.495a18.023 18.023 0 01-3.827-5.802" />
|
<path stroke-linecap="round" stroke-linejoin="round" d="M10.5 21l5.25-11.25L21 21m-9-3h7.5M3 5.621a48.474 48.474 0 016-.371m0 0c1.12 0 2.233.038 3.334.114M9 5.25V3m3.334 2.364C11.176 10.658 7.69 15.08 3 17.502m9.334-12.138c.896.061 1.785.147 2.666.257m-4.589 8.495a18.023 18.023 0 01-3.827-5.802" />
|
||||||
</svg>
|
</svg>
|
||||||
</span>
|
</span>
|
||||||
|
|
|
@ -1,14 +1,16 @@
|
||||||
<dialog>
|
<dialog>
|
||||||
<h1>Welcome to {{ "title" | i18n(lang=lang) }}!</h1>
|
<div class="welcome">
|
||||||
<p>In order to participate in challenges, you must be a member of a participating Discord server.</p>
|
<h1>Welcome to {{ "title" | i18n(lang=lang) }}!</h1>
|
||||||
<h2>Join a participating server</h2>
|
<p>In order to participate in challenges, you must be a member of a participating Discord server.</p>
|
||||||
|
<h2>Join a participating server</h2>
|
||||||
|
</div>
|
||||||
<div class="servers">
|
<div class="servers">
|
||||||
{% for guild in settings.guilds %}
|
{% for guild in settings.guilds %}
|
||||||
{% if guild.hidden or not guild.invite %}{% continue %}{% endif %}
|
{% if guild.hidden or not guild.invite %}{% continue %}{% endif %}
|
||||||
<div{% if guild.recommended %} class="recommended"{% endif %}>
|
<div id="{{ guild.id }}" {% if guild.recommended %} class="recommended"{% endif %}>
|
||||||
<img src="https://cdn.discordapp.com/icons/{{ guild.id }}/{{ guild.icon }}.webp?size=96" alt="Server icon">
|
<img src="https://cdn.discordapp.com/icons/{{ guild.id }}/{{ guild.icon }}.webp?size=96" alt="Server icon">
|
||||||
<div class="name">{{ guild.name }}</div>
|
<div class="name">{{ guild.name }}</div>
|
||||||
<a href="https://discord.gg/{{ guild.invite }}">Join</a>
|
<a href="https://discord.gg/{{ guild.invite }}" class="joinButton">Join</a>
|
||||||
{% if guild.recommended %}
|
{% if guild.recommended %}
|
||||||
<div class="label-wrapper">
|
<div class="label-wrapper">
|
||||||
<div class="label">Recommended</div>
|
<div class="label">Recommended</div>
|
||||||
|
@ -19,5 +21,103 @@
|
||||||
</div>
|
</div>
|
||||||
</dialog>
|
</dialog>
|
||||||
<script>
|
<script>
|
||||||
document.querySelector("dialog").showModal();
|
const dialog = document.querySelector("dialog");
|
||||||
|
// Fetch guilds
|
||||||
|
let guilds = null;
|
||||||
|
let is_in_participating_server = null;
|
||||||
|
const getGuilds = async () => {
|
||||||
|
if (guilds === null) {
|
||||||
|
// Create the guilds dictionary object
|
||||||
|
guilds = {};
|
||||||
|
|
||||||
|
// Create an EventSource object to listen to SSE events
|
||||||
|
const eventSource = new EventSource('/get_guilds');
|
||||||
|
|
||||||
|
// Event listener for 'message' events
|
||||||
|
eventSource.addEventListener('message', event => {
|
||||||
|
// Parse the data received from the event
|
||||||
|
const [id, is_member] = event.data.split(',');
|
||||||
|
|
||||||
|
// Add the key-value pair to the guilds dictionary
|
||||||
|
guilds[id] = is_member === 'true';
|
||||||
|
|
||||||
|
if (guilds[id]) {
|
||||||
|
setJoined(id);
|
||||||
|
is_in_participating_server = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Function to block execution until SSE events are done
|
||||||
|
function waitForSSE() {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
// Event listener for 'error' event
|
||||||
|
// For some reason errors when completing
|
||||||
|
eventSource.addEventListener('error', () => {
|
||||||
|
eventSource.close();
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call the function to block execution until SSE events are done
|
||||||
|
await waitForSSE()
|
||||||
|
}
|
||||||
|
return guilds;
|
||||||
|
}
|
||||||
|
// Show server modal, both for welcome and server list
|
||||||
|
const showServers = async () => {
|
||||||
|
dialog.showModal();
|
||||||
|
// If the user is logged in, load the guild list
|
||||||
|
// and check if they have joined the available servers
|
||||||
|
const guilds = await getGuilds();
|
||||||
|
Object.keys(guilds).forEach(id => {
|
||||||
|
if (guilds[id]) {
|
||||||
|
setJoined(id);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
};
|
||||||
|
// Switch Join button to Joined for server
|
||||||
|
const setJoined = id => {
|
||||||
|
element = document.getElementById(id);
|
||||||
|
if (element === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Check if already set as joined
|
||||||
|
a = element.querySelector("a");
|
||||||
|
if (a === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const button = document.createElement("button");
|
||||||
|
button.tagName = "button";
|
||||||
|
button.className = "joinButton";
|
||||||
|
button.disabled = true;
|
||||||
|
button.innerHTML = "Joined"
|
||||||
|
element.querySelector("a").replaceWith(button);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check for welcomed cookie
|
||||||
|
const welcomed = () => document.cookie.split("; ").includes("welcomed=true");
|
||||||
|
// Load welcome modal, check if it is necessary
|
||||||
|
const loadServers = async () => {
|
||||||
|
if (welcomed()) {
|
||||||
|
// If already welcomed, remove welcome text
|
||||||
|
// in case user pressed Join servers button
|
||||||
|
document.querySelector("dialog .welcome").style.display = "none";
|
||||||
|
} else {
|
||||||
|
const guilds = await getGuilds();
|
||||||
|
if (!is_in_participating_server) {
|
||||||
|
showServers();
|
||||||
|
}
|
||||||
|
// Even if already in participating server,
|
||||||
|
// set welcomed cookie so we don't have to run this request again
|
||||||
|
document.cookie = 'welcomed=true; expires=Tue, 19 Jan 2038 03:14:07 UTC; SameSite=Strict; path=/';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
if (welcomed()) {
|
||||||
|
// TODO: Remove welcome text via cookie check in template?
|
||||||
|
document.querySelector("dialog .welcome").style.display = "none";
|
||||||
|
} else {
|
||||||
|
// If not welcomed, see if welcome is necessary
|
||||||
|
loadServers();
|
||||||
|
}
|
||||||
</script>
|
</script>
|
Loading…
Add table
Reference in a new issue