Rust Web 全栈开发(九):增加教师管理功能

Rust Web 全栈开发(九):增加教师管理功能

参考视频:https://www.bilibili.com/video/BV1RP4y1G7KF

继续之前的 Actix 项目,项目现状如下所示:

在这里插入图片描述

新增教师结构体

在 models 目录下,新建一个 teacher.rs。

为 CreateTeacher 和 UpdateTeacher 分别实现了 From trait。

use actix_web::web;
use serde::{Deserialize, Serialize};

#[derive(Deserialize, Serialize, Debug, Clone, sqlx::FromRow)]
pub struct Teacher {
    pub id: i32,
    pub name: String,
    pub picture_url: String,
    pub profile: String,
}

#[derive(Deserialize, Debug, Clone)]
pub struct CreateTeacher {
    pub name: String,
    pub picture_url: String,
    pub profile: String,
}

impl From<web::Json<CreateTeacher>> for CreateTeacher {
    fn from(new_teacher: web::Json<CreateTeacher>) -> Self {
        CreateTeacher {
            name: new_teacher.name.clone(),
            picture_url: new_teacher.picture_url.clone(),
            profile: new_teacher.profile.clone(),
        }
    }
}

#[derive(Deserialize, Debug, Clone)]
pub struct UpdateTeacher {
    pub name: Option<String>,
    pub picture_url: Option<String>,
    pub profile: Option<String>,
}

impl From<web::Json<UpdateTeacher>> for UpdateTeacher {
    fn from(update_teacher: web::Json<UpdateTeacher>) -> Self {
        UpdateTeacher {
            name: update_teacher.name.clone(),
            picture_url: update_teacher.picture_url.clone(),
            profile: update_teacher.profile.clone(),
        }
    }
}

对应要修改 models/mod.rs,新增:

pub mod teacher;

数据库准备

在这里插入图片描述

在名为 course 的 MySQL 数据库中,新建一个名为 teacher 的表:

在这里插入图片描述

新增访问 MySQL 数据库中 teacher 表的方法

在 dbaccess 目录下,新建一个 teacher.rs,其中涉及了对 teacher 表增删查改的方法。

use sqlx::MySqlPool;
use crate::errors::MyError;
use crate::models::teacher::{CreateTeacher, Teacher, UpdateTeacher};

pub async fn post_new_teacher_db(pool: &MySqlPool, new_teacher: CreateTeacher) -> Result<(), MyError> {
    let _insert_query = sqlx::query!(
        "INSERT INTO teacher (name, picture_url, profile)
            VALUES (?, ?, ?)",
        new_teacher.name,
        new_teacher.picture_url,
        new_teacher.profile,
    )
        .execute(pool)
        .await?;

    Ok(())
}

pub async fn delete_teacher_db(pool: &MySqlPool, teacher_id: i32) -> Result<String, MyError> {
    let row = sqlx::query!(
        "DELETE FROM teacher
            WHERE id = ?",
        teacher_id,
    )
        .execute(pool)
        .await
        .map_err(|_err| MyError::DBError("Unable to delete teacher".into()))?;

    Ok(format!("Deleted {:?} record", row))
}

pub async fn update_teacher_details_db(
    pool: &MySqlPool,
    teacher_id: i32,
    update_teacher: UpdateTeacher
) -> Result<String, MyError> {
    let current_teacher_row: Teacher = sqlx::query_as(
        "SELECT id, name, picture_url, profile
                FROM teacher
                WHERE id = ?"
    )
        .bind(teacher_id)
        .fetch_one(pool) // 获取单条记录
        .await
        .map_err(|_err| MyError::NotFound("Teacher Id not found".into()))?;

    let teacher = Teacher {
        id: current_teacher_row.id,
        name: if let Some(name) = update_teacher.name {
            name
        } else {
            current_teacher_row.name
        },
        picture_url: if let Some(picture_url) = update_teacher.picture_url {
            picture_url
        } else {
            current_teacher_row.picture_url
        },
        profile: if let Some(profile) = update_teacher.profile {
            profile
        } else {
            current_teacher_row.profile
        },
    };

    let row = sqlx::query!(
        "UPDATE teacher
            SET name = ?, picture_url = ?, profile = ?
            WHERE id = ?",
        teacher.name,
        teacher.picture_url,
        teacher.profile,
        teacher_id,
    )
        .execute(pool)
        .await?;

    Ok(format!("Update {:?} record", row))
}
pub async fn get_all_teachers_db(pool: &MySqlPool) -> Result<Vec<Teacher>, MyError> {
    let rows: Vec<Teacher> = sqlx::query_as(
        "SELECT id, name, picture_url, profile
                FROM teacher"
    )
        .fetch_all(pool) // 获取所有记录
        .await?;

    match rows.len() {
        0 => Err(MyError::NotFound("Teacher not found".into())),
        _ => Ok(rows),
    }
}

