Polish routes more
This commit is contained in:
parent
95dc6ebd61
commit
5f2905faab
@ -52,4 +52,4 @@ $ sqlx migrate run
|
|||||||
- Query (Can have multiple)
|
- Query (Can have multiple)
|
||||||
|
|
||||||
- `dump`: Dumps a table in the database (Mostly for debug)
|
- `dump`: Dumps a table in the database (Mostly for debug)
|
||||||
- Table name (`accounts`, `profiles`, `sessions`, `tokens`)
|
- Table name (`accounts`, `profiles`, `sessions`, `tokens`)
|
||||||
|
@ -14,4 +14,3 @@
|
|||||||
pub use util::*;
|
pub use util::*;
|
||||||
|
|
||||||
mod util;
|
mod util;
|
||||||
|
|
||||||
|
@ -14,14 +14,12 @@ pub use tide::{Middleware, prelude::*, Request, Response, Result, utils::After};
|
|||||||
|
|
||||||
pub use yggdrasil::*;
|
pub use yggdrasil::*;
|
||||||
|
|
||||||
mod capes;
|
|
||||||
|
|
||||||
pub fn nest(db: Database) -> tide::Server<Database> {
|
pub fn nest(db: Database) -> tide::Server<Database> {
|
||||||
info!("Loading nest");
|
info!("Loading nest");
|
||||||
|
|
||||||
let mut nest = tide::with_state(db);
|
let mut nest = tide::with_state(db.to_owned());
|
||||||
nest.at("/profiles/capes/active").put(capes::upload_cape);
|
|
||||||
nest.at("/profiles/capes/active").delete(capes::delete_cape);
|
nest.at("/user/profile/").nest(super::profile::nest(db.to_owned()));
|
||||||
|
|
||||||
nest
|
nest
|
||||||
}
|
}
|
@ -9,13 +9,17 @@
|
|||||||
* You should have received a copy of the GNU General Public License along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* You should have received a copy of the GNU General Public License along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
use anyhow::anyhow;
|
pub use log::{info, log, warn};
|
||||||
use tide::{Request, Result};
|
pub use tide::{Middleware, prelude::*, Request, Response, Result, utils::After};
|
||||||
|
|
||||||
use yggdrasil::Database;
|
pub use yggdrasil::*;
|
||||||
|
|
||||||
pub async fn profiles(req: Request<Database>) -> Result {
|
pub fn nest(db: Database) -> tide::Server<Database> {
|
||||||
|
info!("Loading nest");
|
||||||
|
|
||||||
|
let mut nest = tide::with_state(db.to_owned());
|
||||||
|
|
||||||
|
|
||||||
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
|
||||||
|
nest
|
||||||
}
|
}
|
@ -88,7 +88,6 @@ pub async fn authenticate(mut req: Request<Database>) -> Result {
|
|||||||
None => "".to_string()
|
None => "".to_string()
|
||||||
},
|
},
|
||||||
"active_cape": profile.active_cape,
|
"active_cape": profile.active_cape,
|
||||||
"attributes": profile.attributes.to_json()
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -21,7 +21,9 @@ pub fn nest(db: Database) -> tide::Server<Database> {
|
|||||||
|
|
||||||
nest.at("/").get(authlib_meta);
|
nest.at("/").get(authlib_meta);
|
||||||
nest.at("/authserver").nest(super::auth::nest(db.to_owned()));
|
nest.at("/authserver").nest(super::auth::nest(db.to_owned()));
|
||||||
nest.at("/sessionserver").nest(super::session::nest(db.to_owned()));
|
nest.at("/sessionserver/session/minecraft").nest(super::session::nest(db.to_owned()));
|
||||||
|
|
||||||
|
nest.at("/api/profiles/minecraft").nest(super::profile::nest(db.to_owned()));
|
||||||
|
|
||||||
nest
|
nest
|
||||||
}
|
}
|
||||||
|
@ -14,11 +14,11 @@ use tide::{Request, Response, utils::After};
|
|||||||
|
|
||||||
use yggdrasil::*;
|
use yggdrasil::*;
|
||||||
|
|
||||||
mod account;
|
pub mod auth;
|
||||||
mod auth;
|
pub mod authlib;
|
||||||
mod authlib;
|
pub mod profile;
|
||||||
mod minecraft;
|
pub mod session;
|
||||||
mod session;
|
pub mod aliases;
|
||||||
|
|
||||||
pub async fn start(db: &Database) -> anyhow::Result<()> {
|
pub async fn start(db: &Database) -> anyhow::Result<()> {
|
||||||
let mut app = tide::with_state(db.to_owned());
|
let mut app = tide::with_state(db.to_owned());
|
||||||
@ -53,12 +53,14 @@ pub async fn start(db: &Database) -> anyhow::Result<()> {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Routes
|
// Routes
|
||||||
app.at("/authlib/").nest(authlib::nest(db.to_owned()));
|
|
||||||
app.at("/account/").nest(account::nest(db.to_owned()));
|
|
||||||
app.at("/minecraft/").nest(minecraft::nest(db.to_owned()));
|
|
||||||
app.at("/auth/").nest(auth::nest(db.to_owned()));
|
app.at("/auth/").nest(auth::nest(db.to_owned()));
|
||||||
app.at("/session/").nest(session::nest(db.to_owned()));
|
app.at("/session/").nest(session::nest(db.to_owned()));
|
||||||
|
|
||||||
|
app.at("/authlib/").nest(authlib::nest(db.to_owned()));
|
||||||
|
|
||||||
|
app.at("/").nest(aliases::nest(db.to_owned()));
|
||||||
|
|
||||||
|
|
||||||
// Listen
|
// Listen
|
||||||
app.listen(format!("{}:{}", &db.config.address, &db.config.port)).await?;
|
app.listen(format!("{}:{}", &db.config.address, &db.config.port)).await?;
|
||||||
|
|
||||||
|
@ -18,6 +18,10 @@ pub async fn upload_cape(req: Request<Database>) -> Result {
|
|||||||
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn change_cape(req: Request<Database>) -> Result {
|
||||||
|
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
||||||
|
}
|
||||||
|
|
||||||
pub async fn delete_cape(req: Request<Database>) -> Result {
|
pub async fn delete_cape(req: Request<Database>) -> Result {
|
||||||
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
||||||
}
|
}
|
41
src/server/profile/mod.rs
Normal file
41
src/server/profile/mod.rs
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
/*
|
||||||
|
* Yggdrasil: Minecraft authentication server
|
||||||
|
* Copyright (C) 2023 0xf8.dev@proton.me
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
pub use log::{info, log, warn};
|
||||||
|
pub use tide::{Middleware, prelude::*, Request, Response, Result, utils::After};
|
||||||
|
|
||||||
|
pub use yggdrasil::*;
|
||||||
|
|
||||||
|
mod profile;
|
||||||
|
mod skin;
|
||||||
|
mod cape;
|
||||||
|
|
||||||
|
pub fn nest(db: Database) -> tide::Server<Database> {
|
||||||
|
info!("Loading nest");
|
||||||
|
|
||||||
|
let mut nest = tide::with_state(db);
|
||||||
|
|
||||||
|
nest.at("/").post(profile::profiles);
|
||||||
|
|
||||||
|
nest.at("/:uuid").get(profile::profile);
|
||||||
|
|
||||||
|
nest.at("/:uuid/names").get(profile::name_history);
|
||||||
|
|
||||||
|
nest.at("/:uuid/skin").put(skin::put_skin);
|
||||||
|
nest.at("/:uuid/skin/:skin").delete(skin::delete_skin);
|
||||||
|
|
||||||
|
nest.at("/:uuid/capes").put(cape::upload_cape);
|
||||||
|
nest.at("/:uuid/cape/active").put(cape::change_cape);
|
||||||
|
nest.at("/:uuid/cape/active").delete(cape::delete_cape);
|
||||||
|
|
||||||
|
nest
|
||||||
|
}
|
67
src/server/profile/profile.rs
Normal file
67
src/server/profile/profile.rs
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
/*
|
||||||
|
* Yggdrasil: Minecraft authentication server
|
||||||
|
* Copyright (C) 2023 0xf8.dev@proton.me
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
use anyhow::anyhow;
|
||||||
|
use log::{debug, info};
|
||||||
|
use serde_json::Value;
|
||||||
|
use tide::{prelude::*, Request, Result};
|
||||||
|
|
||||||
|
use yggdrasil::Database;
|
||||||
|
use yggdrasil::errors::YggdrasilError;
|
||||||
|
use yggdrasil::structs::game_profile::GameProfile;
|
||||||
|
use yggdrasil::structs::profile::Profile;
|
||||||
|
use yggdrasil::structs::token::Token;
|
||||||
|
|
||||||
|
pub async fn profile(mut req: Request<Database>) -> Result {
|
||||||
|
// TODO: unsigned?
|
||||||
|
let Ok(uuid) = req.param("uuid") else {
|
||||||
|
// No uuid
|
||||||
|
debug!("No uuid");
|
||||||
|
return Err(YggdrasilError::new_bad_request("One or more required fields was missing.").into())
|
||||||
|
};
|
||||||
|
|
||||||
|
// Re-hyphenate if needed
|
||||||
|
let uuid = match uuid.find("-") {
|
||||||
|
None => Token::rehyphenate(uuid.to_string()),
|
||||||
|
Some(_) => uuid.to_string(),
|
||||||
|
};
|
||||||
|
|
||||||
|
let Some(profile) = Profile::from_uuid(req.state(), uuid).await else {
|
||||||
|
return Err(YggdrasilError::new_bad_request("Profile does not exist").into())
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(GameProfile::from_profile(req.state(), &profile).await.into())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn profiles(mut req: Request<Database>) -> Result {
|
||||||
|
let Ok(body) = req.body_json::<Vec<String>>().await else {
|
||||||
|
return Err(YggdrasilError::new_bad_request("One or more required fields was missing.").into())
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut response: Vec<Value> = Vec::new();
|
||||||
|
|
||||||
|
for uuid in body {
|
||||||
|
let profile = Profile::from_uuid(req.state(), uuid).await;
|
||||||
|
|
||||||
|
match profile {
|
||||||
|
None => continue,
|
||||||
|
Some(p) => {
|
||||||
|
response.push(GameProfile::from_profile(req.state(), &p).await)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(serde_json::to_string(&response)?.into())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn name_history(req: Request<Database>) -> Result {
|
||||||
|
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
||||||
|
}
|
@ -24,4 +24,8 @@ pub async fn delete_skin(req: Request<Database>) -> Result {
|
|||||||
// TODO: file uploading
|
// TODO: file uploading
|
||||||
pub async fn put_skin(req: Request<Database>) -> Result {
|
pub async fn put_skin(req: Request<Database>) -> Result {
|
||||||
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn get_skins(req: Request<Database>) -> Result {
|
||||||
|
Err(tide::Error::new(501, anyhow!("Not implemented yet")).into())
|
||||||
}
|
}
|
@ -16,7 +16,7 @@ use yggdrasil::Database;
|
|||||||
use yggdrasil::errors::YggdrasilError;
|
use yggdrasil::errors::YggdrasilError;
|
||||||
use yggdrasil::structs::profile::Profile;
|
use yggdrasil::structs::profile::Profile;
|
||||||
use yggdrasil::structs::session::Session;
|
use yggdrasil::structs::session::Session;
|
||||||
use yggdrasil::structs::textured_object::TexturedObject;
|
use yggdrasil::structs::game_profile::GameProfile;
|
||||||
|
|
||||||
#[derive(Deserialize, Debug)]
|
#[derive(Deserialize, Debug)]
|
||||||
struct HasJoinedBody {
|
struct HasJoinedBody {
|
||||||
@ -54,5 +54,5 @@ pub async fn has_joined(mut req: Request<Database>) -> Result {
|
|||||||
// Remove session
|
// Remove session
|
||||||
session.delete(req.state()).await?;
|
session.delete(req.state()).await?;
|
||||||
|
|
||||||
Ok(TexturedObject::from_profile(req.state(), &profile).await.into())
|
Ok(GameProfile::from_profile(req.state(), &profile).await.into())
|
||||||
}
|
}
|
@ -21,10 +21,11 @@ mod profile;
|
|||||||
pub fn nest(db: Database) -> tide::Server<Database> {
|
pub fn nest(db: Database) -> tide::Server<Database> {
|
||||||
info!("Loading nest");
|
info!("Loading nest");
|
||||||
|
|
||||||
let mut nest = tide::with_state(db);
|
let mut nest = tide::with_state(db.to_owned());
|
||||||
nest.at("minecraft/hasJoined").get(has_joined::has_joined);
|
|
||||||
nest.at("minecraft/join").post(join::join);
|
nest.at("hasJoined").get(has_joined::has_joined);
|
||||||
nest.at("profile/:uuid").get(profile::profile);
|
nest.at("join").post(join::join);
|
||||||
|
nest.at("profile").nest(super::profile::nest(db.to_owned()));
|
||||||
|
|
||||||
nest
|
nest
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,7 @@ use tide::{prelude::*, Request, Result};
|
|||||||
use yggdrasil::Database;
|
use yggdrasil::Database;
|
||||||
use yggdrasil::errors::YggdrasilError;
|
use yggdrasil::errors::YggdrasilError;
|
||||||
use yggdrasil::structs::profile::Profile;
|
use yggdrasil::structs::profile::Profile;
|
||||||
use yggdrasil::structs::textured_object::TexturedObject;
|
use yggdrasil::structs::game_profile::GameProfile;
|
||||||
use yggdrasil::structs::token::Token;
|
use yggdrasil::structs::token::Token;
|
||||||
|
|
||||||
// TODO: unsigned?
|
// TODO: unsigned?
|
||||||
@ -36,5 +36,5 @@ pub async fn profile(mut req: Request<Database>) -> Result {
|
|||||||
return Err(YggdrasilError::new_bad_request("Profile does not exist").into())
|
return Err(YggdrasilError::new_bad_request("Profile does not exist").into())
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(TexturedObject::from_profile(req.state(), &profile).await.into())
|
Ok(GameProfile::from_profile(req.state(), &profile).await.into())
|
||||||
}
|
}
|
@ -14,17 +14,13 @@ pub use tide::{Middleware, prelude::*, Request, Response, Result, utils::After};
|
|||||||
|
|
||||||
pub use yggdrasil::*;
|
pub use yggdrasil::*;
|
||||||
|
|
||||||
mod profiles;
|
|
||||||
mod skin;
|
|
||||||
|
|
||||||
pub fn nest(db: Database) -> tide::Server<Database> {
|
pub fn nest(db: Database) -> tide::Server<Database> {
|
||||||
info!("Loading nest");
|
info!("Loading nest");
|
||||||
|
|
||||||
let mut nest = tide::with_state(db);
|
let mut nest = tide::with_state(db.to_owned());
|
||||||
|
|
||||||
nest.at("profiles/minecraft").post(profiles::profiles);
|
nest.at("/profile").nest(super::profile::nest(db.to_owned()));
|
||||||
nest.at("profile/skins").put(skin::put_skin);
|
nest.at("/profiles").nest(super::profile::nest(db.to_owned()));
|
||||||
nest.at("profile/active").delete(skin::delete_skin);
|
|
||||||
|
|
||||||
nest
|
nest
|
||||||
}
|
}
|
@ -18,10 +18,11 @@ use structs::profile::Profile;
|
|||||||
use crate::*;
|
use crate::*;
|
||||||
|
|
||||||
#[derive(Deserialize, Serialize, Debug)]
|
#[derive(Deserialize, Serialize, Debug)]
|
||||||
pub struct TexturedObject {}
|
pub struct GameProfile {}
|
||||||
|
|
||||||
impl TexturedObject {
|
impl GameProfile {
|
||||||
pub async fn from_profile(db: &Database, profile: &Profile) -> Value {
|
pub async fn from_profile(db: &Database, profile: &Profile) -> Value {
|
||||||
|
// Textures object
|
||||||
let mut object = json!({
|
let mut object = json!({
|
||||||
"timestamp": get_unix_timestamp() as u64,
|
"timestamp": get_unix_timestamp() as u64,
|
||||||
"profile_id": profile.uuid.to_owned(),
|
"profile_id": profile.uuid.to_owned(),
|
||||||
@ -45,6 +46,7 @@ impl TexturedObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// This is the actual game profile
|
||||||
json!({
|
json!({
|
||||||
"id": profile.uuid.replace("-", ""),
|
"id": profile.uuid.replace("-", ""),
|
||||||
"name": profile.name.to_owned(),
|
"name": profile.name.to_owned(),
|
@ -16,6 +16,6 @@ pub mod cape;
|
|||||||
pub mod profile;
|
pub mod profile;
|
||||||
pub mod profile_attributes;
|
pub mod profile_attributes;
|
||||||
pub mod session;
|
pub mod session;
|
||||||
pub mod textured_object;
|
pub mod game_profile;
|
||||||
pub mod token;
|
pub mod token;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user