use async_std::{
fs,
path::{Path, PathBuf},
};
use data::LicenseDatabase;
use jealousy::FromEnv;
use serde::Deserialize;
use std::net::Ipv4Addr;
use tide::{http::mime, log, Response, Result, Server, StatusCode};
use crate::data::Queryable;
mod data;
#[derive(Debug, Deserialize, FromEnv, Clone)]
#[from_env(prefix = "LICENSE_API")]
struct ServerConfig {
#[serde(default = "ServerConfig::default_base_url")]
base_url: String,
database: String,
#[serde(default = "ServerConfig::default_server_ip")]
server_ip: Ipv4Addr,
#[serde(default = "ServerConfig::default_server_port")]
server_port: u16,
}
impl ServerConfig {
fn default_base_url() -> String {
String::from("/")
}
fn default_server_ip() -> Ipv4Addr {
Ipv4Addr::new(0, 0, 0, 0)
}
fn default_server_port() -> u16 {
8080
}
}
#[derive(Debug, Clone)]
struct RunningConfig {
config: ServerConfig,
database: LicenseDatabase,
}
const ERROR_LICENSE_NOT_FOUND: &str =
" is undefined. use the 'list' endpoint to see defined licenses.";
#[async_std::main]
async fn main() -> Result<()> {
log::start();
match ServerConfig::from_env() {
Ok(config) => {
let run_config = RunningConfig {
config: config.clone(),
database: LicenseDatabase::load(config.database.clone()),
};
let mut app = tide::with_state(run_config);
create_routes(&mut app);
app.listen(format!("{}:{}", config.server_ip, config.server_port))
.await?;
}
Err(_) => (),
};
Ok(())
}
fn create_routes(server: &mut Server<RunningConfig>) {
let url = server.clone().state().config.base_url.clone();
let mut base_route = server.at(&url);
base_route.get(get_welcome_message);
base_route.at(":license").get(get_requested_license);
base_route.at("list").get(get_license_list);
base_route.at("list/full").get(get_detailed_license_list);
base_route.at("").get(get_welcome_message);
base_route.at("list/").get(get_license_list);
base_route.at("list/full/").get(get_detailed_license_list);
}
fn build_response(status: StatusCode, data: String) -> Response {
let response = Response::builder(status)
.body(format!("{}\n", data))
.content_type(mime::PLAIN)
.build();
response
}
async fn get_welcome_message(req: tide::Request<RunningConfig>) -> tide::Result<Response> {
log::info!("{:?}", req);
Ok(build_response(
StatusCode::Ok,
format!("{}", "Welcome message goes here").to_string(),
))
}
async fn get_license_list(req: tide::Request<RunningConfig>) -> tide::Result<Response> {
log::info!("{:?}", req);
match req.state().database.get_licenses(5) {
Some(licenses) => Ok(build_response(StatusCode::Ok, format!("{}", licenses))),
None => Ok(build_response(
StatusCode::NotFound,
String::from("License not found"),
)),
}
}
async fn get_detailed_license_list(req: tide::Request<RunningConfig>) -> tide::Result<Response> {
log::info!("{:?}", req);
match req.state().database.get_detailed_licenses() {
Some(licenses) => Ok(build_response(StatusCode::Ok, format!("{}", licenses))),
None => Ok(build_response(
StatusCode::NotFound,
String::from("No licenses found."),
)),
}
}
async fn get_requested_license(req: tide::Request<RunningConfig>) -> tide::Result<Response> {
log::info!("{:?}", req);
let request = req.url().path().strip_prefix("/").unwrap_or("");
let response = match req.state().database.get_license(request) {
Some(license) => {
let source_dir = PathBuf::from(req.state().config.database.clone());
let source_dir = source_dir.parent().unwrap_or(Path::new(""));
let mut license_file_path = PathBuf::from(source_dir);
license_file_path.push(license.filename);
match fs::read_to_string(&license_file_path).await {
Ok(contents) => {
build_response(StatusCode::Accepted, contents.trim_end().to_string())
}
Err(_) => {
log::error!(
"Unable to read `{}`.",
license_file_path.to_string_lossy().to_string()
);
build_response(
StatusCode::NotFound,
format!("ERROR - `{}`{}", request, ERROR_LICENSE_NOT_FOUND).to_string(),
)
}
}
}
None => {
log::error!("License not found: `{}`", request);
build_response(
StatusCode::NotFound,
format!("ERROR - `{}`{}", request, ERROR_LICENSE_NOT_FOUND).to_string(),
)
}
};
Ok(response)
}