pub async fn get_teacher_details_db(pool: &MySqlPool, teacher_id: i32) -> Result<Teacher, MyError> {
    let row = sqlx::query_as(
        "SELECT id, name, picture_url, profile
                FROM teacher
                WHERE id = ?"
    )
        .bind(teacher_id)
        .fetch_one(pool) // 获取单条记录
        .await
        .map(|teacher: Teacher| Teacher {
            id: teacher.id,
            name: teacher.name,
            picture_url: teacher.picture_url,
            profile: teacher.profile,
        })
        .map_err(|_err| MyError::NotFound("Teacher Id not found".into()))?;

    Ok(row)
}

对应要修改 dbaccess/mod.rs,新增:

pub mod teacher;

新增 Handler

在 handlers 目录下,新建一个 teacher.rs。

其中的函数调用 dbaccess/teacher.rs 的方法,得到结果,转换为 HTTP Response。

use std::env;
use std::sync::Mutex;
use actix_web::{web, HttpResponse};
use actix_web::web::Data;
use dotenv::dotenv;
use sqlx::mysql::MySqlPoolOptions;
use crate::errors::MyError;
use crate::state::AppState;
use crate::dbaccess::teacher::*;
use crate::models::teacher::{CreateTeacher, UpdateTeacher};

pub async fn post_new_teacher(
    new_teacher: web::Json<CreateTeacher>,
    app_state: web::Data<AppState>,
) -> Result<HttpResponse, MyError> {
    post_new_teacher_db(&app_state.db, new_teacher.into())
    .await
    .map(|_| HttpResponse::Ok().json("Post new teacher successfully."))
}

pub async fn get_all_teachers(app_state: web::Data<AppState>) -> Result<HttpResponse, MyError> {
    get_all_teachers_db(&app_state.db)
        .await
        .map(|teachers| HttpResponse::Ok().json(teachers))
}

pub async fn get_teacher_detail(
    app_state: web::Data<AppState>,
    params: web::Path<i32>
) -> Result<HttpResponse, MyError> {
    let teacher_id = params.into_inner();
    get_teacher_details_db(&app_state.db, teacher_id)
        .await
        .map(|teacher| HttpResponse::Ok().json(teacher))
}

pub async fn update_teacher_detail(
    app_state: web::Data<AppState>,
    update_teacher: web::Json<UpdateTeacher>,
    params: web::Path<i32>
) -> Result<HttpResponse, MyError> {
    let teacher_id = params.into_inner();
    update_teacher_details_db(&app_state.db, teacher_id, update_teacher.into())
        .await
        .map(|msg| HttpResponse::Ok().json(msg))
}

pub async fn delete_teacher(
    app_state: web::Data<AppState>,
    params: web::Path<i32>
) -> Result<HttpResponse, MyError> {
    let teacher_id = params.into_inner();
    delete_teacher_db(&app_state.db, teacher_id)
        .await
        .map(|msg| HttpResponse::Ok().json(msg))
}

#[cfg(test)]
mod tests {
    use super::*;
    use actix_web::http::StatusCode;
    use std::sync::Mutex;
    use dotenv::dotenv;
    use sqlx::mysql::MySqlPoolOptions;
    use std::env;
    use actix_web::ResponseError;

    // #[ignore]
    #[actix_rt::test]
    async fn post_new_teacher_success() {
        // 检测并读取 .env 文件中的内容,若不存在也会跳过异常
        dotenv().ok();

        let db_url = env::var("DATABASE_URL")
            .expect("DATABASE_URL 没有在 .env 文件里设置");

        // 创建数据库连接池
        let db_pool = MySqlPoolOptions::new()
            .connect(&db_url)
            .await
            .unwrap();

        let app_state: web::Data<AppState> = web::Data::new(AppState {
            health_check_response: "".to_string(),
            visit_count: Mutex::new(0),
            db: db_pool,
        });

        let teacher = web::Json(CreateTeacher {
            name: "A New Teacher".into(),
            picture_url: "https://i2.hdslb.com/bfs/article/1f8a3ece569b3d61903fe2062cf71d96435d6f8b.jpg@1192w.avif".into(),
            profile: "This is a test profile".into(),
        });

        // 模拟添加教师的请求
        let response = post_new_teacher(teacher, app_state).await.unwrap();

        assert_eq!(response.status(), StatusCode::OK);
    }

