feat: process url encoded and json generic forms

This commit is contained in:
Aravinth Manivannan 2022-09-08 17:49:51 +05:30
parent d6ecefa605
commit f79343e9e9
Signed by: realaravinth
GPG key ID: AD9F0F08E855ED88
2 changed files with 167 additions and 1 deletions

159
src/api/v1/forms.rs Normal file
View file

@ -0,0 +1,159 @@
/*
* Copyright (C) 2022 Aravinth Manivannan <realaravinth@batsense.net>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero 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 Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
use std::collections::HashMap;
use actix_web::{web, Error, HttpRequest, HttpResponse, Responder};
use serde::{Deserialize, Serialize};
use super::API_V1_ROUTES;
use crate::AppCtx;
pub mod routes {
use super::*;
#[derive(Debug, Eq, PartialEq, Deserialize, Serialize)]
pub struct Forms {
pub submit: &'static str,
}
impl Forms {
pub const fn new() -> Self {
Self {
submit: "/api/v1/forms/submit",
}
}
}
}
pub fn services(cfg: &mut web::ServiceConfig) {
cfg.service(upload);
}
#[derive(Deserialize, Serialize, Debug)]
#[serde(untagged)]
enum FormDType {
Num(f64),
Str(String),
}
impl FormDType {
fn apply_types(&mut self) {
if let Self::Str(data) = self {
if let Ok(num) = data.parse::<f64>() {
*self = Self::Num(num);
}
}
}
}
pub type FormValue = HashMap<String, FormDType>;
#[actix_web_codegen_const_routes::post(path = "API_V1_ROUTES.forms.submit")]
async fn upload(
req: HttpRequest,
ctx: AppCtx,
payload: web::Either<web::Json<serde_json::Value>, web::Form<FormValue>>,
) -> Result<impl Responder, Error> {
let c = req.connection_info();
let host = c.host();
let path = req.uri();
let data = match payload {
web::Either::Left(json) => json.into_inner(),
web::Either::Right(form) => {
let mut form = form.into_inner();
for (_, v) in form.iter_mut() {
v.apply_types();
}
serde_json::to_value(&form).unwrap()
}
};
println!("{:?}", data);
//ctx.gitea.report(&payload).await;
Ok(HttpResponse::Ok().json(data))
}
#[cfg(test)]
pub mod tests {
use actix_web::{
http::{header, StatusCode},
test, App,
};
use super::*;
use crate::*;
#[derive(Serialize, Clone, Deserialize, PartialEq)]
struct Foo {
foo: String,
num: f64,
}
#[actix_rt::test]
async fn submit_works() {
// const USERNAME: &str = "index_works";
// const PASSWORD: &str = "23k4j;123k4j1;l23kj4";
let settings = Settings::new().unwrap();
// let settings = Settings::new().unwrap();
let ctx = AppCtx::new(crate::ctx::Ctx::new(&settings).await);
let app = test::init_service(
App::new()
.app_data(ctx.clone())
.configure(crate::routes::services),
)
.await;
let foo = Foo {
foo: "Foo".into(),
num: 2.33,
};
// upload json
let upload_json = test::call_service(
&app,
test::TestRequest::post()
.uri(API_V1_ROUTES.forms.submit)
.set_json(&foo)
.to_request(),
)
.await;
if upload_json.status() != StatusCode::OK {
let resp_err: crate::errors::ErrorToResponse = test::read_body_json(upload_json).await;
panic!("{:?}", resp_err.error);
}
assert_eq!(upload_json.status(), StatusCode::OK);
let json: serde_json::Value = test::read_body_json(upload_json).await;
// upload url encoded
let upload_form = test::call_service(
&app,
test::TestRequest::post()
.uri(API_V1_ROUTES.forms.submit)
.set_form(&foo)
.to_request(),
)
.await;
if upload_form.status() != StatusCode::OK {
let resp_err: crate::errors::ErrorToResponse = test::read_body_json(upload_form).await;
panic!("{:?}", resp_err.error);
}
assert_eq!(upload_form.status(), StatusCode::OK);
let form: serde_json::Value = test::read_body_json(upload_form).await;
assert_eq!(form, json);
}
}

View file

@ -16,6 +16,7 @@
*/ */
use actix_web::web; use actix_web::web;
pub mod forms;
pub mod meta; pub mod meta;
pub const API_V1_ROUTES: routes::Routes = routes::Routes::new(); pub const API_V1_ROUTES: routes::Routes = routes::Routes::new();
@ -25,18 +26,24 @@ pub struct SignedInUser(String);
pub fn services(cfg: &mut web::ServiceConfig) { pub fn services(cfg: &mut web::ServiceConfig) {
meta::services(cfg); meta::services(cfg);
forms::services(cfg);
} }
pub mod routes { pub mod routes {
use crate::api::v1::forms::routes::Forms;
use crate::api::v1::meta::routes::Meta; use crate::api::v1::meta::routes::Meta;
pub struct Routes { pub struct Routes {
pub meta: Meta, pub meta: Meta,
pub forms: Forms,
} }
impl Routes { impl Routes {
pub const fn new() -> Self { pub const fn new() -> Self {
Self { meta: Meta::new() } Self {
meta: Meta::new(),
forms: Forms::new(),
}
} }
} }
} }