From 66f554f370ebb7b844cee2ba2cbf0ed45d0d9731 Mon Sep 17 00:00:00 2001 From: thuan2172001 Date: Thu, 28 Mar 2024 23:01:06 +0700 Subject: [PATCH] fix: structure --- Cargo.lock | 19 ++----- Cargo.toml | 2 +- src/adapter/Cargo.toml | 12 ++-- src/adapter/src/repositories/grpc/config.rs | 15 ----- src/adapter/src/repositories/grpc/mod.rs | 1 - .../repositories/grpc/models/gpt_answer.rs | 56 ++++++++++++++++--- .../repositories/postgres/models/question.rs | 9 ++- .../src/repositories/postgres/question_db.rs | 18 +++--- .../src/repositories/repository_test.rs | 2 +- src/common/Cargo.toml | 11 ++++ src/{grpc => common}/build.rs | 0 src/{grpc => common}/proto/gpt_answer.proto | 0 src/common/src/grpc/gpt_answer.rs | 9 +++ .../src/interfaces => common/src/grpc}/mod.rs | 0 src/common/src/lib.rs | 1 + src/core/Cargo.toml | 2 +- src/core/src/common/errors.rs | 9 ++- src/core/src/entities/pagination_entity.rs | 44 ++++----------- src/core/src/entities/question_filter.rs | 49 +++++----------- .../Cargo.toml | 8 +-- .../build.rs | 0 .../config/00-default.toml | 4 +- .../src/controllers/gpt_answer.rs | 49 ++++++++++++++++ .../src/controllers/mod.rs | 0 .../src/lib.rs | 0 .../src/main.rs | 23 ++++++-- .../src/options.rs | 23 ++------ src/grpc/Cargo.toml | 38 ------------- src/grpc/src/interfaces/gpt_answer.rs | 3 - src/grpc/src/lib.rs | 1 - src/grpc_server/src/controllers/gpt_answer.rs | 52 ----------------- src/public/Cargo.toml | 21 +++---- src/public/config/00-default.toml | 5 +- src/public/src/controllers/question.rs | 49 ++++++++-------- src/public/src/errors.rs | 3 +- src/public/src/main.rs | 13 ++--- src/public/src/options.rs | 20 +++---- src/public/src/router.rs | 22 +++----- src/public/tests/questions_router_test.rs | 11 +--- 39 files changed, 268 insertions(+), 336 deletions(-) delete mode 100644 src/adapter/src/repositories/grpc/config.rs rename src/{grpc => common}/build.rs (100%) rename src/{grpc => common}/proto/gpt_answer.proto (100%) create mode 100644 src/common/src/grpc/gpt_answer.rs rename src/{grpc/src/interfaces => common/src/grpc}/mod.rs (100%) rename src/{grpc_server => gpt_answer_service}/Cargo.toml (90%) rename src/{grpc_server => gpt_answer_service}/build.rs (100%) rename src/{grpc_server => gpt_answer_service}/config/00-default.toml (60%) create mode 100644 src/gpt_answer_service/src/controllers/gpt_answer.rs rename src/{grpc_server => gpt_answer_service}/src/controllers/mod.rs (100%) rename src/{grpc_server => gpt_answer_service}/src/lib.rs (100%) rename src/{grpc_server => gpt_answer_service}/src/main.rs (64%) rename src/{grpc_server => gpt_answer_service}/src/options.rs (50%) delete mode 100644 src/grpc/Cargo.toml delete mode 100644 src/grpc/src/interfaces/gpt_answer.rs delete mode 100644 src/grpc/src/lib.rs delete mode 100644 src/grpc_server/src/controllers/gpt_answer.rs diff --git a/Cargo.lock b/Cargo.lock index 41b2555..94d7717 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -8,10 +8,10 @@ version = "0.0.1" dependencies = [ "anyhow", "async-trait", + "common", "deadpool-diesel", "diesel", "diesel_migrations", - "grpc_interface", "rust_core", "serde", "testcontainers-modules", @@ -330,7 +330,6 @@ dependencies = [ "deadpool-diesel", "diesel", "diesel_migrations", - "grpc_interface", "openssl", "opentelemetry", "rand", @@ -361,7 +360,10 @@ dependencies = [ "opentelemetry-otlp", "opentelemetry-semantic-conventions", "opentelemetry_sdk", + "prost", "serde", + "tonic", + "tonic-build", "tracing", "tracing-bunyan-formatter", "tracing-opentelemetry", @@ -821,22 +823,11 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b" [[package]] -name = "grpc_interface" -version = "0.0.1" -dependencies = [ - "glob", - "prost", - "tonic", - "tonic-build", -] - -[[package]] -name = "grpc_server" +name = "gpt_answer_service" version = "0.0.1" dependencies = [ "clap", "common", - "grpc_interface", "opentelemetry", "prost", "readonly", diff --git a/Cargo.toml b/Cargo.toml index 25646c6..0064c08 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,3 +1,3 @@ [workspace] -members = ["src/public", "src/grpc_server"] +members = ["src/public", "src/gpt_answer_service"] resolver = "2" diff --git a/src/adapter/Cargo.toml b/src/adapter/Cargo.toml index d04f690..01a8ed8 100644 --- a/src/adapter/Cargo.toml +++ b/src/adapter/Cargo.toml @@ -12,6 +12,12 @@ autoexamples = true autotests = true autobenches = true +[dependencies.rust_core] +path = "../core" + +[dependencies.common] +path = "../common" + [dependencies] diesel_migrations = "2.1.0" @@ -26,9 +32,6 @@ features = ["postgres", "serde"] version = "2.1.4" features = ["postgres", "postgres_backend", "uuid"] -[dependencies.rust_core] -path = "../core" - [dependencies.serde] version = "1.0" features = ["derive"] @@ -46,6 +49,3 @@ version = "1.0.80" [dependencies.tonic] version = "0.11.0" - -[dependencies.grpc_interface] -path = "../grpc" diff --git a/src/adapter/src/repositories/grpc/config.rs b/src/adapter/src/repositories/grpc/config.rs deleted file mode 100644 index c078b18..0000000 --- a/src/adapter/src/repositories/grpc/config.rs +++ /dev/null @@ -1,15 +0,0 @@ -use serde::Deserialize; - -/// Represents servers configuration options. -#[derive(Deserialize, Debug)] -pub struct GrpcServers { - /// Configuration for using in-memory database. - pub gpt_answer_service: Option, -} - -/// Represents service server configuration. -#[derive(Debug, Deserialize, Clone)] -pub struct ServiceServer { - /// URL for the server. - pub url: String, -} diff --git a/src/adapter/src/repositories/grpc/mod.rs b/src/adapter/src/repositories/grpc/mod.rs index b6a44ac..c446ac8 100644 --- a/src/adapter/src/repositories/grpc/mod.rs +++ b/src/adapter/src/repositories/grpc/mod.rs @@ -1,2 +1 @@ -pub mod config; pub mod models; diff --git a/src/adapter/src/repositories/grpc/models/gpt_answer.rs b/src/adapter/src/repositories/grpc/models/gpt_answer.rs index 4e42922..8073f03 100644 --- a/src/adapter/src/repositories/grpc/models/gpt_answer.rs +++ b/src/adapter/src/repositories/grpc/models/gpt_answer.rs @@ -1,40 +1,80 @@ -use rust_core::common::errors::CoreError; use tonic::transport::Channel; -use grpc_interface::interfaces::gpt_answer::gpt_answer::{ +use common::grpc::gpt_answer::gpt_answer::{ gpt_answer_service_client::GptAnswerServiceClient, GetAnswerPayload, }; +use rust_core::common::errors::CoreError; +/// gRPC client for interacting with a GPT (Generative Pre-trained Transformer) answer service. +/// +/// This struct represents a client for making gRPC calls to a GPT answer service. It provides +/// methods for connecting to the service, sending a question, and receiving an answer. pub struct GptAnswerGrpcClient { client: GptAnswerServiceClient, } impl GptAnswerGrpcClient { + /// Creates a new `GptAnswerGrpcClient` instance with the provided gRPC channel. + /// + /// # Arguments + /// + /// * `channel`: A `Channel` representing the gRPC communication channel. + /// + /// # Returns + /// + /// Returns a new instance of `GptAnswerGrpcClient`. fn new(channel: Channel) -> Self { let client = GptAnswerServiceClient::new(channel); Self { client } } - pub async fn get_instance(uri: &'static str) -> Result { - let channel = Channel::from_static(uri).connect().await.map_err(|err| { - eprintln!("Error connecting to GPT: {:?}", err); - CoreError::InternalError - })?; + /// Establishes a connection to the GPT answer service at the specified URI. + /// + /// # Arguments + /// + /// * `uri`: The URI of the GPT answer service. + /// + /// # Returns + /// + /// Returns a `Result` containing the connected `GptAnswerGrpcClient` if successful, + /// or a `CoreError` if an error occurs during connection. + pub async fn connect(uri: &'static str) -> Result { + // Establish connection to the gRPC server + let channel: Channel = match Channel::from_static(uri).connect().await { + Ok(channel) => channel, + Err(err) => { + return Err(CoreError::InternalError(err.into())); + } + }; + // Create a new client instance with the connected channel let client = Self::new(channel); Ok(client) } + /// Sends a question to the GPT answer service and retrieves the generated answer. + /// + /// # Arguments + /// + /// * `question`: A `&str` representing the question to be sent to the service. + /// + /// # Returns + /// + /// Returns a `Result` containing the generated answer as a `String` if successful, + /// or a `CoreError` if an error occurs during communication with the service. pub async fn get_answer(&mut self, question: &str) -> Result { + // Create a gRPC request with the question payload let request = tonic::Request::new(GetAnswerPayload { question: question.to_string(), }); + // Send the request to the gRPC server and await the response let response = self.client.get_answer(request).await.map_err(|err| { eprintln!("Error getting answer from GPT: {:?}", err); - CoreError::InternalError + CoreError::InternalError(err.into()) })?; + // Extract and return the answer from the response Ok(response.into_inner().answer) } } diff --git a/src/adapter/src/repositories/postgres/models/question.rs b/src/adapter/src/repositories/postgres/models/question.rs index 93e804c..ad2aa18 100644 --- a/src/adapter/src/repositories/postgres/models/question.rs +++ b/src/adapter/src/repositories/postgres/models/question.rs @@ -1,12 +1,11 @@ -use std::{ - io::{Error, ErrorKind}, - time::SystemTime, -}; +use std::io::{Error, ErrorKind}; +use std::time::SystemTime; use diesel::{AsChangeset, Identifiable, Insertable, Queryable, Selectable}; -use rust_core::entities::question::{QuestionEntity, QuestionId}; use serde::Serialize; +use rust_core::entities::question::{QuestionEntity, QuestionId}; + #[derive(Debug, Queryable, Serialize, Selectable, Insertable, AsChangeset, Identifiable)] #[diesel(table_name = super::super::schema::questions)] #[cfg_attr(feature = "postgres", derive(diesel::pg::Pg))] diff --git a/src/adapter/src/repositories/postgres/question_db.rs b/src/adapter/src/repositories/postgres/question_db.rs index 9b3b40d..9627a7a 100644 --- a/src/adapter/src/repositories/postgres/question_db.rs +++ b/src/adapter/src/repositories/postgres/question_db.rs @@ -37,14 +37,14 @@ impl QuestionPort for QuestionDBRepository { .await .unwrap() .interact(move |conn| { - let question = - QuestionModel::try_from(question).map_err(|_| CoreError::InternalError)?; + let question = QuestionModel::try_from(question) + .map_err(|err| CoreError::InternalError(err.into()))?; let response = insert_into(questions) .values(&question) .get_result::(conn) .map_err(|err| match err { diesel::result::Error::NotFound => CoreError::NotFound, - _ => CoreError::InternalError, + _ => CoreError::InternalError(err.into()), }) .unwrap(); Ok(response.into()) @@ -59,14 +59,14 @@ impl QuestionPort for QuestionDBRepository { .await .unwrap() .interact(move |conn| { - let question = - QuestionModel::try_from(question).map_err(|_| CoreError::InternalError)?; + let question = QuestionModel::try_from(question) + .map_err(|err| CoreError::InternalError(err.into()))?; let response = update(questions.filter(id.eq(question.id))) .set(&question) .get_result::(conn) .map_err(|err| match err { diesel::result::Error::NotFound => CoreError::NotFound, - _ => CoreError::InternalError, + _ => CoreError::InternalError(err.into()), })? .into(); @@ -87,7 +87,7 @@ impl QuestionPort for QuestionDBRepository { .execute(conn) .map_err(|err| match err { diesel::result::Error::NotFound => CoreError::NotFound, - _ => CoreError::InternalError, + _ => CoreError::InternalError(err.into()), })?; Ok(()) @@ -109,7 +109,7 @@ impl QuestionPort for QuestionDBRepository { .first(conn) .map_err(|err| match err { diesel::result::Error::NotFound => CoreError::NotFound, - _ => CoreError::InternalError, + _ => CoreError::InternalError(err.into()), })? .into(); @@ -133,7 +133,7 @@ impl QuestionPort for QuestionDBRepository { .load(conn) .map_err(|err| match err { diesel::result::Error::NotFound => CoreError::NotFound, - _ => CoreError::InternalError, + _ => CoreError::InternalError(err.into()), })?; Ok(question_list diff --git a/src/adapter/src/repositories/repository_test.rs b/src/adapter/src/repositories/repository_test.rs index 495e8bb..1b12f5a 100644 --- a/src/adapter/src/repositories/repository_test.rs +++ b/src/adapter/src/repositories/repository_test.rs @@ -51,7 +51,7 @@ mod tests { query_params.insert("start".to_string(), "0".to_string()); query_params.insert("end".to_string(), "10".to_string()); - let pagination = match PaginationEntity::from_query(&query_params) { + let pagination = match PaginationEntity::try_from(query_params) { Ok(pagination_entity) => pagination_entity, Err(err) => { panic!("Failed to parse pagination entity: {:?}", err); diff --git a/src/common/Cargo.toml b/src/common/Cargo.toml index c727049..4bdee6d 100644 --- a/src/common/Cargo.toml +++ b/src/common/Cargo.toml @@ -39,3 +39,14 @@ features = ["derive"] [dependencies.tracing-subscriber] version = "0.3.18" features = ["env-filter"] + + +[dependencies.tonic] +version = "0.11.0" + +[dependencies.prost] +version = "0.12.3" + +[build-dependencies] +tonic-build = "0.11.0" +glob = "0.3.1" diff --git a/src/grpc/build.rs b/src/common/build.rs similarity index 100% rename from src/grpc/build.rs rename to src/common/build.rs diff --git a/src/grpc/proto/gpt_answer.proto b/src/common/proto/gpt_answer.proto similarity index 100% rename from src/grpc/proto/gpt_answer.proto rename to src/common/proto/gpt_answer.proto diff --git a/src/common/src/grpc/gpt_answer.rs b/src/common/src/grpc/gpt_answer.rs new file mode 100644 index 0000000..9bdbe8f --- /dev/null +++ b/src/common/src/grpc/gpt_answer.rs @@ -0,0 +1,9 @@ +/// Module for gRPC service definitions related to answering questions with GPT (Generative Pre-trained Transformer) models. +/// +/// This module includes generated gRPC service definitions for answering questions using GPT models. +/// The `tonic::include_proto!` macro is used to include the protobuf definitions, enabling easy +/// integration of gRPC services into Rust code. +pub mod gpt_answer { + // Include the protobuf definitions for the gpt_answer service. + tonic::include_proto!("gpt_answer"); +} diff --git a/src/grpc/src/interfaces/mod.rs b/src/common/src/grpc/mod.rs similarity index 100% rename from src/grpc/src/interfaces/mod.rs rename to src/common/src/grpc/mod.rs diff --git a/src/common/src/lib.rs b/src/common/src/lib.rs index c9499bd..4598395 100644 --- a/src/common/src/lib.rs +++ b/src/common/src/lib.rs @@ -1,2 +1,3 @@ +pub mod grpc; pub mod loggers; pub mod options; diff --git a/src/core/Cargo.toml b/src/core/Cargo.toml index 189d377..5dbd443 100644 --- a/src/core/Cargo.toml +++ b/src/core/Cargo.toml @@ -23,4 +23,4 @@ features = ["derive"] version = "1.0.57" [dependencies] -anyhow = "1.0.80" \ No newline at end of file +anyhow = "1.0.80" diff --git a/src/core/src/common/errors.rs b/src/core/src/common/errors.rs index 0f2d689..c2d1168 100644 --- a/src/core/src/common/errors.rs +++ b/src/core/src/common/errors.rs @@ -1,3 +1,5 @@ +use anyhow::Error; + #[derive(thiserror::Error, Debug)] pub enum CoreError { #[error("parse error {0}")] @@ -8,10 +10,13 @@ pub enum CoreError { #[error("missing parameters")] MissingParameters, + #[error("not found")] NotFound, - #[error("transparent")] - InternalError, + + #[error("internal error {0}")] + InternalError(#[from] Error), + #[error("unknown data store error")] Unknown, } diff --git a/src/core/src/entities/pagination_entity.rs b/src/core/src/entities/pagination_entity.rs index a2d38c9..944eb5e 100644 --- a/src/core/src/entities/pagination_entity.rs +++ b/src/core/src/entities/pagination_entity.rs @@ -15,37 +15,15 @@ pub struct PaginationEntity { pub sort: Option>, } -impl PaginationEntity { - /// Constructs a `PaginationEntity` from query parameters. - /// - /// # Arguments - /// - /// * `query` - A HashMap containing query parameters. - /// - /// # Returns - /// - /// A Result containing the constructed `PaginationEntity` or an `Error` if parsing fails. - /// - /// # Example - /// - /// ```rust - /// use std::collections::HashMap; - /// use rust_core::entities::pagination_entity::PaginationEntity; - /// - /// let mut query_params = HashMap::new(); - /// query_params.insert("start".to_string(), "0".to_string()); - /// query_params.insert("end".to_string(), "10".to_string()); - /// - /// match PaginationEntity::from_query(&query_params) { - /// Ok(pagination_entity) => { - /// println!("Parsed pagination entity: {:?}", pagination_entity); - /// } - /// Err(err) => { - /// eprintln!("Failed to parse pagination entity: {:?}", err); - /// } - /// } - /// ``` - pub fn from_query(query: &HashMap) -> Result { +/// Implementation of the `TryFrom` trait to convert a HashMap into a `PaginationEntity`. +/// +/// This implementation allows converting a HashMap containing query parameters into a `PaginationEntity`. +/// It attempts to parse the start and end pagination parameters from the HashMap and constructs a `PaginationEntity` +/// instance. If parsing fails for any reason, it returns a `CoreError`. +impl TryFrom> for PaginationEntity { + type Error = CoreError; + + fn try_from(query: HashMap) -> Result { let start = query .get("start") .unwrap_or(&"0".to_string()) @@ -73,7 +51,7 @@ mod tests { query_params_1.insert("start".to_string(), "0".to_string()); query_params_1.insert("end".to_string(), "10".to_string()); - match PaginationEntity::from_query(&query_params_1) { + match PaginationEntity::try_from(query_params_1) { Ok(pagination_entity) => { assert_eq!(pagination_entity.start, 0); assert_eq!(pagination_entity.end, 10); @@ -91,7 +69,7 @@ mod tests { let mut query_params_2 = HashMap::new(); query_params_2.insert("start".to_string(), "abs".to_string()); query_params_2.insert("end".to_string(), "10".to_string()); - match PaginationEntity::from_query(&query_params_2) { + match PaginationEntity::try_from(query_params_2) { Ok(_) => { panic!("Expected an error, but got Ok"); } diff --git a/src/core/src/entities/question_filter.rs b/src/core/src/entities/question_filter.rs index 0e43dc6..53f3054 100644 --- a/src/core/src/entities/question_filter.rs +++ b/src/core/src/entities/question_filter.rs @@ -1,4 +1,5 @@ use std::collections::HashMap; +use std::convert::TryFrom; use serde::{Deserialize, Serialize}; @@ -13,39 +14,17 @@ pub struct QuestionFilter { pub pagination: PaginationEntity, } -impl QuestionFilter { - /// Constructs a `QuestionFilter` from query parameters. - /// - /// # Arguments - /// - /// * `query` - A HashMap containing query parameters. - /// - /// # Returns - /// - /// A Result containing the constructed `QuestionFilter` or an `Error` if parsing fails. - /// - /// # Example - /// - /// ```rust - /// use std::collections::HashMap; - /// use rust_core::entities::question_filter::QuestionFilter; - /// - /// let mut query_params = HashMap::new(); - /// query_params.insert("start".to_string(), "0".to_string()); - /// query_params.insert("end".to_string(), "10".to_string()); - /// - /// match QuestionFilter::from_query(&query_params) { - /// Ok(question_filter) => { - /// println!("Parsed question filter: {:?}", question_filter); - /// } - /// Err(err) => { - /// eprintln!("Failed to parse question filter: {:?}", err); - /// } - /// } - /// ``` - pub fn from_query(query: &HashMap) -> Result { +/// Implementation of the `TryFrom` trait to convert a HashMap into a `QuestionFilter`. +/// +/// This implementation allows converting a HashMap containing query parameters into a `QuestionFilter`. +/// It attempts to parse the pagination parameters from the HashMap and constructs a `QuestionFilter` +/// instance. If parsing fails for any reason, it returns a `CoreError`. +impl TryFrom> for QuestionFilter { + type Error = CoreError; + + fn try_from(query: HashMap) -> Result { Ok(QuestionFilter { - pagination: PaginationEntity::from_query(query)?, + pagination: PaginationEntity::try_from(query)?, }) } } @@ -63,7 +42,7 @@ mod tests { query_params_1.insert("start".to_string(), "0".to_string()); query_params_1.insert("end".to_string(), "10".to_string()); - match QuestionFilter::from_query(&query_params_1) { + match QuestionFilter::try_from(query_params_1) { Ok(question_filter) => { assert_eq!(question_filter.pagination.start, 0); assert_eq!(question_filter.pagination.end, 10); @@ -81,14 +60,14 @@ mod tests { query_params_2.insert("start".to_string(), "asd".to_string()); query_params_2.insert("end".to_string(), "10".to_string()); - match QuestionFilter::from_query(&query_params_2) { + match QuestionFilter::try_from(query_params_2) { Ok(_) => { panic!("Expected an error, but got Ok"); } Err(err) => match err { CoreError::ParseError(_) => {} _ => { - panic!("Expected MissingParameters error, but got {:?}", err); + panic!("Expected ParseError error, but got {:?}", err); } }, } diff --git a/src/grpc_server/Cargo.toml b/src/gpt_answer_service/Cargo.toml similarity index 90% rename from src/grpc_server/Cargo.toml rename to src/gpt_answer_service/Cargo.toml index d6a048b..44b71e0 100644 --- a/src/grpc_server/Cargo.toml +++ b/src/gpt_answer_service/Cargo.toml @@ -4,7 +4,7 @@ test = [] example = [] [package] -name = "grpc_server" +name = "gpt_answer_service" edition = "2021" version = "0.0.1" autobins = true @@ -44,16 +44,12 @@ tracing = "0.1" [dependencies.common] path = "../common" -[dependencies.grpc_interface] -path = "../grpc" - [build-dependencies] tonic-build = "0.11.0" - [lib] path = "src/lib.rs" -name = "grpc_server" +name = "gpt_answer_service" test = true doctest = true bench = true diff --git a/src/grpc_server/build.rs b/src/gpt_answer_service/build.rs similarity index 100% rename from src/grpc_server/build.rs rename to src/gpt_answer_service/build.rs diff --git a/src/grpc_server/config/00-default.toml b/src/gpt_answer_service/config/00-default.toml similarity index 60% rename from src/grpc_server/config/00-default.toml rename to src/gpt_answer_service/config/00-default.toml index 1f4048f..baf3a2b 100644 --- a/src/grpc_server/config/00-default.toml +++ b/src/gpt_answer_service/config/00-default.toml @@ -1,5 +1,3 @@ service_name = "rust-grpc-server" exporter_endpoint = "http://localhost:7281" - -[servers.gpt_answer_service] -url = "0.0.0.0:50051" \ No newline at end of file +server_endpoint = "0.0.0.0:50051" \ No newline at end of file diff --git a/src/gpt_answer_service/src/controllers/gpt_answer.rs b/src/gpt_answer_service/src/controllers/gpt_answer.rs new file mode 100644 index 0000000..86f96e7 --- /dev/null +++ b/src/gpt_answer_service/src/controllers/gpt_answer.rs @@ -0,0 +1,49 @@ +use tonic::{Request, Response, Status}; + +use common::grpc::gpt_answer::gpt_answer::{ + gpt_answer_service_server::GptAnswerService, GetAnswerPayload, GetAnswerResponse, +}; + +/// Implementation of the gRPC service for generating answers to questions. +/// +/// This struct represents the gRPC server implementation for answering questions. It implements +/// the `GptAnswerService` trait generated by Tonic, which defines the RPC methods for answering +/// questions. +#[derive(Debug, Default)] +pub struct GptAnswerServer; + +#[tonic::async_trait] +impl GptAnswerService for GptAnswerServer { + /// Handle the gRPC `get_answer` request. + /// + /// This method is called when a gRPC client sends a request to get an answer to a question. + /// It receives a request containing the question payload and generates an answer based on + /// the received question. The logic for generating the answer is yet to be implemented. + /// + /// # Arguments + /// + /// * `request`: A `Request` containing the `GetAnswerPayload` with the question. + /// + /// # Returns + /// + /// Returns a `Result` containing a `Response` with the `GetAnswerResponse` containing the + /// generated answer if successful. If there's an error during processing, it returns a + /// `Status` indicating the error. + async fn get_answer( + &self, + request: Request, + ) -> Result, Status> { + // Extract the payload containing the question from the request + let payload = request.into_inner(); + + // TODO: Implement your logic to generate an answer based on the question. + // Placeholder logic: Generate an answer string + let answer = format!("Answer to: {}", payload.question); + + // Construct a response containing the generated answer + let response = GetAnswerResponse { answer }; + + // Return the response + Ok(Response::new(response)) + } +} diff --git a/src/grpc_server/src/controllers/mod.rs b/src/gpt_answer_service/src/controllers/mod.rs similarity index 100% rename from src/grpc_server/src/controllers/mod.rs rename to src/gpt_answer_service/src/controllers/mod.rs diff --git a/src/grpc_server/src/lib.rs b/src/gpt_answer_service/src/lib.rs similarity index 100% rename from src/grpc_server/src/lib.rs rename to src/gpt_answer_service/src/lib.rs diff --git a/src/grpc_server/src/main.rs b/src/gpt_answer_service/src/main.rs similarity index 64% rename from src/grpc_server/src/main.rs rename to src/gpt_answer_service/src/main.rs index 201752b..23a6acb 100644 --- a/src/grpc_server/src/main.rs +++ b/src/gpt_answer_service/src/main.rs @@ -1,11 +1,26 @@ use clap::{Parser, Subcommand}; -use grpc_server::{controllers, options}; use opentelemetry::global; +use tonic::transport::Server; +use common::grpc::gpt_answer::gpt_answer::gpt_answer_service_server::GptAnswerServiceServer; use common::loggers::telemetry::init_telemetry; use common::options::parse_options; -use controllers::gpt_answer::init_gpt_answer_server; -use options::Options; +use gpt_answer_service::controllers::gpt_answer::GptAnswerServer; +use gpt_answer_service::options::Options; + +pub async fn init_grpc_server(options: Options) { + let server_endpoint = options.server_endpoint.clone(); + let gpt_answer_server = GptAnswerServer::default(); + let address = server_endpoint.parse().unwrap(); + + Server::builder() + .add_service(GptAnswerServiceServer::new(gpt_answer_server)) + .serve(address) + .await + .unwrap(); + + println!("GPT Answer server started at {}", server_endpoint); +} #[tokio::main] async fn main() { @@ -34,7 +49,7 @@ async fn main() { options.log.level.as_str(), ); - let gpt_answer_server = tokio::spawn(init_gpt_answer_server(options)); + let gpt_answer_server = tokio::spawn(init_grpc_server(options)); tokio::try_join!(gpt_answer_server).expect("Failed to run servers"); diff --git a/src/grpc_server/src/options.rs b/src/gpt_answer_service/src/options.rs similarity index 50% rename from src/grpc_server/src/options.rs rename to src/gpt_answer_service/src/options.rs index f660c2c..f32eb2b 100644 --- a/src/grpc_server/src/options.rs +++ b/src/gpt_answer_service/src/options.rs @@ -1,15 +1,16 @@ -use common::options::{default_log, Log}; use serde::Deserialize; +use common::options::{default_log, Log}; + /// Configuration options for the application. /// /// This struct represents the configuration options for the application, including server settings, -/// database configuration, endpoint for the exporter, service name, and logging configuration. +/// endpoint for server, endpoint for the exporter, service name, and logging configuration. #[readonly::make] #[derive(Deserialize, Debug)] pub struct Options { - /// Configuration for the servers. - pub servers: GrpcServers, + /// Configuration for the grpc server endpoint. + pub server_endpoint: String, /// The endpoint for the exporter. pub exporter_endpoint: String, /// The name of the service. @@ -18,17 +19,3 @@ pub struct Options { #[serde(default = "default_log")] pub log: Log, } - -/// Represents servers configuration options. -#[derive(Deserialize, Debug)] -pub struct GrpcServers { - /// Configuration for using in-memory database. - pub gpt_answer_service: Option, -} - -/// Represents service server configuration. -#[derive(Debug, Deserialize, Clone)] -pub struct ServiceServer { - /// URL for the server. - pub url: String, -} diff --git a/src/grpc/Cargo.toml b/src/grpc/Cargo.toml deleted file mode 100644 index 652ee3f..0000000 --- a/src/grpc/Cargo.toml +++ /dev/null @@ -1,38 +0,0 @@ -bin = [] -bench = [] -test = [] -example = [] - -[package] -name = "grpc_interface" -edition = "2021" -version = "0.0.1" -autobins = true -autoexamples = true -autotests = true -autobenches = true - -[dependencies.tonic] -version = "0.11.0" - -[dependencies.prost] -version = "0.12.3" - -[build-dependencies] -tonic-build = "0.11.0" -glob = "0.3.1" - - -[lib] -path = "src/lib.rs" -name = "grpc_interface" -test = true -doctest = true -bench = true -doc = true -plugin = false -proc-macro = false -harness = true -edition = "2021" -required-features = [] -crate-type = ["rlib"] diff --git a/src/grpc/src/interfaces/gpt_answer.rs b/src/grpc/src/interfaces/gpt_answer.rs deleted file mode 100644 index ef738af..0000000 --- a/src/grpc/src/interfaces/gpt_answer.rs +++ /dev/null @@ -1,3 +0,0 @@ -pub mod gpt_answer { - tonic::include_proto!("gpt_answer"); -} diff --git a/src/grpc/src/lib.rs b/src/grpc/src/lib.rs deleted file mode 100644 index 43b15ec..0000000 --- a/src/grpc/src/lib.rs +++ /dev/null @@ -1 +0,0 @@ -pub mod interfaces; diff --git a/src/grpc_server/src/controllers/gpt_answer.rs b/src/grpc_server/src/controllers/gpt_answer.rs deleted file mode 100644 index 7aae3d4..0000000 --- a/src/grpc_server/src/controllers/gpt_answer.rs +++ /dev/null @@ -1,52 +0,0 @@ -use grpc_interface::interfaces::gpt_answer::gpt_answer::gpt_answer_service_server::{ - GptAnswerService, GptAnswerServiceServer, -}; -use grpc_interface::interfaces::gpt_answer::gpt_answer::{GetAnswerPayload, GetAnswerResponse}; -use rust_core::common::errors::CoreError; -use tonic::{transport::Server, Request, Response, Status}; - -use crate::options::Options; - -#[derive(Debug, Default)] -pub struct GptAnswerServer; - -#[tonic::async_trait] -impl GptAnswerService for GptAnswerServer { - async fn get_answer( - &self, - request: Request, - ) -> Result, Status> { - let payload = request.into_inner(); - // TODO: Implement your logic to generate an answer based on the question. - let answer = format!("Answer to: {}", payload.question); - - let response = GetAnswerResponse { answer }; - Ok(Response::new(response)) - } -} - -pub async fn init_gpt_answer_server(options: Options) { - let gpt_answer_config = options.servers.gpt_answer_service.clone().unwrap(); - let result = gpt_answer_config.url.parse().map_err(|err| { - eprintln!("Error: {:?}", err); - CoreError::InternalError - }); - - if result.is_ok() { - let addr = result.unwrap(); - - println!("GPT Answer server config at {}", addr); - - let gpt_answer_server = GptAnswerServer::default(); - - Server::builder() - .add_service(GptAnswerServiceServer::new(gpt_answer_server)) - .serve(addr) - .await - .unwrap(); - - println!("GPT Answer server started at {}", addr); - } else { - eprintln!("GPT Answer server failed to start"); - } -} diff --git a/src/public/Cargo.toml b/src/public/Cargo.toml index 74b8a55..674b509 100644 --- a/src/public/Cargo.toml +++ b/src/public/Cargo.toml @@ -50,6 +50,15 @@ autoexamples = true autotests = true autobenches = true +[dependencies.common] +path = "../common" + +[dependencies.adapter] +path = "../adapter" + +[dependencies.rust_core] +path = "../core" + [dependencies] diesel_migrations = "2.1.0" rand = "0.8.4" @@ -58,16 +67,10 @@ serde_json = "1.0" tracing = "0.1" warp = "0.3.6" -[dependencies.adapter] -path = "../adapter" - [dependencies.clap] version = "4.4.7" features = ["derive"] -[dependencies.common] -path = "../common" - [dependencies.deadpool-diesel] version = "0.5.0" features = ["postgres", "serde"] @@ -82,9 +85,6 @@ version = "0.10.64" [dependencies.opentelemetry] version = "0.22.0" -[dependencies.rust_core] -path = "../core" - [dependencies.serde] version = "1.0" features = ["derive"] @@ -103,9 +103,6 @@ version = "1.0.57" [dependencies.anyhow] version = "1.0.80" -[dependencies.grpc_interface] -path = "../grpc" - [lib] path = "src/lib.rs" name = "cli" diff --git a/src/public/config/00-default.toml b/src/public/config/00-default.toml index 029a598..e3f02de 100644 --- a/src/public/config/00-default.toml +++ b/src/public/config/00-default.toml @@ -6,4 +6,7 @@ exporter_endpoint = "http://localhost:7281" [server] url = "0.0.0.0" -port = 8000 \ No newline at end of file +port = 8000 + +[grpc_client] +gpt_answer_service_url = "0.0.0.0:50051" diff --git a/src/public/src/controllers/question.rs b/src/public/src/controllers/question.rs index 6bb8797..4b6c275 100644 --- a/src/public/src/controllers/question.rs +++ b/src/public/src/controllers/question.rs @@ -2,18 +2,18 @@ use std::collections::HashMap; use std::str::FromStr; use std::sync::Arc; -use adapter::repositories::grpc::models::gpt_answer::GptAnswerGrpcClient; use tracing::instrument; use warp::http::StatusCode; use warp::reject::Rejection; use warp::Reply; +use adapter::repositories::grpc::models::gpt_answer::GptAnswerGrpcClient; use rust_core::entities::question::{QuestionEntity, QuestionId}; use rust_core::entities::question_filter::QuestionFilter; use rust_core::ports::question::QuestionPort; use crate::errors::WarpError; -use crate::options::GrpcClients; +use crate::options::GrpcClient; /// Handler for retrieving questions based on query parameters. /// @@ -23,10 +23,9 @@ use crate::options::GrpcClients; #[instrument(level = "info", skip(question_port))] pub async fn get_questions( question_port: Arc, - server_config: Arc, query: HashMap, ) -> Result { - let question_filter = QuestionFilter::from_query(&query).map_err(WarpError::from)?; + let question_filter = QuestionFilter::try_from(query).map_err(WarpError::from)?; let questions = question_port .list(&question_filter) @@ -44,7 +43,6 @@ pub async fn get_questions( #[instrument(level = "info", skip(question_port))] pub async fn get_question( question_port: Arc, - server_config: Arc, id: String, ) -> Result { let question_id = QuestionId::from_str(id.as_str()).map_err(WarpError::from)?; @@ -65,7 +63,6 @@ pub async fn get_question( #[instrument(level = "info", skip(question_port))] pub async fn add_question( question_port: Arc, - server_config: Arc, question: QuestionEntity, ) -> Result { question_port.add(question).await.map_err(WarpError::from)?; @@ -81,7 +78,6 @@ pub async fn add_question( #[instrument(level = "info", skip(question_port))] pub async fn delete_question( question_port: Arc, - server_config: Arc, id: String, ) -> Result { let question_id = QuestionId::from_str(id.as_str()).map_err(WarpError::from)?; @@ -103,12 +99,10 @@ pub async fn delete_question( #[instrument(level = "info", skip(question_port))] pub async fn update_question( question_port: Arc, - server_config: Arc, id: String, mut question: QuestionEntity, ) -> Result { - let question_id = QuestionId::from_str(id.as_str()).map_err(WarpError::from)?; - question.id = question_id; + question.id = QuestionId::from_str(id.as_str()).map_err(WarpError::from)?; question_port .update(question) @@ -121,39 +115,44 @@ pub async fn update_question( /// Controller for handling HTTP GET requests to fetch answers for a given question ID. /// /// This controller retrieves a question from the provided `QuestionPort` based on the -/// specified ID, calls the gRPC client (`GrpcClient`) to get an answer using the -/// question's text, and responds with the answer in a JSON format. +/// specified ID, calls the gRPC client (`GptAnswerGrpcClient`) to get an answer using the +/// question's content, and responds with the answer in a JSON format. /// /// # Arguments /// /// * `question_port`: A trait object implementing `QuestionPort` for interacting with questions. -/// * `gpt_client`: An instance of `GrpcClient` used for communication with the gRPC server. +/// * `gpt_answer_service_url`: The URL of the gRPC service for answering questions. /// * `id`: The ID of the question to fetch the answer for. /// /// # Returns /// -/// Returns a `Result` containing the HTTP response. If successful, responds with a String -/// representation of the answer and a status code of 200 OK. If there's an error during -/// question retrieval, gRPC communication, or response construction, it returns a Warp Rejection. +/// Returns a `Result` containing the HTTP response. If successful, responds with the answer +/// as a JSON string and a status code of `200 OK`. If there's an error during +/// question retrieval, gRPC communication, or response construction, it returns a Warp `Rejection`. #[instrument(level = "info", skip(question_port))] -pub async fn get_question_answer_controller( +pub async fn get_question_answer( question_port: Arc, - server_config: Arc, + gpt_answer_service_url: String, id: String, ) -> Result { + // Retrieve the question + let question_id = QuestionId::from_str(id.as_str()).map_err(WarpError::from)?; + let question = question_port - .get(&QuestionId::from_str(id.as_str()).unwrap()) + .get(&question_id) .await .map_err(WarpError::from)?; - // Clone the gpt_client string to have a static lifetime - let gpt_client = server_config.gpt_answer_service.clone().unwrap().url; - - // Use the cloned gpt_client string - let client = GptAnswerGrpcClient::get_instance(gpt_client.leak()).await; + // Establish connection to the gRPC server + let mut client = GptAnswerGrpcClient::connect(gpt_answer_service_url.leak()) + .await + .map_err(|err| { + tracing::error!("Error connecting to gRPC server: {:?}", err); + WarpError::from(err) + })?; + // Call gRPC service to get the answer let answer = client - .unwrap() .get_answer(&question.content) .await .map_err(WarpError::from)?; diff --git a/src/public/src/errors.rs b/src/public/src/errors.rs index 555164c..e92437f 100644 --- a/src/public/src/errors.rs +++ b/src/public/src/errors.rs @@ -1,4 +1,5 @@ use std::io; + use thiserror::Error; use warp::body::BodyDeserializeError; use warp::cors::CorsForbidden; @@ -34,7 +35,7 @@ pub async fn return_error(r: Rejection) -> Result { "MissingParameters".to_string(), StatusCode::BAD_REQUEST, )), - CoreError::InternalError => Ok(warp::reply::with_status( + CoreError::InternalError(_) => Ok(warp::reply::with_status( "InternalError".to_string(), StatusCode::INTERNAL_SERVER_ERROR, )), diff --git a/src/public/src/main.rs b/src/public/src/main.rs index a5c0302..e26783a 100644 --- a/src/public/src/main.rs +++ b/src/public/src/main.rs @@ -1,13 +1,12 @@ +use std::net::{Ipv4Addr, SocketAddrV4}; +use std::str::FromStr; +use std::sync::Arc; + #[cfg_attr(debug_assertions, allow(dead_code, unused_imports))] use openssl; #[rustfmt::skip] #[cfg_attr(debug_assertions, allow(dead_code, unused_imports))] use diesel; - -use std::net::{Ipv4Addr, SocketAddrV4}; -use std::str::FromStr; -use std::sync::Arc; - use clap::{Parser, Subcommand}; use deadpool_diesel::postgres::Pool; use deadpool_diesel::{Manager, Runtime}; @@ -93,8 +92,8 @@ pub async fn run_warp_server(options: Options) { Arc::new(QuestionInMemoryRepository::new()) }; - let grpc_clients = options.grpc_clients.clone(); - let router = Router::new(question_port, Arc::new(grpc_clients)); + let grpc_client = options.grpc_client.clone(); + let router = Router::new(question_port, Arc::new(grpc_client)); let address = SocketAddrV4::new( Ipv4Addr::from_str(options.server.url.as_str()).unwrap(), diff --git a/src/public/src/options.rs b/src/public/src/options.rs index ad1c30a..1636e20 100644 --- a/src/public/src/options.rs +++ b/src/public/src/options.rs @@ -12,9 +12,9 @@ use common::options::{default_log, Log}; pub struct Options { /// Configuration for the server. pub server: Server, - /// Specifies the backend database will be used. - pub grpc_clients: GrpcClients, - /// Specifies the backend database will be used. + /// Specifies the gRPC client configuration for communicating with the backend services. + pub grpc_client: GrpcClient, + /// Specifies the configuration of grpc clients will be connected. pub db: Database, /// The endpoint for the exporter. pub exporter_endpoint: String, @@ -47,15 +47,9 @@ pub struct Server { pub url: String, } -/// Represents server configuration. -#[derive(Debug, Deserialize, Clone)] -pub struct GrpcClients { - pub gpt_answer_service: Option, -} - -/// Represents service server configuration. +/// Represents grpc client configuration. #[derive(Debug, Deserialize, Clone)] -pub struct ServiceServer { - /// URL for the server. - pub url: String, +pub struct GrpcClient { + /// URL for the GPT Answer gRPC client. + pub gpt_answer_service_url: String, } diff --git a/src/public/src/router.rs b/src/public/src/router.rs index 98c33a6..04a6e16 100644 --- a/src/public/src/router.rs +++ b/src/public/src/router.rs @@ -1,28 +1,27 @@ use std::sync::Arc; - use warp::http::Method; use warp::{Filter, Rejection, Reply}; use rust_core::ports::question::QuestionPort; use crate::controllers::question::{ - add_question, delete_question, get_question, get_question_answer_controller, get_questions, + add_question, delete_question, get_question, get_question_answer, get_questions, update_question, }; use crate::errors::return_error; -use crate::options::GrpcClients; +use crate::options::GrpcClient; /// Router for handling HTTP requests related to questions. pub struct Router { question_port: Arc, - server_config: Arc, + server_config: Arc, } impl Router { /// Creates a new Router instance with the specified QuestionPort. pub fn new( question_port: Arc, - server_config: Arc, + server_config: Arc, ) -> Self { Router { question_port: question_port.clone(), @@ -33,23 +32,23 @@ impl Router { /// Configures and returns the Warp filter for handling HTTP requests. pub fn routes(self) -> impl Filter + Clone { let store_filter = warp::any().map(move || self.question_port.clone()); - let server_config = warp::any().map(move || self.server_config.clone()); + let server_config = self.server_config.to_owned(); + let cors = warp::cors() .allow_any_origin() .allow_header("content-type") .allow_methods(&[Method::PUT, Method::DELETE, Method::GET, Method::POST]); + let get_questions = warp::get() .and(warp::path("questions")) .and(warp::path::end()) .and(store_filter.clone()) - .and(server_config.clone()) .and(warp::query()) .and_then(get_questions); let get_question = warp::get() .and(warp::path("questions")) .and(store_filter.clone()) - .and(server_config.clone()) .and(warp::path::param::()) .and(warp::path::end()) .and_then(get_question); @@ -58,14 +57,12 @@ impl Router { .and(warp::path("questions")) .and(warp::path::end()) .and(store_filter.clone()) - .and(server_config.clone()) .and(warp::body::json()) .and_then(add_question); let update_question = warp::put() .and(warp::path("questions")) .and(store_filter.clone()) - .and(server_config.clone()) .and(warp::path::param::()) .and(warp::path::end()) .and(warp::body::json()) @@ -74,7 +71,6 @@ impl Router { let delete_question = warp::delete() .and(warp::path("questions")) .and(store_filter.clone()) - .and(server_config.clone()) .and(warp::path::param::()) .and(warp::path::end()) .and_then(delete_question); @@ -82,10 +78,10 @@ impl Router { let get_question_answer = warp::get() .and(warp::path("questions")) .and(store_filter.clone()) - .and(server_config.clone()) + .and(warp::any().map(move || server_config.gpt_answer_service_url.clone())) .and(warp::path::param::()) .and(warp::path("answer")) - .and_then(get_question_answer_controller); + .and_then(get_question_answer); get_questions .with(cors) diff --git a/src/public/tests/questions_router_test.rs b/src/public/tests/questions_router_test.rs index 0672db1..d4a4f34 100644 --- a/src/public/tests/questions_router_test.rs +++ b/src/public/tests/questions_router_test.rs @@ -15,10 +15,7 @@ mod tests { in_memory::question::QuestionInMemoryRepository, postgres::question_db::{QuestionDBRepository, MIGRATIONS}, }; - use cli::{ - options::{GrpcClients, ServiceServer}, - router::Router, - }; + use cli::{options::GrpcClient, router::Router}; use rust_core::{ entities::question::{QuestionEntity, QuestionId}, ports::question::QuestionPort, @@ -33,10 +30,8 @@ mod tests { where T: QuestionPort + Send + Sync + 'static, { - let server_config: GrpcClients = GrpcClients { - gpt_answer_service: Some(ServiceServer { - url: "http://localhost:50051".to_string(), - }), + let server_config: GrpcClient = GrpcClient { + gpt_answer_service_url: "http://localhost:50051".to_string(), }; let router = Router::new(question_port, Arc::new(server_config));