    #[actix_rt::test]
    async fn get_all_teachers_success() {
        dotenv().ok();
        let db_url = env::var("DATABASE_URL").expect("DATABASE_URL 没有在 .env 文件里设置");
        let db_pool = MySqlPoolOptions::new().connect(&db_url).await.unwrap();
        let app_state: web::Data<AppState> = web::Data::new(AppState {
            health_check_response: "".to_string(),
            visit_count: Mutex::new(0),
            db: db_pool,
        });

        let response = get_all_teachers(app_state).await.unwrap();

        assert_eq!(response.status(), StatusCode::OK);
    }

    #[actix_rt::test]
    async fn get_teacher_detail_success() {
        dotenv().ok();
        let db_url = env::var("DATABASE_URL").expect("DATABASE_URL 没有在 .env 文件里设置");
        let db_pool = MySqlPoolOptions::new().connect(&db_url).await.unwrap();
        let app_state: web::Data<AppState> = web::Data::new(AppState {
            health_check_response: "".to_string(),
            visit_count: Mutex::new(0),
            db: db_pool,
        });

        let params: web::Path<i32> = web::Path::from(1);
        let response = get_teacher_detail(app_state, params).await.unwrap();

        assert_eq!(response.status(), StatusCode::OK);
    }

    #[actix_rt::test]
    async fn get_teacher_detail_failure() {
        dotenv().ok();
        let db_url = env::var("DATABASE_URL").expect("DATABASE_URL 没有在 .env 文件里设置");
        let db_pool = MySqlPoolOptions::new().connect(&db_url).await.unwrap();
        let app_state: web::Data<AppState> = web::Data::new(AppState {
            health_check_response: "".to_string(),
            visit_count: Mutex::new(0),
            db: db_pool,
        });

        let params: web::Path<i32> = web::Path::from(100);
        let response = get_teacher_detail(app_state, params).await;

        match response {
            Ok(_) => println!("Something went wrong"),
            Err(err) => assert_eq!(err.status_code(), StatusCode::NOT_FOUND),
        }
    }

    #[actix_rt::test]
    async fn update_teacher_success() {
        dotenv().ok();
        let db_url = env::var("DATABASE_URL").expect("DATABASE_URL 没有在 .env 文件里设置");
        let db_pool = MySqlPoolOptions::new().connect(&db_url).await.unwrap();
        let app_state: web::Data<AppState> = web::Data::new(AppState {
            health_check_response: "".to_string(),
            visit_count: Mutex::new(0),
            db: db_pool,
        });

        let params: web::Path<i32> = web::Path::from(2);
        let update_param = web::Json(UpdateTeacher {
            name: Some("Teacher name changed".into()),
            picture_url: Some("https://i2.hdslb.com/bfs/article/6e82414fa0c96d530caa120caed421240df75cfc.jpg@1192w.avif".into()),
            profile: Some("This is a update profile".into()),
        });
        let response = update_teacher_detail(app_state, update_param, params).await.unwrap();

        assert_eq!(response.status(), StatusCode::OK);

    }

    // #[ignore]
    #[actix_rt::test]
    async fn delete_teacher_success() {
        dotenv().ok();
        let db_url = env::var("DATABASE_URL").expect("DATABASE_URL 没有在 .env 文件里设置");
        let db_pool = MySqlPoolOptions::new().connect(&db_url).await.unwrap();
        let app_state: web::Data<AppState> = web::Data::new(AppState {
            health_check_response: "".to_string(),
            visit_count: Mutex::new(0),
            db: db_pool,
        });

        let params: web::Path<i32> = web::Path::from(3);
        let response = delete_teacher(app_state, params).await.unwrap();

        assert_eq!(response.status(), StatusCode::OK);
    }

    #[actix_rt::test]
    async fn delete_teacher_failure() {
        dotenv().ok();
        let db_url = env::var("DATABASE_URL").expect("DATABASE_URL 没有在 .env 文件里设置");
        let db_pool = MySqlPoolOptions::new().connect(&db_url).await.unwrap();
        let app_state: web::Data<AppState> = web::Data::new(AppState {
            health_check_response: "".to_string(),
            visit_count: Mutex::new(0),
            db: db_pool,
        });

        let params: web::Path<i32> = web::Path::from(100);
        let response = delete_teacher(app_state, params).await;

        match response {
            Ok(_) => println!("Something went wrong"),
            Err(err) => assert_eq!(err.status_code(), StatusCode::NOT_FOUND),
        }
    }
}

对应要修改 handlers/mod.rs,新增:

pub mod teacher;

新增 Router

修改 routers.rs,新增关于教师的路由:

pub fn teacher_routes(cfg: &mut web::ServiceConfig) {
    cfg.service(
        web::scope("/teachers")
            .route("/", web::post().to(post_new_teacher))
            .route("/", web::get().to(get_all_teachers))
            .route("/{teacher_id}", web::get().to(get_teacher_detail))
            .route("/{teacher_id}", web::put().to(update_teacher_detail))
            .route("/{teacher_id}", web::delete().to(delete_teacher))
    );
}

