summaryrefslogtreecommitdiff
path: root/src/response.rs
diff options
context:
space:
mode:
authorTyler Murphy <tylerm@tylerm.dev>2023-07-05 21:34:20 -0400
committerTyler Murphy <tylerm@tylerm.dev>2023-07-05 21:34:20 -0400
commit168b8937eb0fe88311fe474ab9569691a19d087f (patch)
tree3e70d7de89adc9e62987cea9341ba2cc10d6cf25 /src/response.rs
downloadhttp-168b8937eb0fe88311fe474ab9569691a19d087f.tar.gz
http-168b8937eb0fe88311fe474ab9569691a19d087f.tar.bz2
http-168b8937eb0fe88311fe474ab9569691a19d087f.zip
changes
Diffstat (limited to 'src/response.rs')
-rw-r--r--src/response.rs111
1 files changed, 111 insertions, 0 deletions
diff --git a/src/response.rs b/src/response.rs
new file mode 100644
index 0000000..7f6f326
--- /dev/null
+++ b/src/response.rs
@@ -0,0 +1,111 @@
+use std::str::FromStr;
+use crate::{status::{Status, self}, version::Version, header::{HeaderMap, Header}, parse::TryParse, error::HTTPError};
+
+pub struct Response<T: ToString + FromStr> {
+ version: Version,
+ status: Status,
+ headers: HeaderMap,
+ body: Option<T>
+}
+
+impl<T: ToString + FromStr> Response<T> {
+ pub fn new() -> Self {
+ Self {
+ version: Version::HTTP11,
+ status: status::SUCCESS,
+ headers: HeaderMap::new(),
+ body: None
+ }
+ }
+
+ pub fn set_version(mut self, version: impl Into<Version>) -> Self {
+ self.version = version.into();
+ self
+ }
+
+ pub fn set_status(mut self, status: impl Into<Status>) -> Self {
+ self.status = status.into();
+ self
+ }
+
+ pub fn add_header(mut self, header: impl Into<Header>) -> Self {
+ self.headers.insert(header.into());
+ self
+ }
+
+ pub fn set_body(mut self, body: T) -> Self {
+ self.body = Some(body);
+ self
+ }
+}
+
+impl<T: ToString + FromStr> TryParse for Response<T> {
+ fn try_parse(s: impl Into<String>) -> Result<Self, HTTPError> {
+ let s = s.into();
+ let mut lines = s.lines();
+
+ let Some(header_str) = lines.next() else {
+ return Err(HTTPError::MissingHeader)
+ };
+
+ let mut header_parts = header_str.split(" ");
+ let Some(version_str) = header_parts.next() else {
+ return Err(HTTPError::InvalidHeader(header_str.into()))
+ };
+ let version = version_str.into();
+
+ let status_str: String = header_parts.collect();
+ let status = status_str.try_into()?;
+
+ let mut headers = HeaderMap::new();
+
+ loop {
+ let Some(next) = lines.next() else {
+ return Err(HTTPError::MissingHeaderBreak)
+ };
+
+ if next.len() < 1 { break }
+
+ let header = Header::try_parse(next)?;
+ headers.insert(header);
+ }
+
+ let rest: String = lines.collect();
+ let body;
+ if rest.len() < 1 {
+ body = None
+ } else {
+ body = match T::from_str(&rest) {
+ Ok(body) => Some(body),
+ Err(_) => return Err(HTTPError::InvalidBody)
+ }
+ }
+
+ Ok(Self { version, status, headers, body })
+ }
+}
+
+impl<T: ToString + FromStr> ToString for Response<T> {
+ fn to_string(&self) -> String {
+ let mut s = String::new();
+ s.push_str(self.version.as_str());
+ s.push(' ');
+ s.push_str(&self.status.to_string());
+ s.push_str("\r\n");
+ let iter = self.headers.iter();
+ for headers in iter {
+ for header in headers {
+ s.push_str(header.name.as_str());
+ s.push_str(": ");
+ s.push_str(header.value.as_str());
+ s.push_str("\r\n");
+ }
+ }
+ s.push_str("\r\n");
+ if let Some(body) = &self.body {
+ s.push_str(&body.to_string());
+ }
+ s
+ }
+}
+