修改 Error

修改 errors.rs,在 MyError 中新增一种 InvalidInput 错误类型,表示输入参数错误。

#[derive(Debug, Serialize)]
pub enum MyError {
    DBError(String),
    ActixError(String),
    #[allow(dead_code)]
    NotFound(String),
    InvalidInput(String),
}

对应修改 MyError 的 error_response 方法,以及 error::ResponseError trait 中的 status_code 方法,发生输入参数错误时,返回 HTTP 400 Bad Request。

impl MyError {
    fn error_response(&self) -> String {
        match self {
            MyError::DBError(msg) => {
                println!("Database error occurred: {:?}", msg);
                "Database error".into()
            }
            MyError::ActixError(msg) => {
                println!("Server error occurred: {:?}", msg);
                "Internal server error".into()
            }
            MyError::NotFound(msg) => {
                println!("Not found error occurred: {:?}", msg);
                msg.into()
            }
            MyError::InvalidInput(msg) => {
                println!("Invalid input error occurred: {:?}", msg);
                msg.into()
            }
        }
    }
}

impl error::ResponseError for MyError {
    fn status_code(&self) -> StatusCode {
        match self {
            MyError::DBError(_msg) | MyError::ActixError(_msg) => StatusCode::INTERNAL_SERVER_ERROR,
            MyError::NotFound(_msg) => StatusCode::NOT_FOUND,
            MyError::InvalidInput(_msg) => StatusCode::BAD_REQUEST,
        }
    }

    fn error_response(&self) -> HttpResponse {
        HttpResponse::build(self.status_code()).json(MyErrorResponse {
            error_message: self.error_response(),
        })
    }
}

修改 teacher_service

修改 webservice/bin/teacher_service.rs:

    let app = move || {
        App::new()
            .app_data(shared_data.clone())
            .app_data(web::JsonConfig::default().error_handler(|_err, _req| {
                MyError::InvalidInput("Please provide valid json input".to_string()).into()
            }))
            .configure(general_routes)
            .configure(course_routes)
            .configure(teacher_routes)
    };

Web 应用程序新增 teacher_routes 路由。

在应用程序执行前,先执行输入参数错误的 handler,返回 MyError::InvalidInput 错误信息。

测试

测试前的数据库 teacher 表:

在这里插入图片描述

cd 到 webservice,在终端执行命令 cargo test teacher,7 个关于教师的测试都通过了:

在这里插入图片描述

测试后的数据库 teacher 表:

在这里插入图片描述

可以看出:新插入了一个教师信息,删除了一个教师信息,更新了一个教师信息。

执行命令 cargo run,在浏览器中测试功能:

在这里插入图片描述

在这里插入图片描述

尾声

修改 handlers/teacher.rs,在 post_new_teacher_success 和 delete_teacher_success 两个测试函数上打上 #[ignore] 注释。

这样在之后执行命令 cargo test 时,会忽略这两个测试。

我们发现在 handlers/teacher.rs 的 test 模块的每一个测试函数中,都有这一段连接 MySQL 数据库、创建 AppState 的代码:

        // 检测并读取 .env 文件中的内容,若不存在也会跳过异常
        dotenv().ok();

        let db_url = env::var("DATABASE_URL")
            .expect("DATABASE_URL 没有在 .env 文件里设置");

        // 创建数据库连接池
        let db_pool = MySqlPoolOptions::new()
            .connect(&db_url)
            .await
            .unwrap();

        let app_state: web::Data<AppState> = web::Data::new(AppState {
            health_check_response: "".to_string(),
            visit_count: Mutex::new(0),
            db: db_pool,
        });

我们将其提取为两个独立的函数:

    async fn create_db_pool() -> MySqlPool {
        // 检测并读取 .env 文件中的内容,若不存在也会跳过异常
        dotenv().ok();

        let db_url = env::var("DATABASE_URL")
            .expect("DATABASE_URL 没有在 .env 文件里设置");

        // 创建数据库连接池
        MySqlPoolOptions::new()
            .connect(&db_url)
            .await
            .unwrap()
    }

    async fn create_app_state() -> web::Data<AppState> {
        let db_pool = create_db_pool().await;

        web::Data::new(AppState {
            health_check_response: "".to_string(),
            visit_count: Mutex::new(0),
            db: db_pool,
        })
    }

之后,在每一个测试函数中,创建 AppState 都可以用一行代码搞定:

let app_state = create_app_state().await;

同理,对 handlers/course.rs 中的 test 模块也这样优化。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

UestcXiye

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值