style(rust): Format all Rust code with rustfmt
Change-Id: Iab7e00cc26a4f9727d3ab98691ef379921a33052 Reviewed-on: https://cl.tvl.fyi/c/depot/+/5240 Tested-by: BuildkiteCI Reviewed-by: kanepyork <rikingcoding@gmail.com> Reviewed-by: Profpatsch <mail@profpatsch.de> Reviewed-by: grfn <grfn@gws.fyi> Reviewed-by: tazjin <tazjin@tvl.su>
This commit is contained in:
		
							parent
							
								
									3318982f81
								
							
						
					
					
						commit
						3d8ee62087
					
				
					 42 changed files with 1253 additions and 876 deletions
				
			
		|  | @ -4,7 +4,7 @@ use std::rc::Rc; | ||||||
| use std::sync::RwLock; | use std::sync::RwLock; | ||||||
| 
 | 
 | ||||||
| struct Defer<F: Fn()> { | struct Defer<F: Fn()> { | ||||||
|     f: F |     f: F, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl<F: Fn()> Drop for Defer<F> { | impl<F: Fn()> Drop for Defer<F> { | ||||||
|  | @ -29,7 +29,9 @@ type ErrorHandle<T> = Rc<RwLock<Option<T>>>; | ||||||
| ///////////////////
 | ///////////////////
 | ||||||
| 
 | 
 | ||||||
| #[derive(Debug)] // Debug trait for some default way to print the type.
 | #[derive(Debug)] // Debug trait for some default way to print the type.
 | ||||||
| enum Error { DropError } | enum Error { | ||||||
|  |     DropError, | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| fn main() { | fn main() { | ||||||
|     // Create a place to store the error.
 |     // Create a place to store the error.
 | ||||||
|  | @ -60,7 +62,7 @@ fn main() { | ||||||
| 
 | 
 | ||||||
|     match *drop_err.read().unwrap() { |     match *drop_err.read().unwrap() { | ||||||
|         Some(ref err) => println!("Oh no, an error occured: {:?}!", err), |         Some(ref err) => println!("Oh no, an error occured: {:?}!", err), | ||||||
|         None => println!("Phew, everything went well.") |         None => println!("Phew, everything went well."), | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,7 +1,7 @@ | ||||||
| // Go's defer in Rust!
 | // Go's defer in Rust!
 | ||||||
| 
 | 
 | ||||||
| struct Defer<F: Fn()> { | struct Defer<F: Fn()> { | ||||||
|     f: F |     f: F, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl<F: Fn()> Drop for Defer<F> { | impl<F: Fn()> Drop for Defer<F> { | ||||||
|  |  | ||||||
|  | @ -1,7 +1,7 @@ | ||||||
| // Go's defer in Rust, with a little twist!
 | // Go's defer in Rust, with a little twist!
 | ||||||
| 
 | 
 | ||||||
| struct Defer<F: Fn()> { | struct Defer<F: Fn()> { | ||||||
|     f: F |     f: F, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl<F: Fn()> Drop for Defer<F> { | impl<F: Fn()> Drop for Defer<F> { | ||||||
|  |  | ||||||
|  | @ -1,8 +1,7 @@ | ||||||
| use crate::models::{Entry, Keyword, NewEntry, NewKeyword}; | use crate::models::{Entry, Keyword, NewEntry, NewKeyword}; | ||||||
| use diesel::pg::PgConnection; | use diesel::pg::PgConnection; | ||||||
| use diesel::prelude::*; | use diesel::prelude::*; | ||||||
| use failure::format_err; | use failure::{format_err, Error}; | ||||||
| use failure::Error; |  | ||||||
| use std::borrow::Cow; | use std::borrow::Cow; | ||||||
| 
 | 
 | ||||||
| /// Maximum number of times we'll follow a `see: ` pointer.
 | /// Maximum number of times we'll follow a `see: ` pointer.
 | ||||||
|  |  | ||||||
|  | @ -7,8 +7,7 @@ use crate::cfg::Config; | ||||||
| use crate::keyword::KeywordDetails; | use crate::keyword::KeywordDetails; | ||||||
| use diesel::pg::PgConnection; | use diesel::pg::PgConnection; | ||||||
| use diesel::r2d2::{ConnectionManager, Pool}; | use diesel::r2d2::{ConnectionManager, Pool}; | ||||||
| use failure::format_err; | use failure::{format_err, Error}; | ||||||
| use failure::Error; |  | ||||||
| use irc::client::prelude::*; | use irc::client::prelude::*; | ||||||
| use lazy_static::lazy_static; | use lazy_static::lazy_static; | ||||||
| use log::{debug, info, warn}; | use log::{debug, info, warn}; | ||||||
|  | @ -153,8 +152,9 @@ impl App { | ||||||
|         // Use `nick` here, so things like "grfn: see glittershark" work.
 |         // Use `nick` here, so things like "grfn: see glittershark" work.
 | ||||||
|         let val = if let Some(last) = chan_lastmsgs.get(nick_to_grab) { |         let val = if let Some(last) = chan_lastmsgs.get(nick_to_grab) { | ||||||
|             if last.starts_with("\x01ACTION ") { |             if last.starts_with("\x01ACTION ") { | ||||||
|                 // Yes, this is inefficient, but it's better than writing some hacky CTCP parsing code
 |                 // Yes, this is inefficient, but it's better than writing some hacky CTCP parsing
 | ||||||
|                 // I guess (also, characters are hard, so just blindly slicing seems like a bad idea)
 |                 // code I guess (also, characters are hard, so just blindly slicing
 | ||||||
|  |                 // seems like a bad idea)
 | ||||||
|                 format!( |                 format!( | ||||||
|                     "* {} {}", |                     "* {} {}", | ||||||
|                     nick_to_grab, |                     nick_to_grab, | ||||||
|  |  | ||||||
|  | @ -67,23 +67,24 @@ | ||||||
| //!
 | //!
 | ||||||
| //! [JWKS]: https://tools.ietf.org/html/rfc7517
 | //! [JWKS]: https://tools.ietf.org/html/rfc7517
 | ||||||
| 
 | 
 | ||||||
| #[macro_use] extern crate serde_derive; | #[macro_use] | ||||||
|  | extern crate serde_derive; | ||||||
| 
 | 
 | ||||||
| extern crate base64; | extern crate base64; | ||||||
| extern crate openssl; | extern crate openssl; | ||||||
| extern crate serde; | extern crate serde; | ||||||
| extern crate serde_json; | extern crate serde_json; | ||||||
| 
 | 
 | ||||||
| use base64::{URL_SAFE_NO_PAD, Config, DecodeError}; | use base64::{Config, DecodeError, URL_SAFE_NO_PAD}; | ||||||
| use openssl::bn::BigNum; | use openssl::bn::BigNum; | ||||||
| use openssl::error::ErrorStack; | use openssl::error::ErrorStack; | ||||||
| use openssl::hash::MessageDigest; | use openssl::hash::MessageDigest; | ||||||
| use openssl::pkey::{Public, PKey}; | use openssl::pkey::{PKey, Public}; | ||||||
| use openssl::rsa::Rsa; | use openssl::rsa::Rsa; | ||||||
| use openssl::sign::Verifier; | use openssl::sign::Verifier; | ||||||
| use serde::de::DeserializeOwned; | use serde::de::DeserializeOwned; | ||||||
| use serde_json::Value; | use serde_json::Value; | ||||||
| use std::time::{UNIX_EPOCH, Duration, SystemTime}; | use std::time::{Duration, SystemTime, UNIX_EPOCH}; | ||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests; | mod tests; | ||||||
|  | @ -101,12 +102,16 @@ fn jwt_forgiving() -> Config { | ||||||
| /// JWT algorithm used. The only supported algorithm is currently
 | /// JWT algorithm used. The only supported algorithm is currently
 | ||||||
| /// RS256.
 | /// RS256.
 | ||||||
| #[derive(Clone, Deserialize, Debug)] | #[derive(Clone, Deserialize, Debug)] | ||||||
| enum KeyAlgorithm { RS256 } | enum KeyAlgorithm { | ||||||
|  |     RS256, | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| /// Type of key contained in a JWT. The only supported key type is
 | /// Type of key contained in a JWT. The only supported key type is
 | ||||||
| /// currently RSA.
 | /// currently RSA.
 | ||||||
| #[derive(Clone, Deserialize, Debug)] | #[derive(Clone, Deserialize, Debug)] | ||||||
| enum KeyType { RSA } | enum KeyType { | ||||||
|  |     RSA, | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| /// Representation of a single JSON Web Key. See [RFC
 | /// Representation of a single JSON Web Key. See [RFC
 | ||||||
| /// 7517](https://tools.ietf.org/html/rfc7517#section-4).
 | /// 7517](https://tools.ietf.org/html/rfc7517#section-4).
 | ||||||
|  | @ -217,15 +222,21 @@ pub enum ValidationError { | ||||||
| type JWTResult<T> = Result<T, ValidationError>; | type JWTResult<T> = Result<T, ValidationError>; | ||||||
| 
 | 
 | ||||||
| impl From<ErrorStack> for ValidationError { | impl From<ErrorStack> for ValidationError { | ||||||
|     fn from(err: ErrorStack) -> Self { ValidationError::OpenSSL(err) } |     fn from(err: ErrorStack) -> Self { | ||||||
|  |         ValidationError::OpenSSL(err) | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl From<serde_json::Error> for ValidationError { | impl From<serde_json::Error> for ValidationError { | ||||||
|     fn from(err: serde_json::Error) -> Self { ValidationError::JSON(err) } |     fn from(err: serde_json::Error) -> Self { | ||||||
|  |         ValidationError::JSON(err) | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl From<DecodeError> for ValidationError { | impl From<DecodeError> for ValidationError { | ||||||
|     fn from(err: DecodeError) -> Self { ValidationError::InvalidBase64(err) } |     fn from(err: DecodeError) -> Self { | ||||||
|  |         ValidationError::InvalidBase64(err) | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Attempt to extract the `kid`-claim out of a JWT's header claims.
 | /// Attempt to extract the `kid`-claim out of a JWT's header claims.
 | ||||||
|  | @ -266,9 +277,7 @@ pub fn token_kid(token: &str) -> JWTResult<Option<String>> { | ||||||
| ///
 | ///
 | ||||||
| /// It is the user's task to ensure that the correct JWK is passed in
 | /// It is the user's task to ensure that the correct JWK is passed in
 | ||||||
| /// for validation.
 | /// for validation.
 | ||||||
| pub fn validate(token: &str, | pub fn validate(token: &str, jwk: &JWK, validations: Vec<Validation>) -> JWTResult<ValidJWT> { | ||||||
|                 jwk: &JWK, |  | ||||||
|                 validations: Vec<Validation>) -> JWTResult<ValidJWT> { |  | ||||||
|     let jwt = JWT(token); |     let jwt = JWT(token); | ||||||
|     let public_key = public_key_from_jwk(&jwk)?; |     let public_key = public_key_from_jwk(&jwk)?; | ||||||
|     validate_jwt_signature(&jwt, public_key)?; |     validate_jwt_signature(&jwt, public_key)?; | ||||||
|  | @ -279,7 +288,7 @@ pub fn validate(token: &str, | ||||||
|     if parts.len() != 3 { |     if parts.len() != 3 { | ||||||
|         // This is unlikely considering that validation has already
 |         // This is unlikely considering that validation has already
 | ||||||
|         // been performed at this point, but better safe than sorry.
 |         // been performed at this point, but better safe than sorry.
 | ||||||
|         return Err(ValidationError::InvalidComponents) |         return Err(ValidationError::InvalidComponents); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // Perform claim validations before constructing the valid token:
 |     // Perform claim validations before constructing the valid token:
 | ||||||
|  | @ -362,7 +371,7 @@ fn validate_jwt_signature(jwt: &JWT, key: Rsa<Public>) -> JWTResult<()> { | ||||||
| #[serde(untagged)] | #[serde(untagged)] | ||||||
| enum Audience { | enum Audience { | ||||||
|     Single(String), |     Single(String), | ||||||
|     Multi(Vec<String>) |     Multi(Vec<String>), | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Internal helper struct for claims that are relevant for claim
 | /// Internal helper struct for claims that are relevant for claim
 | ||||||
|  | @ -376,15 +385,14 @@ struct PartialClaims { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Apply a single validation to the claim set of a token.
 | /// Apply a single validation to the claim set of a token.
 | ||||||
| fn apply_validation(claims: &PartialClaims, | fn apply_validation(claims: &PartialClaims, validation: Validation) -> Result<(), &'static str> { | ||||||
|                     validation: Validation) -> Result<(), &'static str> { |  | ||||||
|     match validation { |     match validation { | ||||||
|         // Validate that an 'iss' claim is present and matches the
 |         // Validate that an 'iss' claim is present and matches the
 | ||||||
|         // supplied value.
 |         // supplied value.
 | ||||||
|         Validation::Issuer(iss) => { |         Validation::Issuer(iss) => match claims.iss { | ||||||
|             match claims.iss { |  | ||||||
|             None => Err("'iss' claim is missing"), |             None => Err("'iss' claim is missing"), | ||||||
|                 Some(ref claim) => if *claim == iss { |             Some(ref claim) => { | ||||||
|  |                 if *claim == iss { | ||||||
|                     Ok(()) |                     Ok(()) | ||||||
|                 } else { |                 } else { | ||||||
|                     Err("'iss' claim does not match") |                     Err("'iss' claim does not match") | ||||||
|  | @ -394,15 +402,17 @@ fn apply_validation(claims: &PartialClaims, | ||||||
| 
 | 
 | ||||||
|         // Validate that an 'aud' claim is present and matches the
 |         // Validate that an 'aud' claim is present and matches the
 | ||||||
|         // supplied value.
 |         // supplied value.
 | ||||||
|         Validation::Audience(aud) => { |         Validation::Audience(aud) => match claims.aud { | ||||||
|             match claims.aud { |  | ||||||
|             None => Err("'aud' claim is missing"), |             None => Err("'aud' claim is missing"), | ||||||
|                 Some(Audience::Single(ref claim)) => if *claim == aud { |             Some(Audience::Single(ref claim)) => { | ||||||
|  |                 if *claim == aud { | ||||||
|                     Ok(()) |                     Ok(()) | ||||||
|                 } else { |                 } else { | ||||||
|                     Err("'aud' claim does not match") |                     Err("'aud' claim does not match") | ||||||
|                 }, |                 } | ||||||
|                 Some(Audience::Multi(ref claims)) => if claims.contains(&aud) { |             } | ||||||
|  |             Some(Audience::Multi(ref claims)) => { | ||||||
|  |                 if claims.contains(&aud) { | ||||||
|                     Ok(()) |                     Ok(()) | ||||||
|                 } else { |                 } else { | ||||||
|                     Err("'aud' claim does not match") |                     Err("'aud' claim does not match") | ||||||
|  | @ -447,9 +457,9 @@ fn apply_validation(claims: &PartialClaims, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Apply all requested validations to a partial claim set.
 | /// Apply all requested validations to a partial claim set.
 | ||||||
| fn validate_claims(claims: PartialClaims, | fn validate_claims(claims: PartialClaims, validations: Vec<Validation>) -> JWTResult<()> { | ||||||
|                    validations: Vec<Validation>) -> JWTResult<()> { |     let validation_errors: Vec<_> = validations | ||||||
|     let validation_errors: Vec<_> = validations.into_iter() |         .into_iter() | ||||||
|         .map(|v| apply_validation(&claims, v)) |         .map(|v| apply_validation(&claims, v)) | ||||||
|         .filter_map(|result| match result { |         .filter_map(|result| match result { | ||||||
|             Ok(_) => None, |             Ok(_) => None, | ||||||
|  |  | ||||||
|  | @ -21,14 +21,19 @@ fn test_fragment_decoding() { | ||||||
|     let bignum = decode_fragment(fragment).expect("Failed to decode fragment"); |     let bignum = decode_fragment(fragment).expect("Failed to decode fragment"); | ||||||
| 
 | 
 | ||||||
|     let expected = "19947781743618558124649689124245117083485690334420160711273532766920651190711502679542723943527557680293732686428091794139998732541701457212387600480039297092835433997837314251024513773285252960725418984381935183495143908023024822433135775773958512751261112853383693442999603704969543668619221464654540065497665889289271044207667765128672709218996183649696030570183970367596949687544839066873508106034650634722970893169823917299050098551447676778961773465887890052852528696684907153295689693676910831376066659456592813140662563597179711588277621736656871685099184755908108451080261403193680966083938080206832839445289"; |     let expected = "19947781743618558124649689124245117083485690334420160711273532766920651190711502679542723943527557680293732686428091794139998732541701457212387600480039297092835433997837314251024513773285252960725418984381935183495143908023024822433135775773958512751261112853383693442999603704969543668619221464654540065497665889289271044207667765128672709218996183649696030570183970367596949687544839066873508106034650634722970893169823917299050098551447676778961773465887890052852528696684907153295689693676910831376066659456592813140662563597179711588277621736656871685099184755908108451080261403193680966083938080206832839445289"; | ||||||
|     assert_eq!(expected, format!("{}", bignum), "Decoded fragment should match "); |     assert_eq!( | ||||||
|  |         expected, | ||||||
|  |         format!("{}", bignum), | ||||||
|  |         "Decoded fragment should match " | ||||||
|  |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | #[test] | ||||||
| fn test_decode_find_jwks() { | fn test_decode_find_jwks() { | ||||||
|     let json = "{\"keys\":[{\"kty\":\"RSA\",\"alg\":\"RS256\",\"use\":\"sig\",\"kid\":\"mUjI\\/rIMLLtung35BKZfdbrqtlEAAYJ4JX\\/SKvnLxJc=\",\"n\":\"ngRRjNbXgPW29oNtF0JgsyyfTwPyEL0u_X16s453X2AOc33XGFxVKLEQ7R_TiMenaKcr-tPifYqgps_deyi0XOr4I3SOdOMtAVKDZJCANe--CANOHZb-meIfjKhCHisvT90fm5Apd6qPRVsXsZ7A8pmClZHKM5fwZUkBv8NsPLm2Xy2sGOZIiwP_7z8m3j0abUzniPQsx2b3xcWimB9vRtshFHN1KgPUf1ALQ5xzLfJnlFkCxC7kmOxKC7_NpQ4kJR_DKzKFV_r3HxTqf-jddHcXIrrMcLQXCSyeLQtLaz7whQ4F-EfL42z4XgwPr4ji3sct2gWL13EqlbE5DDxLKQ\",\"e\":\"GK7oLCDbNPAF59LhvyseqcG04hDnPs58qGYolr_HHmaR4lulWJ90ozx6e4Ut363yKG2p9vwvivR5UIC-aLPtqT2qr-OtjhBFzUFVaMGZ6mPCvMKk0AgMYdOHvWTgBSqQtNJTvl1yYLnhcWyoE2fLQhoEbY9qUyCBCEOScXOZRDpnmBtz5I8q5yYMV6a920J24T_IYbxHgkGcEU2SGg-b1cOMD7Rja7vCfV---CQ2pR4leQ0jufzudDoe7z3mziJm-Ihcdrz2Ujy5kPEMdz6R55prJ-ENKrkD_X4u5aSlSRaetwmHS3oAVkjr1JwUNbqnpM-kOqieqHEp8LUmez-Znw\"}]}"; |     let json = "{\"keys\":[{\"kty\":\"RSA\",\"alg\":\"RS256\",\"use\":\"sig\",\"kid\":\"mUjI\\/rIMLLtung35BKZfdbrqtlEAAYJ4JX\\/SKvnLxJc=\",\"n\":\"ngRRjNbXgPW29oNtF0JgsyyfTwPyEL0u_X16s453X2AOc33XGFxVKLEQ7R_TiMenaKcr-tPifYqgps_deyi0XOr4I3SOdOMtAVKDZJCANe--CANOHZb-meIfjKhCHisvT90fm5Apd6qPRVsXsZ7A8pmClZHKM5fwZUkBv8NsPLm2Xy2sGOZIiwP_7z8m3j0abUzniPQsx2b3xcWimB9vRtshFHN1KgPUf1ALQ5xzLfJnlFkCxC7kmOxKC7_NpQ4kJR_DKzKFV_r3HxTqf-jddHcXIrrMcLQXCSyeLQtLaz7whQ4F-EfL42z4XgwPr4ji3sct2gWL13EqlbE5DDxLKQ\",\"e\":\"GK7oLCDbNPAF59LhvyseqcG04hDnPs58qGYolr_HHmaR4lulWJ90ozx6e4Ut363yKG2p9vwvivR5UIC-aLPtqT2qr-OtjhBFzUFVaMGZ6mPCvMKk0AgMYdOHvWTgBSqQtNJTvl1yYLnhcWyoE2fLQhoEbY9qUyCBCEOScXOZRDpnmBtz5I8q5yYMV6a920J24T_IYbxHgkGcEU2SGg-b1cOMD7Rja7vCfV---CQ2pR4leQ0jufzudDoe7z3mziJm-Ihcdrz2Ujy5kPEMdz6R55prJ-ENKrkD_X4u5aSlSRaetwmHS3oAVkjr1JwUNbqnpM-kOqieqHEp8LUmez-Znw\"}]}"; | ||||||
|     let jwks: JWKS = serde_json::from_str(json).expect("Failed to decode JWKS"); |     let jwks: JWKS = serde_json::from_str(json).expect("Failed to decode JWKS"); | ||||||
|     let jwk = jwks.find("mUjI/rIMLLtung35BKZfdbrqtlEAAYJ4JX/SKvnLxJc=") |     let jwk = jwks | ||||||
|  |         .find("mUjI/rIMLLtung35BKZfdbrqtlEAAYJ4JX/SKvnLxJc=") | ||||||
|         .expect("Failed to find required JWK"); |         .expect("Failed to find required JWK"); | ||||||
| 
 | 
 | ||||||
|     public_key_from_jwk(&jwk).expect("Failed to construct public key from JWK"); |     public_key_from_jwk(&jwk).expect("Failed to construct public key from JWK"); | ||||||
|  | @ -39,18 +44,21 @@ fn test_token_kid() { | ||||||
|     let jwt = "eyJraWQiOiI4ckRxOFB3MEZaY2FvWFdURVZRbzcrVGYyWXpTTDFmQnhOS1BDZWJhYWk0PSIsImFsZyI6IlJTMjU2IiwidHlwIjoiSldUIn0.eyJpc3MiOiJhdXRoLnRlc3QuYXByaWxhLm5vIiwiaWF0IjoxNTM2MDUwNjkzLCJleHAiOjE1MzYwNTQyOTMsInN1YiI6IjQyIiwiZXh0Ijoic21va2V0ZXN0IiwicHJ2IjoiYXJpc3RpIiwic2NwIjoicHJvY2VzcyJ9.gOLsv98109qLkmRK6Dn7WWRHLW7o8W78WZcWvFZoxPLzVO0qvRXXRLYc9h5chpfvcWreLZ4f1cOdvxv31_qnCRSQQPOeQ7r7hj_sPEDzhKjk-q2aoNHaGGJg1vabI--9EFkFsGQfoS7UbMMssS44dgR68XEnKtjn0Vys-Vzbvz_CBSCH6yQhRLik2SU2jR2L7BoFvh4LGZ6EKoQWzm8Z-CHXLGLUs4Hp5aPhF46dGzgAzwlPFW4t9G4DciX1uB4vv1XnfTc5wqJch6ltjKMde1GZwLR757a8dJSBcmGWze3UNE2YH_VLD7NCwH2kkqr3gh8rn7lWKG4AUIYPxsw9CB"; |     let jwt = "eyJraWQiOiI4ckRxOFB3MEZaY2FvWFdURVZRbzcrVGYyWXpTTDFmQnhOS1BDZWJhYWk0PSIsImFsZyI6IlJTMjU2IiwidHlwIjoiSldUIn0.eyJpc3MiOiJhdXRoLnRlc3QuYXByaWxhLm5vIiwiaWF0IjoxNTM2MDUwNjkzLCJleHAiOjE1MzYwNTQyOTMsInN1YiI6IjQyIiwiZXh0Ijoic21va2V0ZXN0IiwicHJ2IjoiYXJpc3RpIiwic2NwIjoicHJvY2VzcyJ9.gOLsv98109qLkmRK6Dn7WWRHLW7o8W78WZcWvFZoxPLzVO0qvRXXRLYc9h5chpfvcWreLZ4f1cOdvxv31_qnCRSQQPOeQ7r7hj_sPEDzhKjk-q2aoNHaGGJg1vabI--9EFkFsGQfoS7UbMMssS44dgR68XEnKtjn0Vys-Vzbvz_CBSCH6yQhRLik2SU2jR2L7BoFvh4LGZ6EKoQWzm8Z-CHXLGLUs4Hp5aPhF46dGzgAzwlPFW4t9G4DciX1uB4vv1XnfTc5wqJch6ltjKMde1GZwLR757a8dJSBcmGWze3UNE2YH_VLD7NCwH2kkqr3gh8rn7lWKG4AUIYPxsw9CB"; | ||||||
| 
 | 
 | ||||||
|     let kid = token_kid(&jwt).expect("Failed to extract token KID"); |     let kid = token_kid(&jwt).expect("Failed to extract token KID"); | ||||||
|     assert_eq!(Some("8rDq8Pw0FZcaoXWTEVQo7+Tf2YzSL1fBxNKPCebaai4=".into()), |     assert_eq!( | ||||||
|                kid, "Extracted KID did not match expected KID"); |         Some("8rDq8Pw0FZcaoXWTEVQo7+Tf2YzSL1fBxNKPCebaai4=".into()), | ||||||
|  |         kid, | ||||||
|  |         "Extracted KID did not match expected KID" | ||||||
|  |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | #[test] | ||||||
| fn test_validate_jwt() { | fn test_validate_jwt() { | ||||||
|     let jwks_json = "{\"keys\":[{\"kty\":\"RSA\",\"alg\":\"RS256\",\"use\":\"sig\",\"kid\":\"8rDq8Pw0FZcaoXWTEVQo7+Tf2YzSL1fBxNKPCebaai4=\",\"n\":\"l4UTgk1zr-8C8utt0E57DtBV6qqAPWzVRrIuQS2j0_hp2CviaNl5XzGRDnB8gwk0Hx95YOhJupAe6RNq5ok3fDdxL7DLvppJNRLz3Ag9CsmDLcbXgNEQys33fBJaPw1v3GcaFC4tisU5p-o1f5RfWwvwdBtdBfGiwT1GRvbc5sFx6M4iYjg9uv1lNKW60PqSJW4iDYrfqzZmB0zF1SJ0BL_rnQZ1Wi_UkFmNe9arM8W9tI9T3Ie59HITFuyVSTCt6qQEtSfa1e5PiBaVuV3qoFI2jPBiVZQ6LPGBWEDyz4QtrHLdECPPoTF30NN6TSVwwlRbCuUUrdNdXdjYe2dMFQ\",\"e\":\"DhaD5zC7mzaDvHO192wKT_9sfsVmdy8w8T8C9VG17_b1jG2srd3cmc6Ycw-0blDf53Wrpi9-KGZXKHX6_uIuJK249WhkP7N1SHrTJxO0sUJ8AhK482PLF09Qtu6cUfJqY1X1y1S2vACJZItU4Vjr3YAfiVGQXeA8frAf7Sm4O1CBStCyg6yCcIbGojII0jfh2vSB-GD9ok1F69Nmk-R-bClyqMCV_Oq-5a0gqClVS8pDyGYMgKTww2RHgZaFSUcG13KeLMQsG2UOB2OjSC8FkOXK00NBlAjU3d0Vv-IamaLIszO7FQBY3Oh0uxNOvIE9ofQyCOpB-xIK6V9CTTphxw\"}]}"; |     let jwks_json = "{\"keys\":[{\"kty\":\"RSA\",\"alg\":\"RS256\",\"use\":\"sig\",\"kid\":\"8rDq8Pw0FZcaoXWTEVQo7+Tf2YzSL1fBxNKPCebaai4=\",\"n\":\"l4UTgk1zr-8C8utt0E57DtBV6qqAPWzVRrIuQS2j0_hp2CviaNl5XzGRDnB8gwk0Hx95YOhJupAe6RNq5ok3fDdxL7DLvppJNRLz3Ag9CsmDLcbXgNEQys33fBJaPw1v3GcaFC4tisU5p-o1f5RfWwvwdBtdBfGiwT1GRvbc5sFx6M4iYjg9uv1lNKW60PqSJW4iDYrfqzZmB0zF1SJ0BL_rnQZ1Wi_UkFmNe9arM8W9tI9T3Ie59HITFuyVSTCt6qQEtSfa1e5PiBaVuV3qoFI2jPBiVZQ6LPGBWEDyz4QtrHLdECPPoTF30NN6TSVwwlRbCuUUrdNdXdjYe2dMFQ\",\"e\":\"DhaD5zC7mzaDvHO192wKT_9sfsVmdy8w8T8C9VG17_b1jG2srd3cmc6Ycw-0blDf53Wrpi9-KGZXKHX6_uIuJK249WhkP7N1SHrTJxO0sUJ8AhK482PLF09Qtu6cUfJqY1X1y1S2vACJZItU4Vjr3YAfiVGQXeA8frAf7Sm4O1CBStCyg6yCcIbGojII0jfh2vSB-GD9ok1F69Nmk-R-bClyqMCV_Oq-5a0gqClVS8pDyGYMgKTww2RHgZaFSUcG13KeLMQsG2UOB2OjSC8FkOXK00NBlAjU3d0Vv-IamaLIszO7FQBY3Oh0uxNOvIE9ofQyCOpB-xIK6V9CTTphxw\"}]}"; | ||||||
| 
 | 
 | ||||||
|     let jwks: JWKS = serde_json::from_str(jwks_json) |     let jwks: JWKS = serde_json::from_str(jwks_json).expect("Failed to decode JWKS"); | ||||||
|         .expect("Failed to decode JWKS"); |  | ||||||
| 
 | 
 | ||||||
|     let jwk = jwks.find("8rDq8Pw0FZcaoXWTEVQo7+Tf2YzSL1fBxNKPCebaai4=") |     let jwk = jwks | ||||||
|  |         .find("8rDq8Pw0FZcaoXWTEVQo7+Tf2YzSL1fBxNKPCebaai4=") | ||||||
|         .expect("Failed to find required JWK"); |         .expect("Failed to find required JWK"); | ||||||
| 
 | 
 | ||||||
|     let pkey = public_key_from_jwk(&jwk).expect("Failed to construct public key"); |     let pkey = public_key_from_jwk(&jwk).expect("Failed to construct public key"); | ||||||
|  |  | ||||||
|  | @ -33,9 +33,12 @@ | ||||||
| //! use crimp::Request;
 | //! use crimp::Request;
 | ||||||
| //!
 | //!
 | ||||||
| //! let response = Request::get("http://httpbin.org/get")
 | //! let response = Request::get("http://httpbin.org/get")
 | ||||||
| //!     .user_agent("crimp test suite").unwrap()
 | //!     .user_agent("crimp test suite")
 | ||||||
| //!     .send().unwrap()
 | //!     .unwrap()
 | ||||||
| //!     .as_string().unwrap();
 | //!     .send()
 | ||||||
|  | //!     .unwrap()
 | ||||||
|  | //!     .as_string()
 | ||||||
|  | //!     .unwrap();
 | ||||||
| //!
 | //!
 | ||||||
| //! println!("Status: {}\nBody: {}", response.status, response.body);
 | //! println!("Status: {}\nBody: {}", response.status, response.body);
 | ||||||
| //! # assert_eq!(response.status, 200);
 | //! # assert_eq!(response.status, 200);
 | ||||||
|  | @ -54,10 +57,9 @@ | ||||||
| //!
 | //!
 | ||||||
| //! All optional features are enabled by default.
 | //! All optional features are enabled by default.
 | ||||||
| //!
 | //!
 | ||||||
| //! * `json`: Adds `Request::json` and `Response::as_json` methods
 | //! * `json`: Adds `Request::json` and `Response::as_json` methods which can be used for convenient
 | ||||||
| //!   which can be used for convenient serialisation of
 | //!   serialisation of request/response bodies using `serde_json`. This feature adds a dependency on
 | ||||||
| //!   request/response bodies using `serde_json`. This feature adds a
 | //!   the `serde` and `serde_json` crates.
 | ||||||
| //!   dependency on the `serde` and `serde_json` crates.
 |  | ||||||
| //!
 | //!
 | ||||||
| //! ## Initialisation
 | //! ## Initialisation
 | ||||||
| //!
 | //!
 | ||||||
|  | @ -72,32 +74,42 @@ | ||||||
| 
 | 
 | ||||||
| extern crate curl; | extern crate curl; | ||||||
| 
 | 
 | ||||||
| #[cfg(feature = "json")] extern crate serde; | #[cfg(feature = "json")] | ||||||
| #[cfg(feature = "json")] extern crate serde_json; | extern crate serde; | ||||||
|  | #[cfg(feature = "json")] | ||||||
|  | extern crate serde_json; | ||||||
| 
 | 
 | ||||||
| pub use curl::init; | pub use curl::init; | ||||||
| 
 | 
 | ||||||
| use curl::easy::{Auth, Easy, Form, List, Transfer, ReadError, WriteError}; | use curl::easy::{Auth, Easy, Form, List, ReadError, Transfer, WriteError}; | ||||||
| use std::collections::HashMap; | use std::collections::HashMap; | ||||||
| use std::io::Write; | use std::io::Write; | ||||||
| use std::path::Path; | use std::path::Path; | ||||||
| use std::string::{FromUtf8Error, ToString}; | use std::string::{FromUtf8Error, ToString}; | ||||||
| use std::time::Duration; | use std::time::Duration; | ||||||
| 
 | 
 | ||||||
| #[cfg(feature = "json")] use serde::Serialize; | #[cfg(feature = "json")] | ||||||
| #[cfg(feature = "json")] use serde::de::DeserializeOwned; | use serde::de::DeserializeOwned; | ||||||
|  | #[cfg(feature = "json")] | ||||||
|  | use serde::Serialize; | ||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests; | mod tests; | ||||||
| 
 | 
 | ||||||
| /// HTTP method to use for the request.
 | /// HTTP method to use for the request.
 | ||||||
| enum Method { | enum Method { | ||||||
|     Get, Post, Put, Patch, Delete |     Get, | ||||||
|  |     Post, | ||||||
|  |     Put, | ||||||
|  |     Patch, | ||||||
|  |     Delete, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Certificate types for client-certificate key pairs.
 | /// Certificate types for client-certificate key pairs.
 | ||||||
| pub enum CertType { | pub enum CertType { | ||||||
|     P12, PEM, DER |     P12, | ||||||
|  |     PEM, | ||||||
|  |     DER, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Builder structure for an HTTP request.
 | /// Builder structure for an HTTP request.
 | ||||||
|  | @ -158,19 +170,29 @@ impl <'a> Request<'a> { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// Initiate a GET request with the given URL.
 |     /// Initiate a GET request with the given URL.
 | ||||||
|     pub fn get(url: &'a str) -> Self { Request::new(Method::Get, url) } |     pub fn get(url: &'a str) -> Self { | ||||||
|  |         Request::new(Method::Get, url) | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     /// Initiate a POST request with the given URL.
 |     /// Initiate a POST request with the given URL.
 | ||||||
|     pub fn post(url: &'a str) -> Self { Request::new(Method::Post, url) } |     pub fn post(url: &'a str) -> Self { | ||||||
|  |         Request::new(Method::Post, url) | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     /// Initiate a PUT request with the given URL.
 |     /// Initiate a PUT request with the given URL.
 | ||||||
|     pub fn put(url: &'a str) -> Self { Request::new(Method::Put, url) } |     pub fn put(url: &'a str) -> Self { | ||||||
|  |         Request::new(Method::Put, url) | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     /// Initiate a PATCH request with the given URL.
 |     /// Initiate a PATCH request with the given URL.
 | ||||||
|     pub fn patch(url: &'a str) -> Self { Request::new(Method::Patch, url) } |     pub fn patch(url: &'a str) -> Self { | ||||||
|  |         Request::new(Method::Patch, url) | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     /// Initiate a DELETE request with the given URL.
 |     /// Initiate a DELETE request with the given URL.
 | ||||||
|     pub fn delete(url: &'a str) -> Self { Request::new(Method::Delete, url) } |     pub fn delete(url: &'a str) -> Self { | ||||||
|  |         Request::new(Method::Delete, url) | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     /// Add an HTTP header to a request.
 |     /// Add an HTTP header to a request.
 | ||||||
|     pub fn header(mut self, k: &str, v: &str) -> Result<Self, curl::Error> { |     pub fn header(mut self, k: &str, v: &str) -> Result<Self, curl::Error> { | ||||||
|  | @ -188,7 +210,8 @@ impl <'a> Request<'a> { | ||||||
|     /// Set the `Authorization` header to a `Bearer` value with the
 |     /// Set the `Authorization` header to a `Bearer` value with the
 | ||||||
|     /// supplied token.
 |     /// supplied token.
 | ||||||
|     pub fn bearer_auth(mut self, token: &str) -> Result<Self, curl::Error> { |     pub fn bearer_auth(mut self, token: &str) -> Result<Self, curl::Error> { | ||||||
|         self.headers.append(&format!("Authorization: Bearer {}", token))?; |         self.headers | ||||||
|  |             .append(&format!("Authorization: Bearer {}", token))?; | ||||||
|         Ok(self) |         Ok(self) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -212,8 +235,11 @@ impl <'a> Request<'a> { | ||||||
|     /// Consult the documentation for the `ssl_cert` and `ssl_key`
 |     /// Consult the documentation for the `ssl_cert` and `ssl_key`
 | ||||||
|     /// functions in `curl::easy::Easy2` for details on supported
 |     /// functions in `curl::easy::Easy2` for details on supported
 | ||||||
|     /// formats and defaults.
 |     /// formats and defaults.
 | ||||||
|     pub fn tls_client_cert<P: AsRef<Path>>(mut self, cert_type: CertType, cert: P) |     pub fn tls_client_cert<P: AsRef<Path>>( | ||||||
|                                            -> Result<Self, curl::Error> { |         mut self, | ||||||
|  |         cert_type: CertType, | ||||||
|  |         cert: P, | ||||||
|  |     ) -> Result<Self, curl::Error> { | ||||||
|         self.handle.ssl_cert(cert)?; |         self.handle.ssl_cert(cert)?; | ||||||
|         self.handle.ssl_cert_type(match cert_type { |         self.handle.ssl_cert_type(match cert_type { | ||||||
|             CertType::P12 => "P12", |             CertType::P12 => "P12", | ||||||
|  | @ -262,13 +288,17 @@ impl <'a> Request<'a> { | ||||||
|     /// ```
 |     /// ```
 | ||||||
|     /// # use crimp::Request;
 |     /// # use crimp::Request;
 | ||||||
|     /// let response = Request::get("https://httpbin.org/get")
 |     /// let response = Request::get("https://httpbin.org/get")
 | ||||||
|     ///     .with_handle(|mut handle| handle.referer("Example-Referer")).unwrap()
 |     ///     .with_handle(|mut handle| handle.referer("Example-Referer"))
 | ||||||
|     ///     .send().unwrap();
 |     ///     .unwrap()
 | ||||||
|  |     ///     .send()
 | ||||||
|  |     ///     .unwrap();
 | ||||||
|     /// #
 |     /// #
 | ||||||
|     /// # assert!(response.is_success());
 |     /// # assert!(response.is_success());
 | ||||||
|     /// ```
 |     /// ```
 | ||||||
|     pub fn with_handle<F>(mut self, function: F) -> Result<Self, curl::Error> |     pub fn with_handle<F>(mut self, function: F) -> Result<Self, curl::Error> | ||||||
|     where F: FnOnce(&mut Easy) -> Result<(), curl::Error> { |     where | ||||||
|  |         F: FnOnce(&mut Easy) -> Result<(), curl::Error>, | ||||||
|  |     { | ||||||
|         function(&mut self.handle)?; |         function(&mut self.handle)?; | ||||||
|         Ok(self) |         Ok(self) | ||||||
|     } |     } | ||||||
|  | @ -293,12 +323,15 @@ impl <'a> Request<'a> { | ||||||
|     /// let mut form = Form::new();
 |     /// let mut form = Form::new();
 | ||||||
|     /// form.part("some-name")
 |     /// form.part("some-name")
 | ||||||
|     ///     .contents("some-data".as_bytes())
 |     ///     .contents("some-data".as_bytes())
 | ||||||
|     ///     .add().unwrap();
 |     ///     .add()
 | ||||||
|  |     ///     .unwrap();
 | ||||||
|     ///
 |     ///
 | ||||||
|     /// let response = Request::post("https://httpbin.org/post")
 |     /// let response = Request::post("https://httpbin.org/post")
 | ||||||
|     ///     .user_agent("crimp test suite").unwrap()
 |     ///     .user_agent("crimp test suite")
 | ||||||
|  |     ///     .unwrap()
 | ||||||
|     ///     .form(form)
 |     ///     .form(form)
 | ||||||
|     ///     .send().unwrap();
 |     ///     .send()
 | ||||||
|  |     ///     .unwrap();
 | ||||||
|     /// #
 |     /// #
 | ||||||
|     /// # assert_eq!(200, response.status, "form POST should succeed");
 |     /// # assert_eq!(200, response.status, "form POST should succeed");
 | ||||||
|     /// # assert_eq!(
 |     /// # assert_eq!(
 | ||||||
|  | @ -354,14 +387,15 @@ impl <'a> Request<'a> { | ||||||
|         match self.body { |         match self.body { | ||||||
|             Body::Bytes { content_type, data } => { |             Body::Bytes { content_type, data } => { | ||||||
|                 self.handle.post_field_size(data.len() as u64)?; |                 self.handle.post_field_size(data.len() as u64)?; | ||||||
|                 self.headers.append(&format!("Content-Type: {}", content_type))?; |                 self.headers | ||||||
|             }, |                     .append(&format!("Content-Type: {}", content_type))?; | ||||||
|  |             } | ||||||
| 
 | 
 | ||||||
|             #[cfg(feature = "json")] |             #[cfg(feature = "json")] | ||||||
|             Body::Json(ref data) => { |             Body::Json(ref data) => { | ||||||
|                 self.handle.post_field_size(data.len() as u64)?; |                 self.handle.post_field_size(data.len() as u64)?; | ||||||
|                 self.headers.append("Content-Type: application/json")?; |                 self.headers.append("Content-Type: application/json")?; | ||||||
|             }, |             } | ||||||
| 
 | 
 | ||||||
|             // Do not set content-type header at all if there is no
 |             // Do not set content-type header at all if there is no
 | ||||||
|             // body, or if the form handler was invoked above.
 |             // body, or if the form handler was invoked above.
 | ||||||
|  | @ -407,9 +441,7 @@ impl <'a> Request<'a> { | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 headers.insert( |                 headers.insert(split[0].trim().to_string(), split[1].trim().to_string()); | ||||||
|                     split[0].trim().to_string(), split[1].trim().to_string() |  | ||||||
|                 ); |  | ||||||
|                 true |                 true | ||||||
|             })?; |             })?; | ||||||
| 
 | 
 | ||||||
|  | @ -427,7 +459,7 @@ impl <'a> Request<'a> { | ||||||
|         Ok(Response { |         Ok(Response { | ||||||
|             status: self.handle.response_code()?, |             status: self.handle.response_code()?, | ||||||
|             headers, |             headers, | ||||||
|             body |             body, | ||||||
|         }) |         }) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -438,13 +470,14 @@ impl <'a> Request<'a> { | ||||||
| ///
 | ///
 | ||||||
| /// As we manually set the expected upload size, cURL will call the
 | /// As we manually set the expected upload size, cURL will call the
 | ||||||
| /// read callback repeatedly until it has all the data it needs.
 | /// read callback repeatedly until it has all the data it needs.
 | ||||||
| fn chunked_read_function<'easy, 'data>(transfer: &mut Transfer<'easy, 'data>, | fn chunked_read_function<'easy, 'data>( | ||||||
|                                        data: &'data [u8]) -> Result<(), curl::Error> { |     transfer: &mut Transfer<'easy, 'data>, | ||||||
|  |     data: &'data [u8], | ||||||
|  | ) -> Result<(), curl::Error> { | ||||||
|     let mut data = data; |     let mut data = data; | ||||||
| 
 | 
 | ||||||
|     transfer.read_function(move |mut into| { |     transfer.read_function(move |mut into| { | ||||||
|         let written = into.write(data) |         let written = into.write(data).map_err(|_| ReadError::Abort)?; | ||||||
|             .map_err(|_| ReadError::Abort)?; |  | ||||||
| 
 | 
 | ||||||
|         data = &data[written..]; |         data = &data[written..]; | ||||||
| 
 | 
 | ||||||
|  | @ -466,9 +499,11 @@ impl <T> Response<T> { | ||||||
|     /// This function exists for convenience to avoid having to write
 |     /// This function exists for convenience to avoid having to write
 | ||||||
|     /// repetitive `if !response.is_success() { ... }` blocks.
 |     /// repetitive `if !response.is_success() { ... }` blocks.
 | ||||||
|     pub fn error_for_status<F, E>(self, closure: F) -> Result<Self, E> |     pub fn error_for_status<F, E>(self, closure: F) -> Result<Self, E> | ||||||
|     where F: FnOnce(Self) -> E { |     where | ||||||
|  |         F: FnOnce(Self) -> E, | ||||||
|  |     { | ||||||
|         if !self.is_success() { |         if !self.is_success() { | ||||||
|             return Err(closure(self)) |             return Err(closure(self)); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         Ok(self) |         Ok(self) | ||||||
|  |  | ||||||
|  | @ -6,7 +6,7 @@ | ||||||
| //    docker run --rm -p 4662:80 kennethreitz/httpbin
 | //    docker run --rm -p 4662:80 kennethreitz/httpbin
 | ||||||
| 
 | 
 | ||||||
| use super::*; | use super::*; | ||||||
| use serde_json::{Value, json}; | use serde_json::{json, Value}; | ||||||
| 
 | 
 | ||||||
| // These tests check whether the correct HTTP method is used in the
 | // These tests check whether the correct HTTP method is used in the
 | ||||||
| // requests.
 | // requests.
 | ||||||
|  | @ -14,7 +14,8 @@ use serde_json::{Value, json}; | ||||||
| #[test] | #[test] | ||||||
| fn test_http_get() { | fn test_http_get() { | ||||||
|     let resp = Request::get("http://127.0.0.1:4662/get") |     let resp = Request::get("http://127.0.0.1:4662/get") | ||||||
|         .send().expect("failed to send request"); |         .send() | ||||||
|  |         .expect("failed to send request"); | ||||||
| 
 | 
 | ||||||
|     assert!(resp.is_success(), "request should have succeeded"); |     assert!(resp.is_success(), "request should have succeeded"); | ||||||
| } | } | ||||||
|  | @ -22,7 +23,8 @@ fn test_http_get() { | ||||||
| #[test] | #[test] | ||||||
| fn test_http_delete() { | fn test_http_delete() { | ||||||
|     let resp = Request::delete("http://127.0.0.1:4662/delete") |     let resp = Request::delete("http://127.0.0.1:4662/delete") | ||||||
|         .send().expect("failed to send request"); |         .send() | ||||||
|  |         .expect("failed to send request"); | ||||||
| 
 | 
 | ||||||
|     assert_eq!(200, resp.status, "response status should be 200 OK"); |     assert_eq!(200, resp.status, "response status should be 200 OK"); | ||||||
| } | } | ||||||
|  | @ -30,7 +32,8 @@ fn test_http_delete() { | ||||||
| #[test] | #[test] | ||||||
| fn test_http_put() { | fn test_http_put() { | ||||||
|     let resp = Request::put("http://127.0.0.1:4662/put") |     let resp = Request::put("http://127.0.0.1:4662/put") | ||||||
|         .send().expect("failed to send request"); |         .send() | ||||||
|  |         .expect("failed to send request"); | ||||||
| 
 | 
 | ||||||
|     assert_eq!(200, resp.status, "response status should be 200 OK"); |     assert_eq!(200, resp.status, "response status should be 200 OK"); | ||||||
| } | } | ||||||
|  | @ -38,7 +41,8 @@ fn test_http_put() { | ||||||
| #[test] | #[test] | ||||||
| fn test_http_patch() { | fn test_http_patch() { | ||||||
|     let resp = Request::patch("http://127.0.0.1:4662/patch") |     let resp = Request::patch("http://127.0.0.1:4662/patch") | ||||||
|         .send().expect("failed to send request"); |         .send() | ||||||
|  |         .expect("failed to send request"); | ||||||
| 
 | 
 | ||||||
|     assert_eq!(200, resp.status, "response status should be 200 OK"); |     assert_eq!(200, resp.status, "response status should be 200 OK"); | ||||||
| } | } | ||||||
|  | @ -50,18 +54,25 @@ fn test_http_patch() { | ||||||
| fn test_http_post() { | fn test_http_post() { | ||||||
|     let body = "test body"; |     let body = "test body"; | ||||||
|     let response = Request::post("http://127.0.0.1:4662/post") |     let response = Request::post("http://127.0.0.1:4662/post") | ||||||
|         .user_agent("crimp test suite").expect("failed to set user-agent") |         .user_agent("crimp test suite") | ||||||
|         .timeout(Duration::from_secs(5)).expect("failed to set request timeout") |         .expect("failed to set user-agent") | ||||||
|  |         .timeout(Duration::from_secs(5)) | ||||||
|  |         .expect("failed to set request timeout") | ||||||
|         .body("text/plain", &body.as_bytes()) |         .body("text/plain", &body.as_bytes()) | ||||||
|         .send().expect("failed to send request") |         .send() | ||||||
|         .as_json::<Value>().expect("failed to deserialize response"); |         .expect("failed to send request") | ||||||
|  |         .as_json::<Value>() | ||||||
|  |         .expect("failed to deserialize response"); | ||||||
| 
 | 
 | ||||||
|     let data = response.body; |     let data = response.body; | ||||||
| 
 | 
 | ||||||
|     assert_eq!(200, response.status, "response status should be 200 OK"); |     assert_eq!(200, response.status, "response status should be 200 OK"); | ||||||
| 
 | 
 | ||||||
|     assert_eq!(data.get("data").unwrap(), &json!("test body"), |     assert_eq!( | ||||||
|                "test body should have been POSTed"); |         data.get("data").unwrap(), | ||||||
|  |         &json!("test body"), | ||||||
|  |         "test body should have been POSTed" | ||||||
|  |     ); | ||||||
| 
 | 
 | ||||||
|     assert_eq!( |     assert_eq!( | ||||||
|         data.get("headers").unwrap().get("Content-Type").unwrap(), |         data.get("headers").unwrap().get("Content-Type").unwrap(), | ||||||
|  | @ -70,26 +81,34 @@ fn test_http_post() { | ||||||
|     ); |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[cfg(feature = "json")] #[test] | #[cfg(feature = "json")] | ||||||
|  | #[test] | ||||||
| fn test_http_post_json() { | fn test_http_post_json() { | ||||||
|     let body = json!({ |     let body = json!({ | ||||||
|         "purpose": "testing!" |         "purpose": "testing!" | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     let response = Request::post("http://127.0.0.1:4662/post") |     let response = Request::post("http://127.0.0.1:4662/post") | ||||||
|         .user_agent("crimp test suite").expect("failed to set user-agent") |         .user_agent("crimp test suite") | ||||||
|         .timeout(Duration::from_secs(5)).expect("failed to set request timeout") |         .expect("failed to set user-agent") | ||||||
|         .json(&body).expect("request serialization failed") |         .timeout(Duration::from_secs(5)) | ||||||
|         .send().expect("failed to send request") |         .expect("failed to set request timeout") | ||||||
|         .as_json::<Value>().expect("failed to deserialize response"); |         .json(&body) | ||||||
| 
 |         .expect("request serialization failed") | ||||||
|  |         .send() | ||||||
|  |         .expect("failed to send request") | ||||||
|  |         .as_json::<Value>() | ||||||
|  |         .expect("failed to deserialize response"); | ||||||
| 
 | 
 | ||||||
|     let data = response.body; |     let data = response.body; | ||||||
| 
 | 
 | ||||||
|     assert_eq!(200, response.status, "response status should be 200 OK"); |     assert_eq!(200, response.status, "response status should be 200 OK"); | ||||||
| 
 | 
 | ||||||
|     assert_eq!(data.get("json").unwrap(), &body, |     assert_eq!( | ||||||
|                "test body should have been POSTed"); |         data.get("json").unwrap(), | ||||||
|  |         &body, | ||||||
|  |         "test body should have been POSTed" | ||||||
|  |     ); | ||||||
| 
 | 
 | ||||||
|     assert_eq!( |     assert_eq!( | ||||||
|         data.get("headers").unwrap().get("Content-Type").unwrap(), |         data.get("headers").unwrap().get("Content-Type").unwrap(), | ||||||
|  | @ -104,8 +123,10 @@ fn test_http_post_json() { | ||||||
| #[test] | #[test] | ||||||
| fn test_bearer_auth() { | fn test_bearer_auth() { | ||||||
|     let response = Request::get("http://127.0.0.1:4662/bearer") |     let response = Request::get("http://127.0.0.1:4662/bearer") | ||||||
|         .bearer_auth("some-token").expect("failed to set auth header") |         .bearer_auth("some-token") | ||||||
|         .send().expect("failed to send request"); |         .expect("failed to set auth header") | ||||||
|  |         .send() | ||||||
|  |         .expect("failed to send request"); | ||||||
| 
 | 
 | ||||||
|     assert!(response.is_success(), "authorized request should succeed"); |     assert!(response.is_success(), "authorized request should succeed"); | ||||||
| } | } | ||||||
|  | @ -115,8 +136,10 @@ fn test_basic_auth() { | ||||||
|     let request = Request::get("http://127.0.0.1:4662/basic-auth/alan_watts/oneness"); |     let request = Request::get("http://127.0.0.1:4662/basic-auth/alan_watts/oneness"); | ||||||
| 
 | 
 | ||||||
|     let response = request |     let response = request | ||||||
|         .basic_auth("alan_watts", "oneness").expect("failed to set auth header") |         .basic_auth("alan_watts", "oneness") | ||||||
|         .send().expect("failed to send request"); |         .expect("failed to set auth header") | ||||||
|  |         .send() | ||||||
|  |         .expect("failed to send request"); | ||||||
| 
 | 
 | ||||||
|     assert!(response.is_success(), "authorized request should succeed"); |     assert!(response.is_success(), "authorized request should succeed"); | ||||||
| } | } | ||||||
|  | @ -129,14 +152,20 @@ fn test_large_body() { | ||||||
| 
 | 
 | ||||||
|     let resp = Request::post("http://127.0.0.1:4662/post") |     let resp = Request::post("http://127.0.0.1:4662/post") | ||||||
|         .body("application/octet-stream", &[0; BODY_SIZE]) |         .body("application/octet-stream", &[0; BODY_SIZE]) | ||||||
|         .send().expect("sending request") |         .send() | ||||||
|         .as_json::<Value>().expect("JSON deserialisation"); |         .expect("sending request") | ||||||
|  |         .as_json::<Value>() | ||||||
|  |         .expect("JSON deserialisation"); | ||||||
| 
 | 
 | ||||||
|     // httpbin returns the uploaded data as a string in the `data`
 |     // httpbin returns the uploaded data as a string in the `data`
 | ||||||
|     // field.
 |     // field.
 | ||||||
|     let data = resp.body.get("data").unwrap().as_str().unwrap(); |     let data = resp.body.get("data").unwrap().as_str().unwrap(); | ||||||
| 
 | 
 | ||||||
|     assert_eq!(BODY_SIZE, data.len(), "uploaded data length should be correct"); |     assert_eq!( | ||||||
|  |         BODY_SIZE, | ||||||
|  |         data.len(), | ||||||
|  |         "uploaded data length should be correct" | ||||||
|  |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Tests for various other features.
 | // Tests for various other features.
 | ||||||
|  | @ -144,9 +173,13 @@ fn test_large_body() { | ||||||
| #[test] | #[test] | ||||||
| fn test_error_for_status() { | fn test_error_for_status() { | ||||||
|     let response = Request::get("http://127.0.0.1:4662/patch") |     let response = Request::get("http://127.0.0.1:4662/patch") | ||||||
|         .send().expect("failed to send request") |         .send() | ||||||
|  |         .expect("failed to send request") | ||||||
|         .error_for_status(|resp| format!("Response error code: {}", resp.status)); |         .error_for_status(|resp| format!("Response error code: {}", resp.status)); | ||||||
| 
 | 
 | ||||||
|     assert_eq!(Err("Response error code: 405".into()), response, |     assert_eq!( | ||||||
|                "returned error should be converted into Result::Err"); |         Err("Response error code: 405".into()), | ||||||
|  |         response, | ||||||
|  |         "returned error should be converted into Result::Err" | ||||||
|  |     ); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,11 +1,11 @@ | ||||||
| extern crate serde_json; | extern crate serde_json; | ||||||
| 
 | 
 | ||||||
| use serde_json::Value; | use serde_json::Value; | ||||||
|  | use std::convert::TryFrom; | ||||||
| use std::ffi::OsString; | use std::ffi::OsString; | ||||||
| use std::os::unix::ffi::{OsStringExt, OsStrExt}; | use std::io::{stderr, stdout, Error, ErrorKind, Write}; | ||||||
| use std::io::{Error, ErrorKind, Write, stdout, stderr}; | use std::os::unix::ffi::{OsStrExt, OsStringExt}; | ||||||
| use std::process::Command; | use std::process::Command; | ||||||
| use std::convert::{TryFrom}; |  | ||||||
| 
 | 
 | ||||||
| fn render_nix_string(s: &OsString) -> OsString { | fn render_nix_string(s: &OsString) -> OsString { | ||||||
|     let mut rendered = Vec::new(); |     let mut rendered = Vec::new(); | ||||||
|  | @ -48,17 +48,14 @@ fn render_nix_list(arr: &[OsString]) -> OsString { | ||||||
| macro_rules! handle_set_output { | macro_rules! handle_set_output { | ||||||
|     ($map_name:ident, $output_name:ident) => { |     ($map_name:ident, $output_name:ident) => { | ||||||
|         match $map_name.get(stringify!($output_name)) { |         match $map_name.get(stringify!($output_name)) { | ||||||
|             Some(Value::String(s)) => |             Some(Value::String(s)) => $output_name().write_all(s.as_bytes()), | ||||||
|                 $output_name().write_all(s.as_bytes()), |             Some(_) => Err(Error::new( | ||||||
|             Some(_) => Err( |  | ||||||
|                 Error::new( |  | ||||||
|                 ErrorKind::Other, |                 ErrorKind::Other, | ||||||
|                 format!("Attribute {} must be a string!", stringify!($output_name)), |                 format!("Attribute {} must be a string!", stringify!($output_name)), | ||||||
|                 ) |             )), | ||||||
|             ), |  | ||||||
|             None => Ok(()), |             None => Ok(()), | ||||||
|         } |         } | ||||||
|     } |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn main() -> std::io::Result<()> { | fn main() -> std::io::Result<()> { | ||||||
|  | @ -116,9 +113,7 @@ fn main() -> std::io::Result<()> { | ||||||
| 
 | 
 | ||||||
|         nix_args.push(argv[0].clone()); |         nix_args.push(argv[0].clone()); | ||||||
| 
 | 
 | ||||||
|         let run = Command::new("nix-instantiate") |         let run = Command::new("nix-instantiate").args(nix_args).output()?; | ||||||
|                           .args(nix_args) |  | ||||||
|                           .output()?; |  | ||||||
| 
 | 
 | ||||||
|         match serde_json::from_slice(&run.stdout[..]) { |         match serde_json::from_slice(&run.stdout[..]) { | ||||||
|             Ok(Value::String(s)) => stdout().write_all(s.as_bytes()), |             Ok(Value::String(s)) => stdout().write_all(s.as_bytes()), | ||||||
|  | @ -132,25 +127,23 @@ fn main() -> std::io::Result<()> { | ||||||
| 
 | 
 | ||||||
|                         match code { |                         match code { | ||||||
|                             Some(i) => std::process::exit(i), |                             Some(i) => std::process::exit(i), | ||||||
|                             None => Err( |                             None => { | ||||||
|                                 Error::new( |                                 Err(Error::new(ErrorKind::Other, "Attribute exit is not an i32")) | ||||||
|                                     ErrorKind::Other, |  | ||||||
|                                     "Attribute exit is not an i32" |  | ||||||
|                                 ) |  | ||||||
|                             ), |  | ||||||
|                             } |                             } | ||||||
|                     }, |                         } | ||||||
|                     Some(_) => Err( |                     } | ||||||
|                         Error::new(ErrorKind::Other, "exit must be a number") |                     Some(_) => Err(Error::new(ErrorKind::Other, "exit must be a number")), | ||||||
|                     ), |  | ||||||
|                     None => Ok(()), |                     None => Ok(()), | ||||||
|                 } |                 } | ||||||
|             }, |             } | ||||||
|             Ok(_) => Err(Error::new(ErrorKind::Other, "output must be a string or an object")), |             Ok(_) => Err(Error::new( | ||||||
|  |                 ErrorKind::Other, | ||||||
|  |                 "output must be a string or an object", | ||||||
|  |             )), | ||||||
|             _ => { |             _ => { | ||||||
|                 stderr().write_all(&run.stderr[..]); |                 stderr().write_all(&run.stderr[..]); | ||||||
|                 Err(Error::new(ErrorKind::Other, "internal nix error")) |                 Err(Error::new(ErrorKind::Other, "internal nix error")) | ||||||
|             }, |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,6 +1,5 @@ | ||||||
| extern crate pkg_config; | extern crate pkg_config; | ||||||
| 
 | 
 | ||||||
| fn main() { | fn main() { | ||||||
|     pkg_config::probe_library("libsystemd") |     pkg_config::probe_library("libsystemd").expect("Could not probe libsystemd"); | ||||||
|         .expect("Could not probe libsystemd"); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -31,11 +31,16 @@ | ||||||
| //! `GOOGLE_APPLICATION_CREDENTIALS`, `GOOGLE_CLOUD_PROJECT` and
 | //! `GOOGLE_APPLICATION_CREDENTIALS`, `GOOGLE_CLOUD_PROJECT` and
 | ||||||
| //! `LOG_NAME` environment variables.
 | //! `LOG_NAME` environment variables.
 | ||||||
| 
 | 
 | ||||||
| #[macro_use] extern crate failure; | #[macro_use] | ||||||
| #[macro_use] extern crate log; | extern crate failure; | ||||||
| #[macro_use] extern crate serde_derive; | #[macro_use] | ||||||
| #[macro_use] extern crate serde_json; | extern crate log; | ||||||
| #[macro_use] extern crate lazy_static; | #[macro_use] | ||||||
|  | extern crate serde_derive; | ||||||
|  | #[macro_use] | ||||||
|  | extern crate serde_json; | ||||||
|  | #[macro_use] | ||||||
|  | extern crate lazy_static; | ||||||
| 
 | 
 | ||||||
| extern crate chrono; | extern crate chrono; | ||||||
| extern crate env_logger; | extern crate env_logger; | ||||||
|  | @ -48,13 +53,11 @@ use chrono::offset::LocalResult; | ||||||
| use chrono::prelude::{DateTime, TimeZone, Utc}; | use chrono::prelude::{DateTime, TimeZone, Utc}; | ||||||
| use failure::ResultExt; | use failure::ResultExt; | ||||||
| use serde_json::{from_str, Value}; | use serde_json::{from_str, Value}; | ||||||
| use std::env; | use std::fs::{self, rename, File}; | ||||||
| use std::fs::{self, File, rename}; | use std::io::{self, ErrorKind, Read, Write}; | ||||||
| use std::io::{self, Read, ErrorKind, Write}; |  | ||||||
| use std::mem; |  | ||||||
| use std::path::PathBuf; | use std::path::PathBuf; | ||||||
| use std::process; |  | ||||||
| use std::time::{Duration, Instant}; | use std::time::{Duration, Instant}; | ||||||
|  | use std::{env, mem, process}; | ||||||
| use systemd::journal::{Journal, JournalFiles, JournalRecord, JournalSeek}; | use systemd::journal::{Journal, JournalFiles, JournalRecord, JournalSeek}; | ||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
|  | @ -62,10 +65,12 @@ mod tests; | ||||||
| 
 | 
 | ||||||
| const LOGGING_SERVICE: &str = "https://logging.googleapis.com/google.logging.v2.LoggingServiceV2"; | const LOGGING_SERVICE: &str = "https://logging.googleapis.com/google.logging.v2.LoggingServiceV2"; | ||||||
| const ENTRIES_WRITE_URL: &str = "https://logging.googleapis.com/v2/entries:write"; | const ENTRIES_WRITE_URL: &str = "https://logging.googleapis.com/v2/entries:write"; | ||||||
| const METADATA_TOKEN_URL: &str = "http://metadata.google.internal/computeMetadata/v1/instance/service-accounts/default/token"; | const METADATA_TOKEN_URL: &str = | ||||||
|  |     "http://metadata.google.internal/computeMetadata/v1/instance/service-accounts/default/token"; | ||||||
| const METADATA_ID_URL: &str = "http://metadata.google.internal/computeMetadata/v1/instance/id"; | const METADATA_ID_URL: &str = "http://metadata.google.internal/computeMetadata/v1/instance/id"; | ||||||
| const METADATA_ZONE_URL: &str = "http://metadata.google.internal/computeMetadata/v1/instance/zone"; | const METADATA_ZONE_URL: &str = "http://metadata.google.internal/computeMetadata/v1/instance/zone"; | ||||||
| const METADATA_PROJECT_URL: &str = "http://metadata.google.internal/computeMetadata/v1/project/project-id"; | const METADATA_PROJECT_URL: &str = | ||||||
|  |     "http://metadata.google.internal/computeMetadata/v1/project/project-id"; | ||||||
| 
 | 
 | ||||||
| /// Convenience type alias for results using failure's `Error` type.
 | /// Convenience type alias for results using failure's `Error` type.
 | ||||||
| type Result<T> = std::result::Result<T, failure::Error>; | type Result<T> = std::result::Result<T, failure::Error>; | ||||||
|  | @ -134,14 +139,17 @@ fn get_metadata(url: &str) -> Result<String> { | ||||||
| 
 | 
 | ||||||
|     if response.ok() { |     if response.ok() { | ||||||
|         // Whitespace is trimmed to remove newlines from responses.
 |         // Whitespace is trimmed to remove newlines from responses.
 | ||||||
|         let body = response.into_string() |         let body = response | ||||||
|  |             .into_string() | ||||||
|             .context("Failed to decode metadata response")? |             .context("Failed to decode metadata response")? | ||||||
|             .trim().to_string(); |             .trim() | ||||||
|  |             .to_string(); | ||||||
| 
 | 
 | ||||||
|         Ok(body) |         Ok(body) | ||||||
|     } else { |     } else { | ||||||
|         let status = response.status_line().to_string(); |         let status = response.status_line().to_string(); | ||||||
|         let body = response.into_string() |         let body = response | ||||||
|  |             .into_string() | ||||||
|             .unwrap_or_else(|e| format!("Metadata body error: {}", e)); |             .unwrap_or_else(|e| format!("Metadata body error: {}", e)); | ||||||
|         bail!("Metadata failure: {} ({})", body, status) |         bail!("Metadata failure: {} ({})", body, status) | ||||||
|     } |     } | ||||||
|  | @ -186,11 +194,9 @@ fn determine_monitored_resource() -> Value { | ||||||
|             } |             } | ||||||
|         }) |         }) | ||||||
|     } else { |     } else { | ||||||
|         let instance_id = get_metadata(METADATA_ID_URL) |         let instance_id = get_metadata(METADATA_ID_URL).expect("Could not determine instance ID"); | ||||||
|             .expect("Could not determine instance ID"); |  | ||||||
| 
 | 
 | ||||||
|         let zone = get_metadata(METADATA_ZONE_URL) |         let zone = get_metadata(METADATA_ZONE_URL).expect("Could not determine instance zone"); | ||||||
|             .expect("Could not determine instance zone"); |  | ||||||
| 
 | 
 | ||||||
|         json!({ |         json!({ | ||||||
|             "type": "gce_instance", |             "type": "gce_instance", | ||||||
|  | @ -253,7 +259,8 @@ fn sign_service_account_token(credentials: &Credentials) -> Result<Token> { | ||||||
|     use medallion::{Algorithm, Header, Payload}; |     use medallion::{Algorithm, Header, Payload}; | ||||||
| 
 | 
 | ||||||
|     let iat = Utc::now(); |     let iat = Utc::now(); | ||||||
|     let exp = iat.checked_add_signed(chrono::Duration::seconds(3600)) |     let exp = iat | ||||||
|  |         .checked_add_signed(chrono::Duration::seconds(3600)) | ||||||
|         .ok_or_else(|| format_err!("Failed to calculate token expiry"))?; |         .ok_or_else(|| format_err!("Failed to calculate token expiry"))?; | ||||||
| 
 | 
 | ||||||
|     let header = Header { |     let header = Header { | ||||||
|  | @ -323,7 +330,9 @@ enum Payload { | ||||||
| /// text format.
 | /// text format.
 | ||||||
| fn message_to_payload(message: Option<String>) -> Payload { | fn message_to_payload(message: Option<String>) -> Payload { | ||||||
|     match message { |     match message { | ||||||
|         None => Payload::TextPayload { text_payload: "empty log entry".into() }, |         None => Payload::TextPayload { | ||||||
|  |             text_payload: "empty log entry".into(), | ||||||
|  |         }, | ||||||
|         Some(text_payload) => { |         Some(text_payload) => { | ||||||
|             // Attempt to deserialize the text payload as a generic
 |             // Attempt to deserialize the text payload as a generic
 | ||||||
|             // JSON value.
 |             // JSON value.
 | ||||||
|  | @ -333,7 +342,7 @@ fn message_to_payload(message: Option<String>) -> Payload { | ||||||
|                 // expect other types of JSON payload) and return it
 |                 // expect other types of JSON payload) and return it
 | ||||||
|                 // in that case.
 |                 // in that case.
 | ||||||
|                 if json_payload.is_object() { |                 if json_payload.is_object() { | ||||||
|                     return Payload::JsonPayload { json_payload } |                     return Payload::JsonPayload { json_payload }; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  | @ -450,9 +459,7 @@ impl From<JournalRecord> for LogEntry { | ||||||
|         // Journald uses syslogd's concept of priority. No idea if this is
 |         // Journald uses syslogd's concept of priority. No idea if this is
 | ||||||
|         // always present, but it's optional in the Stackdriver API, so we just
 |         // always present, but it's optional in the Stackdriver API, so we just
 | ||||||
|         // omit it if we can't find or parse it.
 |         // omit it if we can't find or parse it.
 | ||||||
|         let severity = record |         let severity = record.remove("PRIORITY").and_then(priority_to_severity); | ||||||
|             .remove("PRIORITY") |  | ||||||
|             .and_then(priority_to_severity); |  | ||||||
| 
 | 
 | ||||||
|         LogEntry { |         LogEntry { | ||||||
|             payload, |             payload, | ||||||
|  | @ -468,8 +475,7 @@ impl From<JournalRecord> for LogEntry { | ||||||
| 
 | 
 | ||||||
| /// Attempt to read from the journal. If no new entry is present,
 | /// Attempt to read from the journal. If no new entry is present,
 | ||||||
| /// await the next one up to the specified timeout.
 | /// await the next one up to the specified timeout.
 | ||||||
| fn receive_next_record(timeout: Duration, journal: &mut Journal) | fn receive_next_record(timeout: Duration, journal: &mut Journal) -> Result<Option<JournalRecord>> { | ||||||
|                        -> Result<Option<JournalRecord>> { |  | ||||||
|     let next_record = journal.next_record()?; |     let next_record = journal.next_record()?; | ||||||
|     if next_record.is_some() { |     if next_record.is_some() { | ||||||
|         return Ok(next_record); |         return Ok(next_record); | ||||||
|  | @ -525,11 +531,10 @@ fn persist_cursor(cursor: String) -> Result<()> { | ||||||
|     if cursor.is_empty() { |     if cursor.is_empty() { | ||||||
|         error!("Received empty journald cursor position, refusing to persist!"); |         error!("Received empty journald cursor position, refusing to persist!"); | ||||||
|         error!("Please report this message at https://github.com/tazjin/journaldriver/issues/2"); |         error!("Please report this message at https://github.com/tazjin/journaldriver/issues/2"); | ||||||
|         return Ok(()) |         return Ok(()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     let mut file = File::create(&*CURSOR_TMP_FILE) |     let mut file = File::create(&*CURSOR_TMP_FILE).context("Failed to create cursor file")?; | ||||||
|         .context("Failed to create cursor file")?; |  | ||||||
| 
 | 
 | ||||||
|     write!(file, "{}", cursor).context("Failed to write cursor file")?; |     write!(file, "{}", cursor).context("Failed to write cursor file")?; | ||||||
| 
 | 
 | ||||||
|  | @ -547,9 +552,7 @@ fn persist_cursor(cursor: String) -> Result<()> { | ||||||
| ///
 | ///
 | ||||||
| /// If flushing is successful the last cursor position will be
 | /// If flushing is successful the last cursor position will be
 | ||||||
| /// persisted to disk.
 | /// persisted to disk.
 | ||||||
| fn flush(token: &mut Token, | fn flush(token: &mut Token, entries: Vec<LogEntry>, cursor: String) -> Result<()> { | ||||||
|          entries: Vec<LogEntry>, |  | ||||||
|          cursor: String) -> Result<()> { |  | ||||||
|     if token.is_expired() { |     if token.is_expired() { | ||||||
|         debug!("Refreshing Google metadata access token"); |         debug!("Refreshing Google metadata access token"); | ||||||
|         let new_token = get_token()?; |         let new_token = get_token()?; | ||||||
|  | @ -598,7 +601,8 @@ fn write_entries(token: &Token, request: Value) -> Result<()> { | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } else { |     } else { | ||||||
|         let status = response.status_line().to_string(); |         let status = response.status_line().to_string(); | ||||||
|         let body = response.into_string() |         let body = response | ||||||
|  |             .into_string() | ||||||
|             .unwrap_or_else(|_| "no response body".into()); |             .unwrap_or_else(|_| "no response body".into()); | ||||||
|         bail!("Write failure: {} ({})", body, status) |         bail!("Write failure: {} ({})", body, status) | ||||||
|     } |     } | ||||||
|  | @ -624,10 +628,8 @@ fn initial_cursor() -> Result<JournalSeek> { | ||||||
|         Err(ref err) if err.kind() == ErrorKind::NotFound => { |         Err(ref err) if err.kind() == ErrorKind::NotFound => { | ||||||
|             info!("No previous cursor position, reading from journal tail"); |             info!("No previous cursor position, reading from journal tail"); | ||||||
|             Ok(JournalSeek::Tail) |             Ok(JournalSeek::Tail) | ||||||
|         }, |  | ||||||
|         Err(err) => { |  | ||||||
|             (Err(err).context("Could not read cursor position"))? |  | ||||||
|         } |         } | ||||||
|  |         Err(err) => (Err(err).context("Could not read cursor position"))?, | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -641,17 +643,17 @@ fn main () { | ||||||
|         process::exit(1); |         process::exit(1); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     let cursor_position_dir = CURSOR_FILE.parent() |     let cursor_position_dir = CURSOR_FILE | ||||||
|  |         .parent() | ||||||
|         .expect("Invalid cursor position file path"); |         .expect("Invalid cursor position file path"); | ||||||
| 
 | 
 | ||||||
|     fs::create_dir_all(cursor_position_dir) |     fs::create_dir_all(cursor_position_dir) | ||||||
|         .expect("Could not create directory to store cursor position in"); |         .expect("Could not create directory to store cursor position in"); | ||||||
| 
 | 
 | ||||||
|     let mut journal = Journal::open(JournalFiles::All, false, true) |     let mut journal = | ||||||
|         .expect("Failed to open systemd journal"); |         Journal::open(JournalFiles::All, false, true).expect("Failed to open systemd journal"); | ||||||
| 
 | 
 | ||||||
|     let seek_position = initial_cursor() |     let seek_position = initial_cursor().expect("Failed to determine initial cursor position"); | ||||||
|         .expect("Failed to determine initial cursor position"); |  | ||||||
| 
 | 
 | ||||||
|     match journal.seek(seek_position) { |     match journal.seek(seek_position) { | ||||||
|         Ok(cursor) => info!("Opened journal at cursor '{}'", cursor), |         Ok(cursor) => info!("Opened journal at cursor '{}'", cursor), | ||||||
|  |  | ||||||
|  | @ -15,7 +15,10 @@ fn test_text_entry_serialization() { | ||||||
|     let expected = "{\"labels\":null,\"textPayload\":\"test entry\"}"; |     let expected = "{\"labels\":null,\"textPayload\":\"test entry\"}"; | ||||||
|     let result = to_string(&entry).expect("serialization failed"); |     let result = to_string(&entry).expect("serialization failed"); | ||||||
| 
 | 
 | ||||||
|     assert_eq!(expected, result, "Plain text payload should serialize correctly") |     assert_eq!( | ||||||
|  |         expected, result, | ||||||
|  |         "Plain text payload should serialize correctly" | ||||||
|  |     ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | #[test] | ||||||
|  | @ -26,7 +29,7 @@ fn test_json_entry_serialization() { | ||||||
|         payload: Payload::JsonPayload { |         payload: Payload::JsonPayload { | ||||||
|             json_payload: json!({ |             json_payload: json!({ | ||||||
|                 "message": "JSON test" |                 "message": "JSON test" | ||||||
|             }) |             }), | ||||||
|         }, |         }, | ||||||
|         severity: None, |         severity: None, | ||||||
|     }; |     }; | ||||||
|  | @ -45,7 +48,10 @@ fn test_plain_text_payload() { | ||||||
|         text_payload: "plain text payload".into(), |         text_payload: "plain text payload".into(), | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     assert_eq!(expected, payload, "Plain text payload should be detected correctly"); |     assert_eq!( | ||||||
|  |         expected, payload, | ||||||
|  |         "Plain text payload should be detected correctly" | ||||||
|  |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | #[test] | ||||||
|  | @ -55,7 +61,10 @@ fn test_empty_payload() { | ||||||
|         text_payload: "empty log entry".into(), |         text_payload: "empty log entry".into(), | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     assert_eq!(expected, payload, "Empty payload should be handled correctly"); |     assert_eq!( | ||||||
|  |         expected, payload, | ||||||
|  |         "Empty payload should be handled correctly" | ||||||
|  |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | #[test] | ||||||
|  | @ -66,10 +75,13 @@ fn test_json_payload() { | ||||||
|         json_payload: json!({ |         json_payload: json!({ | ||||||
|             "someKey": "someValue", |             "someKey": "someValue", | ||||||
|             "otherKey": 42 |             "otherKey": 42 | ||||||
|         }) |         }), | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     assert_eq!(expected, payload, "JSON payload should be detected correctly"); |     assert_eq!( | ||||||
|  |         expected, payload, | ||||||
|  |         "JSON payload should be detected correctly" | ||||||
|  |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | #[test] | ||||||
|  | @ -82,14 +94,16 @@ fn test_json_no_object() { | ||||||
|         text_payload: "42".into(), |         text_payload: "42".into(), | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     assert_eq!(expected, payload, "Non-object JSON payload should be plain text"); |     assert_eq!( | ||||||
|  |         expected, payload, | ||||||
|  |         "Non-object JSON payload should be plain text" | ||||||
|  |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | #[test] | ||||||
| fn test_parse_microseconds() { | fn test_parse_microseconds() { | ||||||
|     let input: String = "1529175149291187".into(); |     let input: String = "1529175149291187".into(); | ||||||
|     let expected: DateTime<Utc> = "2018-06-16T18:52:29.291187Z" |     let expected: DateTime<Utc> = "2018-06-16T18:52:29.291187Z".to_string().parse().unwrap(); | ||||||
|         .to_string().parse().unwrap(); |  | ||||||
| 
 | 
 | ||||||
|     assert_eq!(Some(expected), parse_microseconds(input)); |     assert_eq!(Some(expected), parse_microseconds(input)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,36 +1,38 @@ | ||||||
| extern crate clap; | extern crate clap; | ||||||
| extern crate posix_mq; |  | ||||||
| extern crate libc; | extern crate libc; | ||||||
| extern crate nix; | extern crate nix; | ||||||
|  | extern crate posix_mq; | ||||||
| 
 | 
 | ||||||
| use clap::{App, SubCommand, Arg, ArgMatches, AppSettings}; | use clap::{App, AppSettings, Arg, ArgMatches, SubCommand}; | ||||||
| use posix_mq::{Name, Queue, Message}; | use posix_mq::{Message, Name, Queue}; | ||||||
| use std::fs::{read_dir, File}; | use std::fs::{read_dir, File}; | ||||||
| use std::io::{self, Read, Write}; | use std::io::{self, Read, Write}; | ||||||
| use std::process::exit; | use std::process::exit; | ||||||
| 
 | 
 | ||||||
| fn run_ls() { | fn run_ls() { | ||||||
|     let mqueues = read_dir("/dev/mqueue") |     let mqueues = read_dir("/dev/mqueue").expect("Could not read message queues"); | ||||||
|         .expect("Could not read message queues"); |  | ||||||
| 
 | 
 | ||||||
|     for queue in mqueues { |     for queue in mqueues { | ||||||
|         let path = queue.unwrap().path(); |         let path = queue.unwrap().path(); | ||||||
|         let status = { |         let status = { | ||||||
|             let mut file = File::open(&path) |             let mut file = File::open(&path).expect("Could not open queue file"); | ||||||
|                 .expect("Could not open queue file"); |  | ||||||
| 
 | 
 | ||||||
|             let mut content = String::new(); |             let mut content = String::new(); | ||||||
|             file.read_to_string(&mut content).expect("Could not read queue file"); |             file.read_to_string(&mut content) | ||||||
|  |                 .expect("Could not read queue file"); | ||||||
| 
 | 
 | ||||||
|             content |             content | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         let queue_name = path.components().last().unwrap() |         let queue_name = path | ||||||
|  |             .components() | ||||||
|  |             .last() | ||||||
|  |             .unwrap() | ||||||
|             .as_os_str() |             .as_os_str() | ||||||
|             .to_string_lossy(); |             .to_string_lossy(); | ||||||
| 
 | 
 | ||||||
|         println!("/{}: {}", queue_name, status) |         println!("/{}: {}", queue_name, status) | ||||||
|     }; |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn run_inspect(queue_name: &str) { | fn run_inspect(queue_name: &str) { | ||||||
|  | @ -47,8 +49,7 @@ fn run_create(cmd: &ArgMatches) { | ||||||
|         set_rlimit(rlimit.parse().expect("Invalid rlimit value")); |         set_rlimit(rlimit.parse().expect("Invalid rlimit value")); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     let name = Name::new(cmd.value_of("queue").unwrap()) |     let name = Name::new(cmd.value_of("queue").unwrap()).expect("Invalid queue name"); | ||||||
|         .expect("Invalid queue name"); |  | ||||||
| 
 | 
 | ||||||
|     let max_pending: i64 = cmd.value_of("max-pending").unwrap().parse().unwrap(); |     let max_pending: i64 = cmd.value_of("max-pending").unwrap().parse().unwrap(); | ||||||
|     let max_size: i64 = cmd.value_of("max-size").unwrap().parse().unwrap(); |     let max_size: i64 = cmd.value_of("max-size").unwrap().parse().unwrap(); | ||||||
|  | @ -60,7 +61,7 @@ fn run_create(cmd: &ArgMatches) { | ||||||
|         Err(e) => { |         Err(e) => { | ||||||
|             writeln!(io::stderr(), "Could not create queue: {}", e).ok(); |             writeln!(io::stderr(), "Could not create queue: {}", e).ok(); | ||||||
|             exit(1); |             exit(1); | ||||||
|         }, |         } | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -120,7 +121,12 @@ fn run_rlimit() { | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     if errno != 0 { |     if errno != 0 { | ||||||
|         writeln!(io::stderr(), "Could not get message queue rlimit: {}", errno).ok(); |         writeln!( | ||||||
|  |             io::stderr(), | ||||||
|  |             "Could not get message queue rlimit: {}", | ||||||
|  |             errno | ||||||
|  |         ) | ||||||
|  |         .ok(); | ||||||
|     } else { |     } else { | ||||||
|         println!("Message queue rlimit:"); |         println!("Message queue rlimit:"); | ||||||
|         println!("Current limit: {}", rlimit.rlim_cur); |         println!("Current limit: {}", rlimit.rlim_cur); | ||||||
|  | @ -170,16 +176,20 @@ fn main() { | ||||||
|         .about("Create a new queue") |         .about("Create a new queue") | ||||||
|         .arg(&queue_arg) |         .arg(&queue_arg) | ||||||
|         .arg(&rlimit_arg) |         .arg(&rlimit_arg) | ||||||
|         .arg(Arg::with_name("max-size") |         .arg( | ||||||
|  |             Arg::with_name("max-size") | ||||||
|                 .help("maximum message size (in kB)") |                 .help("maximum message size (in kB)") | ||||||
|                 .long("max-size") |                 .long("max-size") | ||||||
|                 .required(true) |                 .required(true) | ||||||
|             .takes_value(true)) |                 .takes_value(true), | ||||||
|         .arg(Arg::with_name("max-pending") |         ) | ||||||
|  |         .arg( | ||||||
|  |             Arg::with_name("max-pending") | ||||||
|                 .help("maximum # of pending messages") |                 .help("maximum # of pending messages") | ||||||
|                 .long("max-pending") |                 .long("max-pending") | ||||||
|                 .required(true) |                 .required(true) | ||||||
|             .takes_value(true)); |                 .takes_value(true), | ||||||
|  |         ); | ||||||
| 
 | 
 | ||||||
|     let receive = SubCommand::with_name("receive") |     let receive = SubCommand::with_name("receive") | ||||||
|         .about("Receive a message from a queue") |         .about("Receive a message from a queue") | ||||||
|  | @ -188,9 +198,11 @@ fn main() { | ||||||
|     let send = SubCommand::with_name("send") |     let send = SubCommand::with_name("send") | ||||||
|         .about("Send a message to a queue") |         .about("Send a message to a queue") | ||||||
|         .arg(&queue_arg) |         .arg(&queue_arg) | ||||||
|         .arg(Arg::with_name("message") |         .arg( | ||||||
|  |             Arg::with_name("message") | ||||||
|                 .help("the message to send") |                 .help("the message to send") | ||||||
|             .required(true)); |                 .required(true), | ||||||
|  |         ); | ||||||
| 
 | 
 | ||||||
|     let rlimit = SubCommand::with_name("rlimit") |     let rlimit = SubCommand::with_name("rlimit") | ||||||
|         .about("Get the message queue rlimit") |         .about("Get the message queue rlimit") | ||||||
|  | @ -215,7 +227,7 @@ fn main() { | ||||||
|         ("receive", Some(cmd)) => run_receive(cmd.value_of("queue").unwrap()), |         ("receive", Some(cmd)) => run_receive(cmd.value_of("queue").unwrap()), | ||||||
|         ("send", Some(cmd)) => run_send( |         ("send", Some(cmd)) => run_send( | ||||||
|             cmd.value_of("queue").unwrap(), |             cmd.value_of("queue").unwrap(), | ||||||
|             cmd.value_of("message").unwrap() |             cmd.value_of("message").unwrap(), | ||||||
|         ), |         ), | ||||||
|         ("rlimit", _) => run_rlimit(), |         ("rlimit", _) => run_rlimit(), | ||||||
|         _ => unimplemented!(), |         _ => unimplemented!(), | ||||||
|  |  | ||||||
|  | @ -1,8 +1,5 @@ | ||||||
| use nix; | use nix; | ||||||
| use std::error; | use std::{error, fmt, io, num}; | ||||||
| use std::fmt; |  | ||||||
| use std::io; |  | ||||||
| use std::num; |  | ||||||
| 
 | 
 | ||||||
| /// This module implements a simple error type to match the errors that can be thrown from the C
 | /// This module implements a simple error type to match the errors that can be thrown from the C
 | ||||||
| /// functions as well as some extra errors resulting from internal validations.
 | /// functions as well as some extra errors resulting from internal validations.
 | ||||||
|  |  | ||||||
|  | @ -4,8 +4,7 @@ use super::*; | ||||||
| fn test_open_delete() { | fn test_open_delete() { | ||||||
|     // Simple test with default queue settings
 |     // Simple test with default queue settings
 | ||||||
|     let name = Name::new("/test-queue").unwrap(); |     let name = Name::new("/test-queue").unwrap(); | ||||||
|     let queue = Queue::open_or_create(name) |     let queue = Queue::open_or_create(name).expect("Opening queue failed"); | ||||||
|         .expect("Opening queue failed"); |  | ||||||
| 
 | 
 | ||||||
|     let message = Message { |     let message = Message { | ||||||
|         data: "test-message".as_bytes().to_vec(), |         data: "test-message".as_bytes().to_vec(), | ||||||
|  |  | ||||||
|  | @ -28,14 +28,19 @@ fn main() { | ||||||
| 
 | 
 | ||||||
|     // Otherwise ask Nix to build it and inject the result.
 |     // Otherwise ask Nix to build it and inject the result.
 | ||||||
|     let output = Command::new("nix-build") |     let output = Command::new("nix-build") | ||||||
|         .arg("-A").arg("third_party.bat_syntaxes") |         .arg("-A") | ||||||
|  |         .arg("third_party.bat_syntaxes") | ||||||
|         // ... assuming cheddar is at //tools/cheddar ...
 |         // ... assuming cheddar is at //tools/cheddar ...
 | ||||||
|         .arg("../..") |         .arg("../..") | ||||||
|         .output() |         .output() | ||||||
|         .expect(ERROR_MESSAGE); |         .expect(ERROR_MESSAGE); | ||||||
| 
 | 
 | ||||||
|     if !output.status.success() { |     if !output.status.success() { | ||||||
|         eprintln!("{}\nNix output: {}", ERROR_MESSAGE, String::from_utf8_lossy(&output.stderr)); |         eprintln!( | ||||||
|  |             "{}\nNix output: {}", | ||||||
|  |             ERROR_MESSAGE, | ||||||
|  |             String::from_utf8_lossy(&output.stderr) | ||||||
|  |         ); | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -5,14 +5,13 @@ | ||||||
| //! 2. As a long-running HTTP server that handles rendering requests
 | //! 2. As a long-running HTTP server that handles rendering requests
 | ||||||
| //!    (matching the SourceGraph protocol).
 | //!    (matching the SourceGraph protocol).
 | ||||||
| use clap::{App, Arg}; | use clap::{App, Arg}; | ||||||
| use rouille::Response; | use rouille::{router, try_or_400, Response}; | ||||||
| use rouille::{router, try_or_400}; |  | ||||||
| use serde::Deserialize; | use serde::Deserialize; | ||||||
| use serde_json::json; | use serde_json::json; | ||||||
| use std::collections::HashMap; | use std::collections::HashMap; | ||||||
| use std::io; | use std::io; | ||||||
| 
 | 
 | ||||||
| use cheddar::{THEMES, format_code, format_markdown}; | use cheddar::{format_code, format_markdown, THEMES}; | ||||||
| 
 | 
 | ||||||
| // Server endpoint for rendering the syntax of source code. This
 | // Server endpoint for rendering the syntax of source code. This
 | ||||||
| // replaces the 'syntect_server' component of Sourcegraph.
 | // replaces the 'syntect_server' component of Sourcegraph.
 | ||||||
|  |  | ||||||
|  | @ -8,12 +8,10 @@ use lazy_static::lazy_static; | ||||||
| use regex::Regex; | use regex::Regex; | ||||||
| use std::cell::RefCell; | use std::cell::RefCell; | ||||||
| use std::collections::HashMap; | use std::collections::HashMap; | ||||||
| use std::env; |  | ||||||
| use std::ffi::OsStr; | use std::ffi::OsStr; | ||||||
| use std::io; | use std::io::{BufRead, Write}; | ||||||
| use std::io::BufRead; |  | ||||||
| use std::io::Write; |  | ||||||
| use std::path::Path; | use std::path::Path; | ||||||
|  | use std::{env, io}; | ||||||
| use syntect::dumps::from_binary; | use syntect::dumps::from_binary; | ||||||
| use syntect::easy::HighlightLines; | use syntect::easy::HighlightLines; | ||||||
| use syntect::highlighting::{Theme, ThemeSet}; | use syntect::highlighting::{Theme, ThemeSet}; | ||||||
|  |  | ||||||
|  | @ -14,42 +14,89 @@ use std::io::Write; | ||||||
| fn main() { | fn main() { | ||||||
|     let mut args = std::env::args_os(); |     let mut args = std::env::args_os(); | ||||||
|     let file = args.nth(1).expect("security advisory md file is $1"); |     let file = args.nth(1).expect("security advisory md file is $1"); | ||||||
|     let crate_version = |     let crate_version = args | ||||||
|         args.nth(0).expect("crate version is $2") |         .nth(0) | ||||||
|         .into_string().expect("crate version string not utf8") |         .expect("crate version is $2") | ||||||
|         ; |         .into_string() | ||||||
|     let crate_version = semver::Version::parse(&crate_version).expect(&format!("this is not a semver version: {}", &crate_version)); |         .expect("crate version string not utf8"); | ||||||
|  |     let crate_version = semver::Version::parse(&crate_version) | ||||||
|  |         .expect(&format!("this is not a semver version: {}", &crate_version)); | ||||||
|     let filename = file.to_string_lossy(); |     let filename = file.to_string_lossy(); | ||||||
| 
 | 
 | ||||||
|     let content = std::fs::read(&file).expect(&format!("could not read {}", filename)); |     let content = std::fs::read(&file).expect(&format!("could not read {}", filename)); | ||||||
|     let content = |     let content = std::str::from_utf8(&content) | ||||||
|         std::str::from_utf8(&content).expect(&format!("file {} was not encoded as utf-8", filename)); |         .expect(&format!("file {} was not encoded as utf-8", filename)); | ||||||
|     let content = content.trim_start(); |     let content = content.trim_start(); | ||||||
| 
 | 
 | ||||||
|     let toml_start = content |     let toml_start = content | ||||||
|         .strip_prefix("```toml").expect(&format!("file did not start with ```toml: {}", filename)); |         .strip_prefix("```toml") | ||||||
|     let toml_end_index = toml_start.find("```").expect(&format!("the toml section did not end, no `` found: {}", filename)); |         .expect(&format!("file did not start with ```toml: {}", filename)); | ||||||
|  |     let toml_end_index = toml_start.find("```").expect(&format!( | ||||||
|  |         "the toml section did not end, no `` found: {}", | ||||||
|  |         filename | ||||||
|  |     )); | ||||||
|     let toml = &toml_start[..toml_end_index]; |     let toml = &toml_start[..toml_end_index]; | ||||||
|     let toml : toml::Value = toml::de::from_slice(toml.as_bytes()).expect(&format!("could not parse toml: {}", filename)); |     let toml: toml::Value = toml::de::from_slice(toml.as_bytes()) | ||||||
|  |         .expect(&format!("could not parse toml: {}", filename)); | ||||||
| 
 | 
 | ||||||
|     let versions = toml |     let versions = toml | ||||||
|         .as_table().expect(&format!("the toml is not a table: {}", filename)) |         .as_table() | ||||||
|         .get("versions").expect(&format!("the toml does not contain the versions field: {}", filename)) |         .expect(&format!("the toml is not a table: {}", filename)) | ||||||
|         .as_table().expect(&format!("the toml versions field must be a table: {}", filename)); |         .get("versions") | ||||||
|  |         .expect(&format!( | ||||||
|  |             "the toml does not contain the versions field: {}", | ||||||
|  |             filename | ||||||
|  |         )) | ||||||
|  |         .as_table() | ||||||
|  |         .expect(&format!( | ||||||
|  |             "the toml versions field must be a table: {}", | ||||||
|  |             filename | ||||||
|  |         )); | ||||||
| 
 | 
 | ||||||
|     let unaffected = match versions.get("unaffected") { |     let unaffected = match versions.get("unaffected") { | ||||||
|         Some(u) => u |         Some(u) => u | ||||||
|             .as_array().expect(&format!("the toml versions.unaffected field must be a list of semvers: {}", filename)) |             .as_array() | ||||||
|  |             .expect(&format!( | ||||||
|  |                 "the toml versions.unaffected field must be a list of semvers: {}", | ||||||
|  |                 filename | ||||||
|  |             )) | ||||||
|             .iter() |             .iter() | ||||||
|             .map(|v| semver::VersionReq::parse(v.as_str().expect(&format!("the version field {} is not a string", v))).expect(&format!("the version field {} is not a valid semver VersionReq", v))) |             .map(|v| { | ||||||
|  |                 semver::VersionReq::parse( | ||||||
|  |                     v.as_str() | ||||||
|  |                         .expect(&format!("the version field {} is not a string", v)), | ||||||
|  |                 ) | ||||||
|  |                 .expect(&format!( | ||||||
|  |                     "the version field {} is not a valid semver VersionReq", | ||||||
|  |                     v | ||||||
|  |                 )) | ||||||
|  |             }) | ||||||
|             .collect(), |             .collect(), | ||||||
|         None => vec![] |         None => vec![], | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     let mut patched : Vec<semver::VersionReq> = versions.get("patched").expect(&format!("the toml versions.patched field must exist: {}", filename)) |     let mut patched: Vec<semver::VersionReq> = versions | ||||||
|         .as_array().expect(&format!("the toml versions.patched field must be a list of semvers: {}", filename)) |         .get("patched") | ||||||
|  |         .expect(&format!( | ||||||
|  |             "the toml versions.patched field must exist: {}", | ||||||
|  |             filename | ||||||
|  |         )) | ||||||
|  |         .as_array() | ||||||
|  |         .expect(&format!( | ||||||
|  |             "the toml versions.patched field must be a list of semvers: {}", | ||||||
|  |             filename | ||||||
|  |         )) | ||||||
|         .iter() |         .iter() | ||||||
|         .map(|v| semver::VersionReq::parse(v.as_str().expect(&format!("the version field {} is not a string", v))).expect(&format!("the version field {} is not a valid semver VersionReq", v))) |         .map(|v| { | ||||||
|  |             semver::VersionReq::parse( | ||||||
|  |                 v.as_str() | ||||||
|  |                     .expect(&format!("the version field {} is not a string", v)), | ||||||
|  |             ) | ||||||
|  |             .expect(&format!( | ||||||
|  |                 "the version field {} is not a valid semver VersionReq", | ||||||
|  |                 v | ||||||
|  |             )) | ||||||
|  |         }) | ||||||
|         .collect(); |         .collect(); | ||||||
| 
 | 
 | ||||||
|     patched.extend_from_slice(&unaffected[..]); |     patched.extend_from_slice(&unaffected[..]); | ||||||
|  | @ -59,9 +106,14 @@ fn main() { | ||||||
|         std::process::exit(0); |         std::process::exit(0); | ||||||
|     } else { |     } else { | ||||||
|         if std::env::var_os("PRINT_ADVISORY").is_some() { |         if std::env::var_os("PRINT_ADVISORY").is_some() { | ||||||
|             write!(std::io::stderr(), "Advisory {} matched!\n{}\n", filename, content).unwrap(); |             write!( | ||||||
|  |                 std::io::stderr(), | ||||||
|  |                 "Advisory {} matched!\n{}\n", | ||||||
|  |                 filename, | ||||||
|  |                 content | ||||||
|  |             ) | ||||||
|  |             .unwrap(); | ||||||
|         } |         } | ||||||
|         std::process::exit(1); |         std::process::exit(1); | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -25,7 +25,8 @@ pub enum NixResult { | ||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod integration_tests { | mod integration_tests { | ||||||
|     use std::{collections::VecDeque, io::Write}; |     use std::collections::VecDeque; | ||||||
|  |     use std::io::Write; | ||||||
| 
 | 
 | ||||||
|     use super::*; |     use super::*; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,13 +1,16 @@ | ||||||
| use std::os::unix::process::CommandExt; |  | ||||||
| use std::ffi::OsStr; | use std::ffi::OsStr; | ||||||
| use std::os::unix::ffi::{OsStringExt, OsStrExt}; | use std::os::unix::ffi::{OsStrExt, OsStringExt}; | ||||||
|  | use std::os::unix::process::CommandExt; | ||||||
| 
 | 
 | ||||||
| pub fn no_args(current_prog_name: &str) -> () { | pub fn no_args(current_prog_name: &str) -> () { | ||||||
|     let mut args = std::env::args_os(); |     let mut args = std::env::args_os(); | ||||||
|     // remove argv[0]
 |     // remove argv[0]
 | ||||||
|     let _ = args.nth(0); |     let _ = args.nth(0); | ||||||
|     if args.len() > 0 { |     if args.len() > 0 { | ||||||
|         die_user_error(current_prog_name, format!("Expected no arguments, got {:?}", args.collect::<Vec<_>>())) |         die_user_error( | ||||||
|  |             current_prog_name, | ||||||
|  |             format!("Expected no arguments, got {:?}", args.collect::<Vec<_>>()), | ||||||
|  |         ) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -16,12 +19,23 @@ pub fn args(current_prog_name: &str, no_of_positional_args: usize) -> Vec<Vec<u8 | ||||||
|     // remove argv[0]
 |     // remove argv[0]
 | ||||||
|     let _ = args.nth(0); |     let _ = args.nth(0); | ||||||
|     if args.len() != no_of_positional_args { |     if args.len() != no_of_positional_args { | ||||||
|         die_user_error(current_prog_name, format!("Expected {} arguments, got {}, namely {:?}", no_of_positional_args, args.len(), args.collect::<Vec<_>>())) |         die_user_error( | ||||||
|  |             current_prog_name, | ||||||
|  |             format!( | ||||||
|  |                 "Expected {} arguments, got {}, namely {:?}", | ||||||
|  |                 no_of_positional_args, | ||||||
|  |                 args.len(), | ||||||
|  |                 args.collect::<Vec<_>>() | ||||||
|  |             ), | ||||||
|  |         ) | ||||||
|     } |     } | ||||||
|     args.map(|arg| arg.into_vec()).collect() |     args.map(|arg| arg.into_vec()).collect() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub fn args_for_exec(current_prog_name: &str, no_of_positional_args: usize) -> (Vec<Vec<u8>>, Vec<Vec<u8>>) { | pub fn args_for_exec( | ||||||
|  |     current_prog_name: &str, | ||||||
|  |     no_of_positional_args: usize, | ||||||
|  | ) -> (Vec<Vec<u8>>, Vec<Vec<u8>>) { | ||||||
|     let mut args = std::env::args_os(); |     let mut args = std::env::args_os(); | ||||||
|     // remove argv[0]
 |     // remove argv[0]
 | ||||||
|     let _ = args.nth(0); |     let _ = args.nth(0); | ||||||
|  | @ -29,17 +43,21 @@ pub fn args_for_exec(current_prog_name: &str, no_of_positional_args: usize) -> ( | ||||||
|     let mut pos_args = vec![]; |     let mut pos_args = vec![]; | ||||||
|     // get positional args
 |     // get positional args
 | ||||||
|     for i in 1..no_of_positional_args + 1 { |     for i in 1..no_of_positional_args + 1 { | ||||||
|             pos_args.push( |         pos_args.push(args.nth(0).expect(&format!( | ||||||
|                 args.nth(0).expect( |             "{}: expects {} positional args, only got {}", | ||||||
|                     &format!("{}: expects {} positional args, only got {}", current_prog_name, no_of_positional_args, i)) |             current_prog_name, no_of_positional_args, i | ||||||
|             ); |         ))); | ||||||
|     } |     } | ||||||
|     // prog... is the rest of the iterator
 |     // prog... is the rest of the iterator
 | ||||||
|     let prog: Vec<Vec<u8>> = args.collect(); |     let prog: Vec<Vec<u8>> = args.collect(); | ||||||
|     (pos_args, prog) |     (pos_args, prog) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub fn exec_into_args<'a, 'b, Args, Arg, Env, Key, Val>(current_prog_name: &str, args: Args, env_additions: Env) -> ! | pub fn exec_into_args<'a, 'b, Args, Arg, Env, Key, Val>( | ||||||
|  |     current_prog_name: &str, | ||||||
|  |     args: Args, | ||||||
|  |     env_additions: Env, | ||||||
|  | ) -> ! | ||||||
| where | where | ||||||
|     Args: IntoIterator<Item = Arg>, |     Args: IntoIterator<Item = Arg>, | ||||||
|     Arg: AsRef<[u8]>, |     Arg: AsRef<[u8]>, | ||||||
|  | @ -50,18 +68,30 @@ pub fn exec_into_args<'a, 'b, Args, Arg, Env, Key, Val>(current_prog_name: &str, | ||||||
|     // TODO: is this possible without collecting into a Vec first, just leaving it an IntoIterator?
 |     // TODO: is this possible without collecting into a Vec first, just leaving it an IntoIterator?
 | ||||||
|     let args = args.into_iter().collect::<Vec<Arg>>(); |     let args = args.into_iter().collect::<Vec<Arg>>(); | ||||||
|     let mut args = args.iter().map(|v| OsStr::from_bytes(v.as_ref())); |     let mut args = args.iter().map(|v| OsStr::from_bytes(v.as_ref())); | ||||||
|     let prog = args.nth(0).expect(&format!("{}: first argument must be an executable", current_prog_name)); |     let prog = args.nth(0).expect(&format!( | ||||||
|  |         "{}: first argument must be an executable", | ||||||
|  |         current_prog_name | ||||||
|  |     )); | ||||||
|     // TODO: same here
 |     // TODO: same here
 | ||||||
|     let env = env_additions.into_iter().collect::<Vec<(Key, Val)>>(); |     let env = env_additions.into_iter().collect::<Vec<(Key, Val)>>(); | ||||||
|     let env = env.iter().map(|(k,v)| (OsStr::from_bytes(k.as_ref()), OsStr::from_bytes(v.as_ref()))); |     let env = env | ||||||
|  |         .iter() | ||||||
|  |         .map(|(k, v)| (OsStr::from_bytes(k.as_ref()), OsStr::from_bytes(v.as_ref()))); | ||||||
|     let err = std::process::Command::new(prog).args(args).envs(env).exec(); |     let err = std::process::Command::new(prog).args(args).envs(env).exec(); | ||||||
|     die_missing_executable(current_prog_name, format!("exec failed: {}, while trying to execing into {:?}", err, prog)); |     die_missing_executable( | ||||||
|  |         current_prog_name, | ||||||
|  |         format!( | ||||||
|  |             "exec failed: {}, while trying to execing into {:?}", | ||||||
|  |             err, prog | ||||||
|  |         ), | ||||||
|  |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Exit 1 to signify a generic expected error
 | /// Exit 1 to signify a generic expected error
 | ||||||
| /// (e.g. something that sometimes just goes wrong, like a nix build).
 | /// (e.g. something that sometimes just goes wrong, like a nix build).
 | ||||||
| pub fn die_expected_error<S>(current_prog_name: &str, msg: S) -> ! | pub fn die_expected_error<S>(current_prog_name: &str, msg: S) -> ! | ||||||
| where S: AsRef<str> | where | ||||||
|  |     S: AsRef<str>, | ||||||
| { | { | ||||||
|     die_with(1, current_prog_name, msg) |     die_with(1, current_prog_name, msg) | ||||||
| } | } | ||||||
|  | @ -70,7 +100,8 @@ where S: AsRef<str> | ||||||
| /// This is a permanent error, if the program is executed the same way
 | /// This is a permanent error, if the program is executed the same way
 | ||||||
| /// it should crash with 100 again.
 | /// it should crash with 100 again.
 | ||||||
| pub fn die_user_error<S>(current_prog_name: &str, msg: S) -> ! | pub fn die_user_error<S>(current_prog_name: &str, msg: S) -> ! | ||||||
| where S: AsRef<str> | where | ||||||
|  |     S: AsRef<str>, | ||||||
| { | { | ||||||
|     die_with(100, current_prog_name, msg) |     die_with(100, current_prog_name, msg) | ||||||
| } | } | ||||||
|  | @ -78,14 +109,16 @@ where S: AsRef<str> | ||||||
| /// Exit 101 to signify an unexpected crash (failing assertion or panic).
 | /// Exit 101 to signify an unexpected crash (failing assertion or panic).
 | ||||||
| /// This is the same exit code that `panic!()` emits.
 | /// This is the same exit code that `panic!()` emits.
 | ||||||
| pub fn die_panic<S>(current_prog_name: &str, msg: S) -> ! | pub fn die_panic<S>(current_prog_name: &str, msg: S) -> ! | ||||||
| where S: AsRef<str> | where | ||||||
|  |     S: AsRef<str>, | ||||||
| { | { | ||||||
|     die_with(101, current_prog_name, msg) |     die_with(101, current_prog_name, msg) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Exit 111 to signify a temporary error (such as resource exhaustion)
 | /// Exit 111 to signify a temporary error (such as resource exhaustion)
 | ||||||
| pub fn die_temporary<S>(current_prog_name: &str, msg: S) -> ! | pub fn die_temporary<S>(current_prog_name: &str, msg: S) -> ! | ||||||
| where S: AsRef<str> | where | ||||||
|  |     S: AsRef<str>, | ||||||
| { | { | ||||||
|     die_with(111, current_prog_name, msg) |     die_with(111, current_prog_name, msg) | ||||||
| } | } | ||||||
|  | @ -93,20 +126,23 @@ where S: AsRef<str> | ||||||
| /// Exit 126 to signify an environment problem
 | /// Exit 126 to signify an environment problem
 | ||||||
| /// (the user has set up stuff incorrectly so the program cannot work)
 | /// (the user has set up stuff incorrectly so the program cannot work)
 | ||||||
| pub fn die_environment_problem<S>(current_prog_name: &str, msg: S) -> ! | pub fn die_environment_problem<S>(current_prog_name: &str, msg: S) -> ! | ||||||
| where S: AsRef<str> | where | ||||||
|  |     S: AsRef<str>, | ||||||
| { | { | ||||||
|     die_with(126, current_prog_name, msg) |     die_with(126, current_prog_name, msg) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Exit 127 to signify a missing executable.
 | /// Exit 127 to signify a missing executable.
 | ||||||
| pub fn die_missing_executable<S>(current_prog_name: &str, msg: S) -> ! | pub fn die_missing_executable<S>(current_prog_name: &str, msg: S) -> ! | ||||||
| where S: AsRef<str> | where | ||||||
|  |     S: AsRef<str>, | ||||||
| { | { | ||||||
|     die_with(127, current_prog_name, msg) |     die_with(127, current_prog_name, msg) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn die_with<S>(status: i32, current_prog_name: &str, msg: S) -> ! | fn die_with<S>(status: i32, current_prog_name: &str, msg: S) -> ! | ||||||
|     where S: AsRef<str> | where | ||||||
|  |     S: AsRef<str>, | ||||||
| { | { | ||||||
|     eprintln!("{}: {}", current_prog_name, msg.as_ref()); |     eprintln!("{}: {}", current_prog_name, msg.as_ref()); | ||||||
|     std::process::exit(status) |     std::process::exit(status) | ||||||
|  |  | ||||||
|  | @ -16,8 +16,12 @@ fn main() { | ||||||
|             .bare(true) |             .bare(true) | ||||||
|             .mkpath(true) |             .mkpath(true) | ||||||
|             .description("git-db database") |             .description("git-db database") | ||||||
|             .initial_head(DEFAULT_BRANCH) |             .initial_head(DEFAULT_BRANCH), | ||||||
|     ).expect(&format!("unable to create or open bare git repo at {}", &git_db.display())); |     ) | ||||||
|  |     .expect(&format!( | ||||||
|  |         "unable to create or open bare git repo at {}", | ||||||
|  |         &git_db.display() | ||||||
|  |     )); | ||||||
| 
 | 
 | ||||||
|     let mut index = repo.index().expect("cannot get the git index file"); |     let mut index = repo.index().expect("cannot get the git index file"); | ||||||
|     eprintln!("{:#?}", index.version()); |     eprintln!("{:#?}", index.version()); | ||||||
|  | @ -34,7 +38,8 @@ fn main() { | ||||||
| 
 | 
 | ||||||
|     let data = "hi, it’s me".as_bytes(); |     let data = "hi, it’s me".as_bytes(); | ||||||
| 
 | 
 | ||||||
|     index.add_frombuffer( |     index | ||||||
|  |         .add_frombuffer( | ||||||
|             &git2::IndexEntry { |             &git2::IndexEntry { | ||||||
|             mtime: now_git_time, |             mtime: now_git_time, | ||||||
|             ctime: now_git_time, |             ctime: now_git_time, | ||||||
|  | @ -50,25 +55,26 @@ fn main() { | ||||||
|             flags_extended: 0, |             flags_extended: 0, | ||||||
|             path: "hi.txt".as_bytes().to_owned(), |             path: "hi.txt".as_bytes().to_owned(), | ||||||
|         }, |         }, | ||||||
|         data |             data, | ||||||
|     ).expect("could not add data to index"); |         ) | ||||||
|  |         .expect("could not add data to index"); | ||||||
| 
 | 
 | ||||||
|     let oid = index.write_tree().expect("could not write index tree"); |     let oid = index.write_tree().expect("could not write index tree"); | ||||||
| 
 | 
 | ||||||
|     let to_add_tree = repo.find_tree(oid) |     let to_add_tree = repo | ||||||
|  |         .find_tree(oid) | ||||||
|         .expect("we just created this tree, where did it go?"); |         .expect("we just created this tree, where did it go?"); | ||||||
| 
 | 
 | ||||||
|     let parent_commits = match repo.find_reference(DEFAULT_BRANCH) { |     let parent_commits = match repo.find_reference(DEFAULT_BRANCH) { | ||||||
|         Ok(ref_) => vec![ |         Ok(ref_) => vec![ref_.peel_to_commit().expect(&format!( | ||||||
|             ref_ |             "reference {} does not point to a commit", | ||||||
|             .peel_to_commit() |             DEFAULT_BRANCH | ||||||
|             .expect(&format!("reference {} does not point to a commit", DEFAULT_BRANCH)) |         ))], | ||||||
|         ], |  | ||||||
|         Err(err) => match err.code() { |         Err(err) => match err.code() { | ||||||
|             // no commit exists yet
 |             // no commit exists yet
 | ||||||
|             git2::ErrorCode::NotFound => vec![], |             git2::ErrorCode::NotFound => vec![], | ||||||
|             _ => panic!("could not read latest commit from {}", DEFAULT_BRANCH), |             _ => panic!("could not read latest commit from {}", DEFAULT_BRANCH), | ||||||
|         } |         }, | ||||||
|     }; |     }; | ||||||
|     repo.commit( |     repo.commit( | ||||||
|         Some(DEFAULT_BRANCH), |         Some(DEFAULT_BRANCH), | ||||||
|  | @ -79,7 +85,6 @@ fn main() { | ||||||
|          I wonder if it supports extended commit descriptions?\n",
 |          I wonder if it supports extended commit descriptions?\n",
 | ||||||
|         &to_add_tree, |         &to_add_tree, | ||||||
|         &parent_commits.iter().collect::<Vec<_>>()[..], |         &parent_commits.iter().collect::<Vec<_>>()[..], | ||||||
|     ).expect("could not commit the index we just wrote"); |     ) | ||||||
| 
 |     .expect("could not commit the index we just wrote"); | ||||||
| 
 |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,16 +1,16 @@ | ||||||
| extern crate exec_helpers; | extern crate exec_helpers; | ||||||
| // extern crate arglib_netencode;
 | // extern crate arglib_netencode;
 | ||||||
| // extern crate netencode;
 | // extern crate netencode;
 | ||||||
| extern crate imap; |  | ||||||
| extern crate epoll; | extern crate epoll; | ||||||
|  | extern crate imap; | ||||||
| 
 | 
 | ||||||
| // use netencode::dec;
 | // use netencode::dec;
 | ||||||
| use std::convert::TryFrom; |  | ||||||
| use std::io::{Read, Write}; |  | ||||||
| use std::fs::File; |  | ||||||
| use std::os::unix::io::{FromRawFd, AsRawFd, RawFd}; |  | ||||||
| use std::time::Duration; |  | ||||||
| use imap::extensions::idle::SetReadTimeout; | use imap::extensions::idle::SetReadTimeout; | ||||||
|  | use std::convert::TryFrom; | ||||||
|  | use std::fs::File; | ||||||
|  | use std::io::{Read, Write}; | ||||||
|  | use std::os::unix::io::{AsRawFd, FromRawFd, RawFd}; | ||||||
|  | use std::time::Duration; | ||||||
| 
 | 
 | ||||||
| /// Implements an UCSPI client that wraps fd 6 & 7
 | /// Implements an UCSPI client that wraps fd 6 & 7
 | ||||||
| /// and implements Write and Read with a timeout.
 | /// and implements Write and Read with a timeout.
 | ||||||
|  | @ -33,7 +33,7 @@ impl UcspiClient { | ||||||
|                 read: File::from_raw_fd(6), |                 read: File::from_raw_fd(6), | ||||||
|                 read_epoll_fd, |                 read_epoll_fd, | ||||||
|                 read_timeout: None, |                 read_timeout: None, | ||||||
|                 write: File::from_raw_fd(7) |                 write: File::from_raw_fd(7), | ||||||
|             }) |             }) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -61,14 +61,16 @@ impl Read for UcspiClient { | ||||||
|             self.read_epoll_fd, |             self.read_epoll_fd, | ||||||
|             epoll::ControlOptions::EPOLL_CTL_ADD, |             epoll::ControlOptions::EPOLL_CTL_ADD, | ||||||
|             self.read.as_raw_fd(), |             self.read.as_raw_fd(), | ||||||
|             epoll::Event::new(epoll::Events::EPOLLIN, NO_DATA) |             epoll::Event::new(epoll::Events::EPOLLIN, NO_DATA), | ||||||
|         )?; |         )?; | ||||||
|         let UNUSED = epoll::Event::new(epoll::Events::EPOLLIN, NO_DATA); |         let UNUSED = epoll::Event::new(epoll::Events::EPOLLIN, NO_DATA); | ||||||
|         let wait = epoll::wait( |         let wait = epoll::wait( | ||||||
|             self.read_epoll_fd, |             self.read_epoll_fd, | ||||||
|             match self.read_timeout { |             match self.read_timeout { | ||||||
|                 Some(duration) => i32::try_from(duration.as_millis()).expect("duration too big for epoll"), |                 Some(duration) => { | ||||||
|                 None => -1 // infinite
 |                     i32::try_from(duration.as_millis()).expect("duration too big for epoll") | ||||||
|  |                 } | ||||||
|  |                 None => -1, // infinite
 | ||||||
|             }, |             }, | ||||||
|             // event that was generated; but we don’t care
 |             // event that was generated; but we don’t care
 | ||||||
|             &mut vec![UNUSED; 1][..], |             &mut vec![UNUSED; 1][..], | ||||||
|  | @ -79,11 +81,14 @@ impl Read for UcspiClient { | ||||||
|             self.read_epoll_fd, |             self.read_epoll_fd, | ||||||
|             epoll::ControlOptions::EPOLL_CTL_DEL, |             epoll::ControlOptions::EPOLL_CTL_DEL, | ||||||
|             self.read.as_raw_fd(), |             self.read.as_raw_fd(), | ||||||
|             UNUSED |             UNUSED, | ||||||
|         )?; |         )?; | ||||||
|         match wait { |         match wait { | ||||||
|             // timeout happened (0 events)
 |             // timeout happened (0 events)
 | ||||||
|             Ok(0) => Err(std::io::Error::new(std::io::ErrorKind::TimedOut, "ucspi read timeout")), |             Ok(0) => Err(std::io::Error::new( | ||||||
|  |                 std::io::ErrorKind::TimedOut, | ||||||
|  |                 "ucspi read timeout", | ||||||
|  |             )), | ||||||
|             // its ready for reading, we can read
 |             // its ready for reading, we can read
 | ||||||
|             Ok(_) => self.read.read(buf), |             Ok(_) => self.read.read(buf), | ||||||
|             // error
 |             // error
 | ||||||
|  | @ -110,18 +115,21 @@ fn main() { | ||||||
|     let username = std::env::var("IMAP_USERNAME").expect("username"); |     let username = std::env::var("IMAP_USERNAME").expect("username"); | ||||||
|     let password = std::env::var("IMAP_PASSWORD").expect("password"); |     let password = std::env::var("IMAP_PASSWORD").expect("password"); | ||||||
| 
 | 
 | ||||||
|     let net = unsafe { |     let net = unsafe { UcspiClient::new_from_6_and_7().expect("no ucspi client for you") }; | ||||||
|         UcspiClient::new_from_6_and_7().expect("no ucspi client for you") |  | ||||||
|     }; |  | ||||||
|     let client = imap::Client::new(net); |     let client = imap::Client::new(net); | ||||||
|     let mut session = client.login(username, password).map_err(|(err, _)| err).expect("unable to login"); |     let mut session = client | ||||||
|  |         .login(username, password) | ||||||
|  |         .map_err(|(err, _)| err) | ||||||
|  |         .expect("unable to login"); | ||||||
|     eprintln!("{:#?}", session); |     eprintln!("{:#?}", session); | ||||||
|     let list = session.list(None, Some("*")); |     let list = session.list(None, Some("*")); | ||||||
|     eprintln!("{:#?}", list); |     eprintln!("{:#?}", list); | ||||||
|     let mailbox = session.examine("INBOX"); |     let mailbox = session.examine("INBOX"); | ||||||
|     eprintln!("{:#?}", mailbox); |     eprintln!("{:#?}", mailbox); | ||||||
|     fn now() -> String { |     fn now() -> String { | ||||||
|         String::from_utf8_lossy(&std::process::Command::new("date").output().unwrap().stdout).trim_right().to_string() |         String::from_utf8_lossy(&std::process::Command::new("date").output().unwrap().stdout) | ||||||
|  |             .trim_right() | ||||||
|  |             .to_string() | ||||||
|     } |     } | ||||||
|     loop { |     loop { | ||||||
|         eprintln!("{}: idling on INBOX", now()); |         eprintln!("{}: idling on INBOX", now()); | ||||||
|  |  | ||||||
|  | @ -1,12 +1,12 @@ | ||||||
| extern crate netencode; |  | ||||||
| extern crate mustache; |  | ||||||
| extern crate arglib_netencode; | extern crate arglib_netencode; | ||||||
|  | extern crate mustache; | ||||||
|  | extern crate netencode; | ||||||
| 
 | 
 | ||||||
| use mustache::{Data}; | use mustache::Data; | ||||||
| use netencode::{T}; | use netencode::T; | ||||||
| use std::collections::HashMap; | use std::collections::HashMap; | ||||||
| use std::os::unix::ffi::{OsStrExt}; | use std::io::Read; | ||||||
| use std::io::{Read}; | use std::os::unix::ffi::OsStrExt; | ||||||
| 
 | 
 | ||||||
| fn netencode_to_mustache_data_dwim(t: T) -> Data { | fn netencode_to_mustache_data_dwim(t: T) -> Data { | ||||||
|     match t { |     match t { | ||||||
|  | @ -25,27 +25,26 @@ fn netencode_to_mustache_data_dwim(t: T) -> Data { | ||||||
|         T::Record(xs) => Data::Map( |         T::Record(xs) => Data::Map( | ||||||
|             xs.into_iter() |             xs.into_iter() | ||||||
|                 .map(|(key, val)| (key, netencode_to_mustache_data_dwim(val))) |                 .map(|(key, val)| (key, netencode_to_mustache_data_dwim(val))) | ||||||
|                 .collect::<HashMap<_,_>>() |                 .collect::<HashMap<_, _>>(), | ||||||
|         ), |         ), | ||||||
|         T::List(xs) => Data::Vec( |         T::List(xs) => Data::Vec( | ||||||
|             xs.into_iter() |             xs.into_iter() | ||||||
|                 .map(|x| netencode_to_mustache_data_dwim(x)) |                 .map(|x| netencode_to_mustache_data_dwim(x)) | ||||||
|                 .collect::<Vec<_>>() |                 .collect::<Vec<_>>(), | ||||||
|         ), |         ), | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub fn from_stdin() -> () { | pub fn from_stdin() -> () { | ||||||
|     let data = netencode_to_mustache_data_dwim( |     let data = netencode_to_mustache_data_dwim(arglib_netencode::arglib_netencode( | ||||||
|         arglib_netencode::arglib_netencode("netencode-mustache", Some(std::ffi::OsStr::new("TEMPLATE_DATA"))) |         "netencode-mustache", | ||||||
|     ); |         Some(std::ffi::OsStr::new("TEMPLATE_DATA")), | ||||||
|  |     )); | ||||||
|     let mut stdin = String::new(); |     let mut stdin = String::new(); | ||||||
|     std::io::stdin().read_to_string(&mut stdin).unwrap(); |     std::io::stdin().read_to_string(&mut stdin).unwrap(); | ||||||
|     mustache::compile_str(&stdin) |     mustache::compile_str(&stdin) | ||||||
|         .and_then(|templ| templ.render_data( |         .and_then(|templ| templ.render_data(&mut std::io::stdout(), &data)) | ||||||
|             &mut std::io::stdout(), |         .unwrap() | ||||||
|             &data |  | ||||||
|         )).unwrap() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub fn main() { | pub fn main() { | ||||||
|  |  | ||||||
|  | @ -1,9 +1,9 @@ | ||||||
| extern crate nom; |  | ||||||
| extern crate exec_helpers; | extern crate exec_helpers; | ||||||
|  | extern crate nom; | ||||||
| 
 | 
 | ||||||
| use std::collections::HashMap; | use std::collections::HashMap; | ||||||
| use std::io::{Write, Read}; | use std::fmt::{Debug, Display}; | ||||||
| use std::fmt::{Display, Debug}; | use std::io::{Read, Write}; | ||||||
| 
 | 
 | ||||||
| #[derive(Debug, PartialEq, Eq, Clone)] | #[derive(Debug, PartialEq, Eq, Clone)] | ||||||
| pub enum T { | pub enum T { | ||||||
|  | @ -46,22 +46,19 @@ impl T { | ||||||
|             T::I7(i) => U::I7(*i), |             T::I7(i) => U::I7(*i), | ||||||
|             T::Text(t) => U::Text(t.as_str()), |             T::Text(t) => U::Text(t.as_str()), | ||||||
|             T::Binary(v) => U::Binary(v), |             T::Binary(v) => U::Binary(v), | ||||||
|             T::Sum(Tag { tag, val }) => U::Sum( |             T::Sum(Tag { tag, val }) => U::Sum(Tag { | ||||||
|                 Tag { tag: tag.as_str(), val: Box::new(val.to_u()) } |                 tag: tag.as_str(), | ||||||
|             ), |                 val: Box::new(val.to_u()), | ||||||
|             T::Record(map) => U::Record( |             }), | ||||||
|                 map.iter().map(|(k, v)| (k.as_str(), v.to_u())).collect() |             T::Record(map) => U::Record(map.iter().map(|(k, v)| (k.as_str(), v.to_u())).collect()), | ||||||
|             ), |             T::List(l) => U::List(l.iter().map(|v| v.to_u()).collect::<Vec<U<'a>>>()), | ||||||
|             T::List(l) => U::List( |  | ||||||
|                 l.iter().map(|v| v.to_u()).collect::<Vec<U<'a>>>() |  | ||||||
|             ), |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn encode<'a>(&'a self) -> Vec<u8> { |     pub fn encode<'a>(&'a self) -> Vec<u8> { | ||||||
|         match self { |         match self { | ||||||
|             // TODO: don’t go via U, inefficient
 |             // TODO: don’t go via U, inefficient
 | ||||||
|             o => o.to_u().encode() |             o => o.to_u().encode(), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -110,15 +107,16 @@ impl<'a> U<'a> { | ||||||
|             U::I7(i) => T::I7(*i), |             U::I7(i) => T::I7(*i), | ||||||
|             U::Text(t) => T::Text((*t).to_owned()), |             U::Text(t) => T::Text((*t).to_owned()), | ||||||
|             U::Binary(v) => T::Binary((*v).to_owned()), |             U::Binary(v) => T::Binary((*v).to_owned()), | ||||||
|             U::Sum(Tag { tag, val }) => T::Sum( |             U::Sum(Tag { tag, val }) => T::Sum(Tag { | ||||||
|                 Tag { tag: (*tag).to_owned(), val: Box::new(val.to_t()) } |                 tag: (*tag).to_owned(), | ||||||
|             ), |                 val: Box::new(val.to_t()), | ||||||
|  |             }), | ||||||
|             U::Record(map) => T::Record( |             U::Record(map) => T::Record( | ||||||
|                 map.iter().map(|(k, v)| ((*k).to_owned(), v.to_t())).collect::<HashMap<String, T>>() |                 map.iter() | ||||||
|             ), |                     .map(|(k, v)| ((*k).to_owned(), v.to_t())) | ||||||
|             U::List(l) => T::List( |                     .collect::<HashMap<String, T>>(), | ||||||
|                 l.iter().map(|v| v.to_t()).collect::<Vec<T>>() |  | ||||||
|             ), |             ), | ||||||
|  |             U::List(l) => T::List(l.iter().map(|v| v.to_t()).collect::<Vec<T>>()), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -127,15 +125,17 @@ impl<'a> U<'a> { | ||||||
| pub struct Tag<S, A> { | pub struct Tag<S, A> { | ||||||
|     // TODO: make into &str
 |     // TODO: make into &str
 | ||||||
|     pub tag: S, |     pub tag: S, | ||||||
|     pub val: Box<A> |     pub val: Box<A>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl<S, A> Tag<S, A> { | impl<S, A> Tag<S, A> { | ||||||
|     fn map<F, B>(self, f: F) -> Tag<S, B> |     fn map<F, B>(self, f: F) -> Tag<S, B> | ||||||
|         where F: Fn(A) -> B { |     where | ||||||
|  |         F: Fn(A) -> B, | ||||||
|  |     { | ||||||
|         Tag { |         Tag { | ||||||
|             tag: self.tag, |             tag: self.tag, | ||||||
|               val: Box::new(f(*self.val)) |             val: Box::new(f(*self.val)), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -149,7 +149,13 @@ fn encode_tag<W: Write>(w: &mut W, tag: &str, val: &U) -> std::io::Result<()> { | ||||||
| pub fn encode<W: Write>(w: &mut W, u: &U) -> std::io::Result<()> { | pub fn encode<W: Write>(w: &mut W, u: &U) -> std::io::Result<()> { | ||||||
|     match u { |     match u { | ||||||
|         U::Unit => write!(w, "u,"), |         U::Unit => write!(w, "u,"), | ||||||
|       U::N1(b) => if *b { write!(w, "n1:1,") } else { write!(w, "n1:0,") }, |         U::N1(b) => { | ||||||
|  |             if *b { | ||||||
|  |                 write!(w, "n1:1,") | ||||||
|  |             } else { | ||||||
|  |                 write!(w, "n1:0,") | ||||||
|  |             } | ||||||
|  |         } | ||||||
|         U::N3(n) => write!(w, "n3:{},", n), |         U::N3(n) => write!(w, "n3:{},", n), | ||||||
|         U::N6(n) => write!(w, "n6:{},", n), |         U::N6(n) => write!(w, "n6:{},", n), | ||||||
|         U::N7(n) => write!(w, "n7:{},", n), |         U::N7(n) => write!(w, "n7:{},", n), | ||||||
|  | @ -165,7 +171,7 @@ pub fn encode<W: Write>(w: &mut W, u: &U) -> std::io::Result<()> { | ||||||
|             write!(w, "b{}:", s.len()); |             write!(w, "b{}:", s.len()); | ||||||
|             w.write_all(&s); |             w.write_all(&s); | ||||||
|             write!(w, ",") |             write!(w, ",") | ||||||
|       }, |         } | ||||||
|         U::Sum(Tag { tag, val }) => encode_tag(w, tag, val), |         U::Sum(Tag { tag, val }) => encode_tag(w, tag, val), | ||||||
|         U::Record(m) => { |         U::Record(m) => { | ||||||
|             let mut c = std::io::Cursor::new(vec![]); |             let mut c = std::io::Cursor::new(vec![]); | ||||||
|  | @ -175,7 +181,7 @@ pub fn encode<W: Write>(w: &mut W, u: &U) -> std::io::Result<()> { | ||||||
|             write!(w, "{{{}:", c.get_ref().len())?; |             write!(w, "{{{}:", c.get_ref().len())?; | ||||||
|             w.write_all(c.get_ref())?; |             w.write_all(c.get_ref())?; | ||||||
|             write!(w, "}}") |             write!(w, "}}") | ||||||
|       }, |         } | ||||||
|         U::List(l) => { |         U::List(l) => { | ||||||
|             let mut c = std::io::Cursor::new(vec![]); |             let mut c = std::io::Cursor::new(vec![]); | ||||||
|             for u in l { |             for u in l { | ||||||
|  | @ -197,27 +203,36 @@ pub fn u_from_stdin_or_die_user_error<'a>(prog_name: &'_ str, stdin_buf: &'a mut | ||||||
|     let u = match parse::u_u(stdin_buf) { |     let u = match parse::u_u(stdin_buf) { | ||||||
|         Ok((rest, u)) => match rest { |         Ok((rest, u)) => match rest { | ||||||
|             b"" => u, |             b"" => u, | ||||||
|             _ => exec_helpers::die_user_error(prog_name, format!("stdin contained some soup after netencode value: {:?}", String::from_utf8_lossy(rest))) |             _ => exec_helpers::die_user_error( | ||||||
|  |                 prog_name, | ||||||
|  |                 format!( | ||||||
|  |                     "stdin contained some soup after netencode value: {:?}", | ||||||
|  |                     String::from_utf8_lossy(rest) | ||||||
|  |                 ), | ||||||
|  |             ), | ||||||
|         }, |         }, | ||||||
|         Err(err) => exec_helpers::die_user_error(prog_name, format!("unable to parse netencode from stdin: {:?}", err)) |         Err(err) => exec_helpers::die_user_error( | ||||||
|  |             prog_name, | ||||||
|  |             format!("unable to parse netencode from stdin: {:?}", err), | ||||||
|  |         ), | ||||||
|     }; |     }; | ||||||
|     u |     u | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub mod parse { | pub mod parse { | ||||||
|     use super::{T, Tag, U}; |     use super::{Tag, T, U}; | ||||||
| 
 | 
 | ||||||
|     use std::str::FromStr; |  | ||||||
|     use std::ops::Neg; |  | ||||||
|     use std::collections::HashMap; |     use std::collections::HashMap; | ||||||
|  |     use std::ops::Neg; | ||||||
|  |     use std::str::FromStr; | ||||||
| 
 | 
 | ||||||
|     use nom::{IResult}; |     use nom::branch::alt; | ||||||
|     use nom::branch::{alt}; |  | ||||||
|     use nom::bytes::streaming::{tag, take}; |     use nom::bytes::streaming::{tag, take}; | ||||||
|     use nom::character::streaming::{digit1, char}; |     use nom::character::streaming::{char, digit1}; | ||||||
|     use nom::sequence::{tuple}; |     use nom::combinator::{flat_map, map, map_parser, map_res, opt}; | ||||||
|     use nom::combinator::{map, map_res, flat_map, map_parser, opt}; |  | ||||||
|     use nom::error::{context, ErrorKind, ParseError}; |     use nom::error::{context, ErrorKind, ParseError}; | ||||||
|  |     use nom::sequence::tuple; | ||||||
|  |     use nom::IResult; | ||||||
| 
 | 
 | ||||||
|     fn unit_t(s: &[u8]) -> IResult<&[u8], ()> { |     fn unit_t(s: &[u8]) -> IResult<&[u8], ()> { | ||||||
|         let (s, _) = context("unit", tag("u,"))(s)?; |         let (s, _) = context("unit", tag("u,"))(s)?; | ||||||
|  | @ -227,9 +242,9 @@ pub mod parse { | ||||||
|     fn usize_t(s: &[u8]) -> IResult<&[u8], usize> { |     fn usize_t(s: &[u8]) -> IResult<&[u8], usize> { | ||||||
|         context( |         context( | ||||||
|             "usize", |             "usize", | ||||||
|             map_res( |             map_res(map_res(digit1, |n| std::str::from_utf8(n)), |s| { | ||||||
|                 map_res(digit1, |n| std::str::from_utf8(n)), |                 s.parse::<usize>() | ||||||
|                 |s| s.parse::<usize>()) |             }), | ||||||
|         )(s) |         )(s) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -238,87 +253,77 @@ pub mod parse { | ||||||
|             // This is the point where we check the descriminator;
 |             // This is the point where we check the descriminator;
 | ||||||
|             // if the beginning char does not match, we can immediately return.
 |             // if the beginning char does not match, we can immediately return.
 | ||||||
|             let (s, _) = char(begin)(s)?; |             let (s, _) = char(begin)(s)?; | ||||||
|             let (s, (len, _)) = tuple(( |             let (s, (len, _)) = tuple((usize_t, char(':')))(s)?; | ||||||
|                 usize_t, |             let (s, (res, _)) = tuple((take(len), char(end)))(s)?; | ||||||
|                 char(':') |  | ||||||
|             ))(s)?; |  | ||||||
|             let (s, (res, _)) = tuple(( |  | ||||||
|                 take(len), |  | ||||||
|                 char(end) |  | ||||||
|             ))(s)?; |  | ||||||
|             Ok((s, res)) |             Ok((s, res)) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     fn uint_t<'a, I: FromStr + 'a>(t: &'static str) -> impl Fn(&'a [u8]) -> IResult<&'a [u8], I> { |     fn uint_t<'a, I: FromStr + 'a>(t: &'static str) -> impl Fn(&'a [u8]) -> IResult<&'a [u8], I> { | ||||||
|         move |s: &'a [u8]| { |         move |s: &'a [u8]| { | ||||||
|             let (s, (_, _, int, _)) = tuple(( |             let (s, (_, _, int, _)) = tuple(( | ||||||
|                 tag(t.as_bytes()), |                 tag(t.as_bytes()), | ||||||
|                 char(':'), |                 char(':'), | ||||||
|                 map_res( |                 map_res(map_res(digit1, |n: &[u8]| std::str::from_utf8(n)), |s| { | ||||||
|                     map_res(digit1, |n: &[u8]| std::str::from_utf8(n)), |                     s.parse::<I>() | ||||||
|                     |s| s.parse::<I>() |                 }), | ||||||
|                 ), |                 char(','), | ||||||
|                 char(',') |  | ||||||
|             ))(s)?; |             ))(s)?; | ||||||
|             Ok((s, int)) |             Ok((s, int)) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn bool_t<'a>() -> impl Fn(&'a [u8]) -> IResult<&'a [u8], bool> { |     fn bool_t<'a>() -> impl Fn(&'a [u8]) -> IResult<&'a [u8], bool> { | ||||||
|         context("bool", alt(( |         context( | ||||||
|             map(tag("n1:0,"), |_| false), |             "bool", | ||||||
|             map(tag("n1:1,"), |_| true), |             alt((map(tag("n1:0,"), |_| false), map(tag("n1:1,"), |_| true))), | ||||||
|         ))) |         ) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn int_t<'a, I: FromStr + Neg<Output=I>>(t: &'static str) -> impl Fn(&'a [u8]) -> IResult<&[u8], I> { |     fn int_t<'a, I: FromStr + Neg<Output = I>>( | ||||||
|         context( |         t: &'static str, | ||||||
|             t, |     ) -> impl Fn(&'a [u8]) -> IResult<&[u8], I> { | ||||||
|             move |s: &'a [u8]| { |         context(t, move |s: &'a [u8]| { | ||||||
|             let (s, (_, _, neg, int, _)) = tuple(( |             let (s, (_, _, neg, int, _)) = tuple(( | ||||||
|                 tag(t.as_bytes()), |                 tag(t.as_bytes()), | ||||||
|                 char(':'), |                 char(':'), | ||||||
|                 opt(char('-')), |                 opt(char('-')), | ||||||
|                     map_res( |                 map_res(map_res(digit1, |n: &[u8]| std::str::from_utf8(n)), |s| { | ||||||
|                         map_res(digit1, |n: &[u8]| std::str::from_utf8(n)), |                     s.parse::<I>() | ||||||
|                         |s| s.parse::<I>() |                 }), | ||||||
|                     ), |                 char(','), | ||||||
|                     char(',') |  | ||||||
|             ))(s)?; |             ))(s)?; | ||||||
|             let res = match neg { |             let res = match neg { | ||||||
|                 Some(_) => -int, |                 Some(_) => -int, | ||||||
|                 None => int, |                 None => int, | ||||||
|             }; |             }; | ||||||
|             Ok((s, res)) |             Ok((s, res)) | ||||||
|             } |         }) | ||||||
|         ) |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn tag_t(s: &[u8]) -> IResult<&[u8], Tag<String, T>> { |     fn tag_t(s: &[u8]) -> IResult<&[u8], Tag<String, T>> { | ||||||
|         // recurses into the main parser
 |         // recurses into the main parser
 | ||||||
|         map(tag_g(t_t), |         map(tag_g(t_t), |Tag { tag, val }| Tag { | ||||||
|             |Tag {tag, val}| |  | ||||||
|             Tag { |  | ||||||
|             tag: tag.to_string(), |             tag: tag.to_string(), | ||||||
|                 val |             val, | ||||||
|         })(s) |         })(s) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn tag_g<'a, P, O>(inner: P) -> impl Fn(&'a [u8]) -> IResult<&'a [u8], Tag<&'a str, O>> |     fn tag_g<'a, P, O>(inner: P) -> impl Fn(&'a [u8]) -> IResult<&'a [u8], Tag<&'a str, O>> | ||||||
|     where |     where | ||||||
|         P: Fn(&'a [u8]) -> IResult<&'a [u8], O> |         P: Fn(&'a [u8]) -> IResult<&'a [u8], O>, | ||||||
|     { |     { | ||||||
|         move |s: &[u8]| { |         move |s: &[u8]| { | ||||||
|             let (s, tag) = sized('<', '|')(s)?; |             let (s, tag) = sized('<', '|')(s)?; | ||||||
|             let (s, val) = inner(s)?; |             let (s, val) = inner(s)?; | ||||||
|             Ok((s, Tag { |             Ok(( | ||||||
|  |                 s, | ||||||
|  |                 Tag { | ||||||
|                     tag: std::str::from_utf8(tag) |                     tag: std::str::from_utf8(tag) | ||||||
|                         .map_err(|_| nom::Err::Failure((s, ErrorKind::Char)))?, |                         .map_err(|_| nom::Err::Failure((s, ErrorKind::Char)))?, | ||||||
|                 val: Box::new(val) |                     val: Box::new(val), | ||||||
|             })) |                 }, | ||||||
| 
 |             )) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -330,9 +335,9 @@ pub mod parse { | ||||||
| 
 | 
 | ||||||
|     fn text_g(s: &[u8]) -> IResult<&[u8], &str> { |     fn text_g(s: &[u8]) -> IResult<&[u8], &str> { | ||||||
|         let (s, res) = sized('t', ',')(s)?; |         let (s, res) = sized('t', ',')(s)?; | ||||||
|         Ok((s, |         Ok(( | ||||||
|             std::str::from_utf8(res) |             s, | ||||||
|                 .map_err(|_| nom::Err::Failure((s, ErrorKind::Char)))?, |             std::str::from_utf8(res).map_err(|_| nom::Err::Failure((s, ErrorKind::Char)))?, | ||||||
|         )) |         )) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -374,22 +379,24 @@ pub mod parse { | ||||||
|     { |     { | ||||||
|         map_parser( |         map_parser( | ||||||
|             sized('[', ']'), |             sized('[', ']'), | ||||||
|             nom::multi::many0(inner_no_empty_string(inner)) |             nom::multi::many0(inner_no_empty_string(inner)), | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn record_t<'a>(s: &'a [u8]) -> IResult<&'a [u8], HashMap<String, T>> { |     fn record_t<'a>(s: &'a [u8]) -> IResult<&'a [u8], HashMap<String, T>> { | ||||||
|         let (s, r) = record_g(t_t)(s)?; |         let (s, r) = record_g(t_t)(s)?; | ||||||
|         Ok((s, |         Ok(( | ||||||
|  |             s, | ||||||
|             r.into_iter() |             r.into_iter() | ||||||
|                 .map(|(k, v)| (k.to_string(), v)) |                 .map(|(k, v)| (k.to_string(), v)) | ||||||
|             .collect::<HashMap<_,_>>())) |                 .collect::<HashMap<_, _>>(), | ||||||
|  |         )) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn record_g<'a, P, O>(inner: P) -> impl Fn(&'a [u8]) -> IResult<&'a [u8], HashMap<&'a str, O>> |     fn record_g<'a, P, O>(inner: P) -> impl Fn(&'a [u8]) -> IResult<&'a [u8], HashMap<&'a str, O>> | ||||||
|     where |     where | ||||||
|         O: Clone, |         O: Clone, | ||||||
|         P: Fn(&'a [u8]) -> IResult<&'a [u8], O> |         P: Fn(&'a [u8]) -> IResult<&'a [u8], O>, | ||||||
|     { |     { | ||||||
|         move |s: &'a [u8]| { |         move |s: &'a [u8]| { | ||||||
|             let (s, map) = map_parser( |             let (s, map) = map_parser( | ||||||
|  | @ -404,8 +411,8 @@ pub mod parse { | ||||||
|                             acc.insert(tag, *val); |                             acc.insert(tag, *val); | ||||||
|                         } |                         } | ||||||
|                         acc |                         acc | ||||||
|                     } |                     }, | ||||||
|                 ) |                 ), | ||||||
|             )(s)?; |             )(s)?; | ||||||
|             if map.is_empty() { |             if map.is_empty() { | ||||||
|                 // records must not be empty, according to the spec
 |                 // records must not be empty, according to the spec
 | ||||||
|  | @ -424,7 +431,6 @@ pub mod parse { | ||||||
|             map(tag_g(u_u), |t| U::Sum(t)), |             map(tag_g(u_u), |t| U::Sum(t)), | ||||||
|             map(list_g(u_u), U::List), |             map(list_g(u_u), U::List), | ||||||
|             map(record_g(u_u), U::Record), |             map(record_g(u_u), U::Record), | ||||||
| 
 |  | ||||||
|             map(bool_t(), |u| U::N1(u)), |             map(bool_t(), |u| U::N1(u)), | ||||||
|             map(uint_t("n3"), |u| U::N3(u)), |             map(uint_t("n3"), |u| U::N3(u)), | ||||||
|             map(uint_t("n6"), |u| U::N6(u)), |             map(uint_t("n6"), |u| U::N6(u)), | ||||||
|  | @ -432,7 +438,6 @@ pub mod parse { | ||||||
|             map(int_t("i3"), |u| U::I3(u)), |             map(int_t("i3"), |u| U::I3(u)), | ||||||
|             map(int_t("i6"), |u| U::I6(u)), |             map(int_t("i6"), |u| U::I6(u)), | ||||||
|             map(int_t("i7"), |u| U::I7(u)), |             map(int_t("i7"), |u| U::I7(u)), | ||||||
| 
 |  | ||||||
|             // less common
 |             // less common
 | ||||||
|             map(uint_t("n2"), |u| U::N3(u)), |             map(uint_t("n2"), |u| U::N3(u)), | ||||||
|             map(uint_t("n4"), |u| U::N6(u)), |             map(uint_t("n4"), |u| U::N6(u)), | ||||||
|  | @ -453,7 +458,6 @@ pub mod parse { | ||||||
|             map(tag_t, |t| T::Sum(t)), |             map(tag_t, |t| T::Sum(t)), | ||||||
|             map(list_t, |l| T::List(l)), |             map(list_t, |l| T::List(l)), | ||||||
|             map(record_t, |p| T::Record(p)), |             map(record_t, |p| T::Record(p)), | ||||||
| 
 |  | ||||||
|             map(bool_t(), |u| T::N1(u)), |             map(bool_t(), |u| T::N1(u)), | ||||||
|             // 8, 64 and 128 bit
 |             // 8, 64 and 128 bit
 | ||||||
|             map(uint_t("n3"), |u| T::N3(u)), |             map(uint_t("n3"), |u| T::N3(u)), | ||||||
|  | @ -462,7 +466,6 @@ pub mod parse { | ||||||
|             map(int_t("i3"), |u| T::I3(u)), |             map(int_t("i3"), |u| T::I3(u)), | ||||||
|             map(int_t("i6"), |u| T::I6(u)), |             map(int_t("i6"), |u| T::I6(u)), | ||||||
|             map(int_t("i7"), |u| T::I7(u)), |             map(int_t("i7"), |u| T::I7(u)), | ||||||
| 
 |  | ||||||
|             // less common
 |             // less common
 | ||||||
|             map(uint_t("n2"), |u| T::N3(u)), |             map(uint_t("n2"), |u| T::N3(u)), | ||||||
|             map(uint_t("n4"), |u| T::N6(u)), |             map(uint_t("n4"), |u| T::N6(u)), | ||||||
|  | @ -481,30 +484,18 @@ pub mod parse { | ||||||
| 
 | 
 | ||||||
|         #[test] |         #[test] | ||||||
|         fn test_parse_unit_t() { |         fn test_parse_unit_t() { | ||||||
|             assert_eq!( |             assert_eq!(unit_t("u,".as_bytes()), Ok(("".as_bytes(), ()))); | ||||||
|                 unit_t("u,".as_bytes()), |  | ||||||
|                 Ok(("".as_bytes(), ())) |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         #[test] |         #[test] | ||||||
|         fn test_parse_bool_t() { |         fn test_parse_bool_t() { | ||||||
|             assert_eq!( |             assert_eq!(bool_t()("n1:0,".as_bytes()), Ok(("".as_bytes(), false))); | ||||||
|                 bool_t()("n1:0,".as_bytes()), |             assert_eq!(bool_t()("n1:1,".as_bytes()), Ok(("".as_bytes(), true))); | ||||||
|                 Ok(("".as_bytes(), false)) |  | ||||||
|             ); |  | ||||||
|             assert_eq!( |  | ||||||
|                 bool_t()("n1:1,".as_bytes()), |  | ||||||
|                 Ok(("".as_bytes(), true)) |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         #[test] |         #[test] | ||||||
|         fn test_parse_usize_t() { |         fn test_parse_usize_t() { | ||||||
|             assert_eq!( |             assert_eq!(usize_t("32foo".as_bytes()), Ok(("foo".as_bytes(), 32))); | ||||||
|                 usize_t("32foo".as_bytes()), |  | ||||||
|                 Ok(("foo".as_bytes(), 32)) |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         #[test] |         #[test] | ||||||
|  | @ -515,7 +506,10 @@ pub mod parse { | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 uint_t::<u8>("n3")("n3:1024,abc".as_bytes()), |                 uint_t::<u8>("n3")("n3:1024,abc".as_bytes()), | ||||||
|                 Err(nom::Err::Error(("1024,abc".as_bytes(), nom::error::ErrorKind::MapRes))) |                 Err(nom::Err::Error(( | ||||||
|  |                     "1024,abc".as_bytes(), | ||||||
|  |                     nom::error::ErrorKind::MapRes | ||||||
|  |                 ))) | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 int_t::<i64>("i6")("i6:-23,abc".as_bytes()), |                 int_t::<i64>("i6")("i6:-23,abc".as_bytes()), | ||||||
|  | @ -544,18 +538,21 @@ pub mod parse { | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 text("t5:hello,".as_bytes()), |                 text("t5:hello,".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), T::Text("hello".to_owned()))), |                 Ok(("".as_bytes(), T::Text("hello".to_owned()))), | ||||||
|                 "{}", r"t5:hello," |                 "{}", | ||||||
|  |                 r"t5:hello," | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 text("t4:fo".as_bytes()), |                 text("t4:fo".as_bytes()), | ||||||
|                 // The content of the text should be 4 long
 |                 // The content of the text should be 4 long
 | ||||||
|                 Err(nom::Err::Incomplete(nom::Needed::Size(4))), |                 Err(nom::Err::Incomplete(nom::Needed::Size(4))), | ||||||
|                 "{}", r"t4:fo," |                 "{}", | ||||||
|  |                 r"t4:fo," | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 text("t9:今日は,".as_bytes()), |                 text("t9:今日は,".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), T::Text("今日は".to_owned()))), |                 Ok(("".as_bytes(), T::Text("今日は".to_owned()))), | ||||||
|                 "{}", r"t9:今日は," |                 "{}", | ||||||
|  |                 r"t9:今日は," | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -564,24 +561,28 @@ pub mod parse { | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 binary()("b5:hello,".as_bytes()), |                 binary()("b5:hello,".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), T::Binary(Vec::from("hello".to_owned())))), |                 Ok(("".as_bytes(), T::Binary(Vec::from("hello".to_owned())))), | ||||||
|                 "{}", r"b5:hello," |                 "{}", | ||||||
|  |                 r"b5:hello," | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 binary()("b4:fo".as_bytes()), |                 binary()("b4:fo".as_bytes()), | ||||||
|                 // The content of the byte should be 4 long
 |                 // The content of the byte should be 4 long
 | ||||||
|                 Err(nom::Err::Incomplete(nom::Needed::Size(4))), |                 Err(nom::Err::Incomplete(nom::Needed::Size(4))), | ||||||
|                 "{}", r"b4:fo," |                 "{}", | ||||||
|  |                 r"b4:fo," | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 binary()("b4:foob".as_bytes()), |                 binary()("b4:foob".as_bytes()), | ||||||
|                 // The content is 4 bytes now, but the finishing , is missing
 |                 // The content is 4 bytes now, but the finishing , is missing
 | ||||||
|                 Err(nom::Err::Incomplete(nom::Needed::Size(1))), |                 Err(nom::Err::Incomplete(nom::Needed::Size(1))), | ||||||
|                     "{}", r"b4:fo," |                 "{}", | ||||||
|  |                 r"b4:fo," | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 binary()("b9:今日は,".as_bytes()), |                 binary()("b9:今日は,".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), T::Binary(Vec::from("今日は".as_bytes())))), |                 Ok(("".as_bytes(), T::Binary(Vec::from("今日は".as_bytes())))), | ||||||
|                 "{}", r"b9:今日は," |                 "{}", | ||||||
|  |                 r"b9:今日は," | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -590,25 +591,23 @@ pub mod parse { | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 list_t("[0:]".as_bytes()), |                 list_t("[0:]".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), vec![])), |                 Ok(("".as_bytes(), vec![])), | ||||||
|                 "{}", r"[0:]" |                 "{}", | ||||||
|  |                 r"[0:]" | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 list_t("[6:u,u,u,]".as_bytes()), |                 list_t("[6:u,u,u,]".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), vec![ |                 Ok(("".as_bytes(), vec![T::Unit, T::Unit, T::Unit,])), | ||||||
|                     T::Unit, |                 "{}", | ||||||
|                     T::Unit, |                 r"[6:u,u,u,]" | ||||||
|                     T::Unit, |  | ||||||
|                 ])), |  | ||||||
|                 "{}", r"[6:u,u,u,]" |  | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 list_t("[15:u,[7:t3:foo,]u,]".as_bytes()), |                 list_t("[15:u,[7:t3:foo,]u,]".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), vec![ |                 Ok(( | ||||||
|                     T::Unit, |                     "".as_bytes(), | ||||||
|                     T::List(vec![T::Text("foo".to_owned())]), |                     vec![T::Unit, T::List(vec![T::Text("foo".to_owned())]), T::Unit,] | ||||||
|                     T::Unit, |                 )), | ||||||
|                 ])), |                 "{}", | ||||||
|                 "{}", r"[15:u,[7:t3:foo,]u,]" |                 r"[15:u,[7:t3:foo,]u,]" | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -616,27 +615,40 @@ pub mod parse { | ||||||
|         fn test_record() { |         fn test_record() { | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 record_t("{21:<1:a|u,<1:b|u,<1:c|u,}".as_bytes()), |                 record_t("{21:<1:a|u,<1:b|u,<1:c|u,}".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), vec![ |                 Ok(( | ||||||
|  |                     "".as_bytes(), | ||||||
|  |                     vec![ | ||||||
|                         ("a".to_owned(), T::Unit), |                         ("a".to_owned(), T::Unit), | ||||||
|                         ("b".to_owned(), T::Unit), |                         ("b".to_owned(), T::Unit), | ||||||
|                         ("c".to_owned(), T::Unit), |                         ("c".to_owned(), T::Unit), | ||||||
|                 ].into_iter().collect::<HashMap<String, T>>())), |                     ] | ||||||
|                 "{}", r"{21:<1:a|u,<1:b|u,<1:c|u,}" |                     .into_iter() | ||||||
|  |                     .collect::<HashMap<String, T>>() | ||||||
|  |                 )), | ||||||
|  |                 "{}", | ||||||
|  |                 r"{21:<1:a|u,<1:b|u,<1:c|u,}" | ||||||
|             ); |             ); | ||||||
|             // duplicated keys are ignored (first is taken)
 |             // duplicated keys are ignored (first is taken)
 | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 record_t("{25:<1:a|u,<1:b|u,<1:a|i1:-1,}".as_bytes()), |                 record_t("{25:<1:a|u,<1:b|u,<1:a|i1:-1,}".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), vec![ |                 Ok(( | ||||||
|                     ("a".to_owned(), T::Unit), |                     "".as_bytes(), | ||||||
|                     ("b".to_owned(), T::Unit), |                     vec![("a".to_owned(), T::Unit), ("b".to_owned(), T::Unit),] | ||||||
|                 ].into_iter().collect::<HashMap<_,_>>())), |                         .into_iter() | ||||||
|                 "{}", r"{25:<1:a|u,<1:b|u,<1:a|i1:-1,}" |                         .collect::<HashMap<_, _>>() | ||||||
|  |                 )), | ||||||
|  |                 "{}", | ||||||
|  |                 r"{25:<1:a|u,<1:b|u,<1:a|i1:-1,}" | ||||||
|             ); |             ); | ||||||
|             // empty records are not allowed
 |             // empty records are not allowed
 | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 record_t("{0:}".as_bytes()), |                 record_t("{0:}".as_bytes()), | ||||||
|                 Err(nom::Err::Failure(("".as_bytes(), nom::error::ErrorKind::Many1))), |                 Err(nom::Err::Failure(( | ||||||
|                 "{}", r"{0:}" |                     "".as_bytes(), | ||||||
|  |                     nom::error::ErrorKind::Many1 | ||||||
|  |                 ))), | ||||||
|  |                 "{}", | ||||||
|  |                 r"{0:}" | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -645,37 +657,62 @@ pub mod parse { | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 t_t("n3:255,".as_bytes()), |                 t_t("n3:255,".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), T::N3(255))), |                 Ok(("".as_bytes(), T::N3(255))), | ||||||
|                 "{}", r"n3:255," |                 "{}", | ||||||
|  |                 r"n3:255," | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 t_t("t6:halloo,".as_bytes()), |                 t_t("t6:halloo,".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), T::Text("halloo".to_owned()))), |                 Ok(("".as_bytes(), T::Text("halloo".to_owned()))), | ||||||
|                 "{}", r"t6:halloo," |                 "{}", | ||||||
|  |                 r"t6:halloo," | ||||||
|             ); |             ); | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 t_t("<3:foo|t6:halloo,".as_bytes()), |                 t_t("<3:foo|t6:halloo,".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), T::Sum (Tag { |                 Ok(( | ||||||
|  |                     "".as_bytes(), | ||||||
|  |                     T::Sum(Tag { | ||||||
|                         tag: "foo".to_owned(), |                         tag: "foo".to_owned(), | ||||||
|                         val: Box::new(T::Text("halloo".to_owned())) |                         val: Box::new(T::Text("halloo".to_owned())) | ||||||
|                 }))), |                     }) | ||||||
|                 "{}", r"<3:foo|t6:halloo," |                 )), | ||||||
|  |                 "{}", | ||||||
|  |                 r"<3:foo|t6:halloo," | ||||||
|             ); |             ); | ||||||
|             // { a: Unit
 |             // { a: Unit
 | ||||||
|             // , foo: List <A: Unit | B: List i3> }
 |             // , foo: List <A: Unit | B: List i3> }
 | ||||||
|             assert_eq!( |             assert_eq!( | ||||||
|                 t_t("{52:<1:a|u,<3:foo|[33:<1:A|u,<1:A|n1:1,<1:B|[7:i3:127,]]}".as_bytes()), |                 t_t("{52:<1:a|u,<3:foo|[33:<1:A|u,<1:A|n1:1,<1:B|[7:i3:127,]]}".as_bytes()), | ||||||
|                 Ok(("".as_bytes(), T::Record(vec![ |                 Ok(( | ||||||
|  |                     "".as_bytes(), | ||||||
|  |                     T::Record( | ||||||
|  |                         vec![ | ||||||
|                             ("a".to_owned(), T::Unit), |                             ("a".to_owned(), T::Unit), | ||||||
|                     ("foo".to_owned(), T::List(vec![ |                             ( | ||||||
|                         T::Sum(Tag { tag: "A".to_owned(), val: Box::new(T::Unit) }), |                                 "foo".to_owned(), | ||||||
|                         T::Sum(Tag { tag: "A".to_owned(), val: Box::new(T::N1(true)) }), |                                 T::List(vec![ | ||||||
|                         T::Sum(Tag { tag: "B".to_owned(), val: Box::new(T::List(vec![T::I3(127)])) }), |                                     T::Sum(Tag { | ||||||
|                     ])) |                                         tag: "A".to_owned(), | ||||||
|                 ].into_iter().collect::<HashMap<String, T>>()))), |                                         val: Box::new(T::Unit) | ||||||
|                 "{}", r"{52:<1:a|u,<3:foo|[33:<1:A|u,<1:A|n1:1,<1:B|[7:i3:127,]]}" |                                     }), | ||||||
|  |                                     T::Sum(Tag { | ||||||
|  |                                         tag: "A".to_owned(), | ||||||
|  |                                         val: Box::new(T::N1(true)) | ||||||
|  |                                     }), | ||||||
|  |                                     T::Sum(Tag { | ||||||
|  |                                         tag: "B".to_owned(), | ||||||
|  |                                         val: Box::new(T::List(vec![T::I3(127)])) | ||||||
|  |                                     }), | ||||||
|  |                                 ]) | ||||||
|  |                             ) | ||||||
|  |                         ] | ||||||
|  |                         .into_iter() | ||||||
|  |                         .collect::<HashMap<String, T>>() | ||||||
|  |                     ) | ||||||
|  |                 )), | ||||||
|  |                 "{}", | ||||||
|  |                 r"{52:<1:a|u,<3:foo|[33:<1:A|u,<1:A|n1:1,<1:B|[7:i3:127,]]}" | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
| 
 |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -735,7 +772,10 @@ pub mod dec { | ||||||
|         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { |         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { | ||||||
|             match u { |             match u { | ||||||
|                 U::Binary(b) => Ok(b), |                 U::Binary(b) => Ok(b), | ||||||
|                 other => Err(DecodeError(format!("Cannot decode {:?} into Binary", other))), |                 other => Err(DecodeError(format!( | ||||||
|  |                     "Cannot decode {:?} into Binary", | ||||||
|  |                     other | ||||||
|  |                 ))), | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -766,16 +806,17 @@ pub mod dec { | ||||||
|     pub struct Record<T>(pub T); |     pub struct Record<T>(pub T); | ||||||
| 
 | 
 | ||||||
|     impl<'a, Inner> Decoder<'a> for Record<Inner> |     impl<'a, Inner> Decoder<'a> for Record<Inner> | ||||||
|         where Inner: Decoder<'a> |     where | ||||||
|  |         Inner: Decoder<'a>, | ||||||
|     { |     { | ||||||
|         type A = HashMap<&'a str, Inner::A>; |         type A = HashMap<&'a str, Inner::A>; | ||||||
|         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { |         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { | ||||||
|             match u { |             match u { | ||||||
|                 U::Record(map) => |                 U::Record(map) => map | ||||||
|                     map.into_iter() |                     .into_iter() | ||||||
|                     .map(|(k, v)| self.0.dec(v).map(|v2| (k, v2))) |                     .map(|(k, v)| self.0.dec(v).map(|v2| (k, v2))) | ||||||
|                     .collect::<Result<Self::A, _>>(), |                     .collect::<Result<Self::A, _>>(), | ||||||
|                 o => Err(DecodeError(format!("Cannot decode {:?} into record", o))) |                 o => Err(DecodeError(format!("Cannot decode {:?} into record", o))), | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -784,18 +825,22 @@ pub mod dec { | ||||||
|     #[derive(Clone, Copy)] |     #[derive(Clone, Copy)] | ||||||
|     pub struct RecordDot<'a, T> { |     pub struct RecordDot<'a, T> { | ||||||
|         pub field: &'a str, |         pub field: &'a str, | ||||||
|         pub inner: T |         pub inner: T, | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     impl<'a, Inner> Decoder<'a> for RecordDot<'_, Inner> |     impl<'a, Inner> Decoder<'a> for RecordDot<'_, Inner> | ||||||
|         where Inner: Decoder<'a> + Clone |     where | ||||||
|  |         Inner: Decoder<'a> + Clone, | ||||||
|     { |     { | ||||||
|         type A = Inner::A; |         type A = Inner::A; | ||||||
|         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { |         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { | ||||||
|             match Record(self.inner.clone()).dec(u) { |             match Record(self.inner.clone()).dec(u) { | ||||||
|                 Ok(mut map) => match map.remove(self.field) { |                 Ok(mut map) => match map.remove(self.field) { | ||||||
|                     Some(inner) => Ok(inner), |                     Some(inner) => Ok(inner), | ||||||
|                     None => Err(DecodeError(format!("Cannot find `{}` in record map", self.field))), |                     None => Err(DecodeError(format!( | ||||||
|  |                         "Cannot find `{}` in record map", | ||||||
|  |                         self.field | ||||||
|  |                     ))), | ||||||
|                 }, |                 }, | ||||||
|                 Err(err) => Err(err), |                 Err(err) => Err(err), | ||||||
|             } |             } | ||||||
|  | @ -810,17 +855,21 @@ pub mod dec { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     impl<'a, Inner> Decoder<'a> for OneOf<Inner, Inner::A> |     impl<'a, Inner> Decoder<'a> for OneOf<Inner, Inner::A> | ||||||
|         where Inner: Decoder<'a>, |     where | ||||||
|               Inner::A: Display + Debug + PartialEq |         Inner: Decoder<'a>, | ||||||
|  |         Inner::A: Display + Debug + PartialEq, | ||||||
|     { |     { | ||||||
|         type A = Inner::A; |         type A = Inner::A; | ||||||
|         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { |         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { | ||||||
|             match self.inner.dec(u) { |             match self.inner.dec(u) { | ||||||
|                 Ok(inner) => match self.list.iter().any(|x| x.eq(&inner)) { |                 Ok(inner) => match self.list.iter().any(|x| x.eq(&inner)) { | ||||||
|                     true => Ok(inner), |                     true => Ok(inner), | ||||||
|                     false => Err(DecodeError(format!("{} is not one of {:?}", inner, self.list))) |                     false => Err(DecodeError(format!( | ||||||
|  |                         "{} is not one of {:?}", | ||||||
|  |                         inner, self.list | ||||||
|  |                     ))), | ||||||
|                 }, |                 }, | ||||||
|                 Err(err) => Err(err) |                 Err(err) => Err(err), | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -830,15 +879,15 @@ pub mod dec { | ||||||
|     pub struct Try<T>(pub T); |     pub struct Try<T>(pub T); | ||||||
| 
 | 
 | ||||||
|     impl<'a, Inner> Decoder<'a> for Try<Inner> |     impl<'a, Inner> Decoder<'a> for Try<Inner> | ||||||
|         where Inner: Decoder<'a> |     where | ||||||
|  |         Inner: Decoder<'a>, | ||||||
|     { |     { | ||||||
|         type A = Option<Inner::A>; |         type A = Option<Inner::A>; | ||||||
|         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { |         fn dec(&self, u: U<'a>) -> Result<Self::A, DecodeError> { | ||||||
|             match self.0.dec(u) { |             match self.0.dec(u) { | ||||||
|                 Ok(inner) => Ok(Some(inner)), |                 Ok(inner) => Ok(Some(inner)), | ||||||
|                 Err(err) => Ok(None) |                 Err(err) => Ok(None), | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,6 +1,6 @@ | ||||||
| extern crate netencode; | extern crate netencode; | ||||||
| 
 | 
 | ||||||
| use netencode::{U, T, Tag}; | use netencode::{Tag, T, U}; | ||||||
| 
 | 
 | ||||||
| pub enum Pretty { | pub enum Pretty { | ||||||
|     Single { |     Single { | ||||||
|  | @ -20,7 +20,7 @@ pub enum Pretty { | ||||||
|         r#type: char, |         r#type: char, | ||||||
|         length: String, |         length: String, | ||||||
|         vals: Vec<Pretty>, |         vals: Vec<Pretty>, | ||||||
|         trailer: char |         trailer: char, | ||||||
|     }, |     }, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -39,7 +39,7 @@ impl Pretty { | ||||||
|                 r#type: 't', |                 r#type: 't', | ||||||
|                 length: format!("{}:", s.len()), |                 length: format!("{}:", s.len()), | ||||||
|                 val: s.to_string(), |                 val: s.to_string(), | ||||||
|                 trailer: ',' |                 trailer: ',', | ||||||
|             }, |             }, | ||||||
|             U::Binary(s) => Pretty::Single { |             U::Binary(s) => Pretty::Single { | ||||||
|                 r#type: 'b', |                 r#type: 'b', | ||||||
|  | @ -47,15 +47,18 @@ impl Pretty { | ||||||
|                 // For pretty printing we want the string to be visible obviously.
 |                 // For pretty printing we want the string to be visible obviously.
 | ||||||
|                 // Instead of not supporting binary, let’s use lossy conversion.
 |                 // Instead of not supporting binary, let’s use lossy conversion.
 | ||||||
|                 val: String::from_utf8_lossy(s).into_owned(), |                 val: String::from_utf8_lossy(s).into_owned(), | ||||||
|                 trailer: ',' |                 trailer: ',', | ||||||
|             }, |             }, | ||||||
|             U::Sum(Tag { tag, val }) => Self::pretty_tag(tag, Self::from_u(*val)), |             U::Sum(Tag { tag, val }) => Self::pretty_tag(tag, Self::from_u(*val)), | ||||||
|             U::Record(m) => Pretty::Multi { |             U::Record(m) => Pretty::Multi { | ||||||
|                 r#type: '{', |                 r#type: '{', | ||||||
|                 // TODO: we are losing the size here, should we recompute it? Keep it?
 |                 // TODO: we are losing the size here, should we recompute it? Keep it?
 | ||||||
|                 length: String::from(""), |                 length: String::from(""), | ||||||
|                 vals: m.into_iter().map(|(k, v)| Self::pretty_tag(k, Self::from_u(v))).collect(), |                 vals: m | ||||||
|                 trailer: '}' |                     .into_iter() | ||||||
|  |                     .map(|(k, v)| Self::pretty_tag(k, Self::from_u(v))) | ||||||
|  |                     .collect(), | ||||||
|  |                 trailer: '}', | ||||||
|             }, |             }, | ||||||
|             U::List(l) => Pretty::Multi { |             U::List(l) => Pretty::Multi { | ||||||
|                 r#type: '[', |                 r#type: '[', | ||||||
|  | @ -68,13 +71,14 @@ impl Pretty { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn scalar<D>(r#type: char, length: &str, d: D) -> Pretty |     fn scalar<D>(r#type: char, length: &str, d: D) -> Pretty | ||||||
|     where D: std::fmt::Display |     where | ||||||
|  |         D: std::fmt::Display, | ||||||
|     { |     { | ||||||
|         Pretty::Single { |         Pretty::Single { | ||||||
|             r#type, |             r#type, | ||||||
|             length: length.to_string(), |             length: length.to_string(), | ||||||
|             val: format!("{}", d), |             val: format!("{}", d), | ||||||
|             trailer: ',' |             trailer: ',', | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -89,43 +93,62 @@ impl Pretty { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn print_multiline<W>(&self, mut w: &mut W) -> std::io::Result<()> |     pub fn print_multiline<W>(&self, mut w: &mut W) -> std::io::Result<()> | ||||||
|         where W: std::io::Write |     where | ||||||
|  |         W: std::io::Write, | ||||||
|     { |     { | ||||||
|         Self::go(&mut w, self, 0, true); |         Self::go(&mut w, self, 0, true); | ||||||
|         write!(w, "\n") |         write!(w, "\n") | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn go<W>(mut w: &mut W, p: &Pretty, depth: usize, is_newline: bool) -> std::io::Result<()> |     fn go<W>(mut w: &mut W, p: &Pretty, depth: usize, is_newline: bool) -> std::io::Result<()> | ||||||
|         where W: std::io::Write |     where | ||||||
|  |         W: std::io::Write, | ||||||
|     { |     { | ||||||
|         const full: usize = 4; |         const full: usize = 4; | ||||||
|         const half: usize = 2; |         const half: usize = 2; | ||||||
|         let i = &vec![b' '; depth * full]; |         let i = &vec![b' '; depth * full]; | ||||||
|         let iandhalf = &vec![b' '; depth * full + half]; |         let iandhalf = &vec![b' '; depth * full + half]; | ||||||
|         let (i, iandhalf) = unsafe {( |         let (i, iandhalf) = unsafe { | ||||||
|  |             ( | ||||||
|                 std::str::from_utf8_unchecked(i), |                 std::str::from_utf8_unchecked(i), | ||||||
|                 std::str::from_utf8_unchecked(iandhalf), |                 std::str::from_utf8_unchecked(iandhalf), | ||||||
|         )}; |             ) | ||||||
|  |         }; | ||||||
|         if is_newline { |         if is_newline { | ||||||
|             write!(&mut w, "{}", i); |             write!(&mut w, "{}", i); | ||||||
|         } |         } | ||||||
|         match p { |         match p { | ||||||
|             Pretty::Single {r#type, length, val, trailer} => |             Pretty::Single { | ||||||
|                 write!(&mut w, "{} {}{}", r#type, val, trailer), |                 r#type, | ||||||
|             Pretty::Tag { r#type, length, key, inner, val } => { |                 length, | ||||||
|  |                 val, | ||||||
|  |                 trailer, | ||||||
|  |             } => write!(&mut w, "{} {}{}", r#type, val, trailer), | ||||||
|  |             Pretty::Tag { | ||||||
|  |                 r#type, | ||||||
|  |                 length, | ||||||
|  |                 key, | ||||||
|  |                 inner, | ||||||
|  |                 val, | ||||||
|  |             } => { | ||||||
|                 write!(&mut w, "{} {} {}", r#type, key, inner)?; |                 write!(&mut w, "{} {} {}", r#type, key, inner)?; | ||||||
|                 Self::go::<W>(&mut w, val, depth, false) |                 Self::go::<W>(&mut w, val, depth, false) | ||||||
|             }, |             } | ||||||
|             // if the length is 0 or 1, we print on one line,
 |             // if the length is 0 or 1, we print on one line,
 | ||||||
|             // only if there’s more than one element we split the resulting value.
 |             // only if there’s more than one element we split the resulting value.
 | ||||||
|             // we never break lines on arbitrary column sizes, since that is just silly.
 |             // we never break lines on arbitrary column sizes, since that is just silly.
 | ||||||
|             Pretty::Multi {r#type, length, vals, trailer} => match vals.len() { |             Pretty::Multi { | ||||||
|  |                 r#type, | ||||||
|  |                 length, | ||||||
|  |                 vals, | ||||||
|  |                 trailer, | ||||||
|  |             } => match vals.len() { | ||||||
|                 0 => write!(&mut w, "{} {}", r#type, trailer), |                 0 => write!(&mut w, "{} {}", r#type, trailer), | ||||||
|                 1 => { |                 1 => { | ||||||
|                     write!(&mut w, "{} ", r#type); |                     write!(&mut w, "{} ", r#type); | ||||||
|                     Self::go::<W>(&mut w, &vals[0], depth, false)?; |                     Self::go::<W>(&mut w, &vals[0], depth, false)?; | ||||||
|                     write!(&mut w, "{}", trailer) |                     write!(&mut w, "{}", trailer) | ||||||
|                 }, |                 } | ||||||
|                 more => { |                 more => { | ||||||
|                     write!(&mut w, "\n{}{} \n", iandhalf, r#type)?; |                     write!(&mut w, "\n{}{} \n", iandhalf, r#type)?; | ||||||
|                     for v in vals { |                     for v in vals { | ||||||
|  |  | ||||||
|  | @ -1,35 +1,33 @@ | ||||||
| extern crate httparse; |  | ||||||
| extern crate netencode; |  | ||||||
| extern crate arglib_netencode; | extern crate arglib_netencode; | ||||||
| extern crate ascii; | extern crate ascii; | ||||||
| extern crate exec_helpers; | extern crate exec_helpers; | ||||||
|  | extern crate httparse; | ||||||
|  | extern crate netencode; | ||||||
| 
 | 
 | ||||||
| use std::os::unix::io::FromRawFd; | use exec_helpers::{die_expected_error, die_temporary, die_user_error}; | ||||||
| use std::io::Read; |  | ||||||
| use std::io::Write; |  | ||||||
| use std::collections::HashMap; | use std::collections::HashMap; | ||||||
| use exec_helpers::{die_user_error, die_expected_error, die_temporary}; | use std::io::{Read, Write}; | ||||||
|  | use std::os::unix::io::FromRawFd; | ||||||
| 
 | 
 | ||||||
| use netencode::{U, T, dec}; |  | ||||||
| use netencode::dec::Decoder; | use netencode::dec::Decoder; | ||||||
|  | use netencode::{dec, T, U}; | ||||||
| 
 | 
 | ||||||
| enum What { | enum What { | ||||||
|     Request, |     Request, | ||||||
|     Response |     Response, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // reads a http request (stdin), and writes all headers to stdout, as netencoded record.
 | // reads a http request (stdin), and writes all headers to stdout, as netencoded record.
 | ||||||
| // The keys are text, but can be lists of text iff headers appear multiple times, so beware.
 | // The keys are text, but can be lists of text iff headers appear multiple times, so beware.
 | ||||||
| fn main() -> std::io::Result<()> { | fn main() -> std::io::Result<()> { | ||||||
| 
 |  | ||||||
|     exec_helpers::no_args("read-http"); |     exec_helpers::no_args("read-http"); | ||||||
| 
 | 
 | ||||||
|     let args = dec::RecordDot { |     let args = dec::RecordDot { | ||||||
|         field: "what", |         field: "what", | ||||||
|         inner: dec::OneOf { |         inner: dec::OneOf { | ||||||
|             list: vec!["request", "response"], |             list: vec!["request", "response"], | ||||||
|             inner: dec::Text |             inner: dec::Text, | ||||||
|         } |         }, | ||||||
|     }; |     }; | ||||||
|     let what: What = match args.dec(arglib_netencode::arglib_netencode("read-http", None).to_u()) { |     let what: What = match args.dec(arglib_netencode::arglib_netencode("read-http", None).to_u()) { | ||||||
|         Ok("request") => What::Request, |         Ok("request") => What::Request, | ||||||
|  | @ -39,7 +37,8 @@ fn main() -> std::io::Result<()> { | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     fn read_stdin_to_complete<F>(mut parse: F) -> () |     fn read_stdin_to_complete<F>(mut parse: F) -> () | ||||||
|         where F: FnMut(&[u8]) -> httparse::Result<usize> |     where | ||||||
|  |         F: FnMut(&[u8]) -> httparse::Result<usize>, | ||||||
|     { |     { | ||||||
|         let mut res = httparse::Status::Partial; |         let mut res = httparse::Status::Partial; | ||||||
|         loop { |         loop { | ||||||
|  | @ -48,16 +47,22 @@ fn main() -> std::io::Result<()> { | ||||||
|             } |             } | ||||||
|             let mut buf = [0; 2048]; |             let mut buf = [0; 2048]; | ||||||
|             match std::io::stdin().read(&mut buf[..]) { |             match std::io::stdin().read(&mut buf[..]) { | ||||||
|                 Ok(size) => if size == 0 { |                 Ok(size) => { | ||||||
|  |                     if size == 0 { | ||||||
|                         break; |                         break; | ||||||
|                 }, |                     } | ||||||
|                 Err(err) => die_temporary("read-http", format!("could not read from stdin, {:?}", err)) |                 } | ||||||
|  |                 Err(err) => { | ||||||
|  |                     die_temporary("read-http", format!("could not read from stdin, {:?}", err)) | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             match parse(&buf) { |             match parse(&buf) { | ||||||
|                 Ok(status) => { |                 Ok(status) => { | ||||||
|                     res = status; |                     res = status; | ||||||
|                 } |                 } | ||||||
|                 Err(err) => die_temporary("read-http", format!("httparse parsing failed: {:#?}", err)) |                 Err(err) => { | ||||||
|  |                     die_temporary("read-http", format!("httparse parsing failed: {:#?}", err)) | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -66,7 +71,10 @@ fn main() -> std::io::Result<()> { | ||||||
|         let mut res = HashMap::new(); |         let mut res = HashMap::new(); | ||||||
|         for httparse::Header { name, value } in headers { |         for httparse::Header { name, value } in headers { | ||||||
|             let val = ascii::AsciiStr::from_ascii(*value) |             let val = ascii::AsciiStr::from_ascii(*value) | ||||||
|                 .expect(&format!("read-http: we require header values to be ASCII, but the header {} was {:?}", name, value)) |                 .expect(&format!( | ||||||
|  |                     "read-http: we require header values to be ASCII, but the header {} was {:?}", | ||||||
|  |                     name, value | ||||||
|  |                 )) | ||||||
|                 .as_str(); |                 .as_str(); | ||||||
|             // lowercase the header names, since the standard doesn’t care
 |             // lowercase the header names, since the standard doesn’t care
 | ||||||
|             // and we want unique strings to match against
 |             // and we want unique strings to match against
 | ||||||
|  | @ -77,13 +85,13 @@ fn main() -> std::io::Result<()> { | ||||||
|                     let name_lower = name.to_lowercase(); |                     let name_lower = name.to_lowercase(); | ||||||
|                     let _ = res.insert(name_lower, U::List(vec![U::Text(t), U::Text(val)])); |                     let _ = res.insert(name_lower, U::List(vec![U::Text(t), U::Text(val)])); | ||||||
|                     () |                     () | ||||||
|                 }, |                 } | ||||||
|                 Some(U::List(mut l)) => { |                 Some(U::List(mut l)) => { | ||||||
|                     let name_lower = name.to_lowercase(); |                     let name_lower = name.to_lowercase(); | ||||||
|                     l.push(U::Text(val)); |                     l.push(U::Text(val)); | ||||||
|                     let _ = res.insert(name_lower, U::List(l)); |                     let _ = res.insert(name_lower, U::List(l)); | ||||||
|                     () |                     () | ||||||
|                 }, |                 } | ||||||
|                 Some(o) => panic!("read-http: header not text nor list: {:?}", o), |                 Some(o) => panic!("read-http: header not text nor list: {:?}", o), | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  | @ -101,9 +109,11 @@ fn main() -> std::io::Result<()> { | ||||||
|                     if chunk.windows(4).any(|c| c == b"\r\n\r\n") { |                     if chunk.windows(4).any(|c| c == b"\r\n\r\n") { | ||||||
|                         return Some(()); |                         return Some(()); | ||||||
|                     } |                     } | ||||||
|                 }, |                 } | ||||||
|                 Some(Err(err)) => die_temporary("read-http", format!("error reading from stdin: {:?}", err)), |                 Some(Err(err)) => { | ||||||
|                 None => return None |                     die_temporary("read-http", format!("error reading from stdin: {:?}", err)) | ||||||
|  |                 } | ||||||
|  |                 None => return None, | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -118,66 +128,97 @@ fn main() -> std::io::Result<()> { | ||||||
|             let mut buf: Vec<u8> = vec![]; |             let mut buf: Vec<u8> = vec![]; | ||||||
|             match read_till_end_of_header(&mut buf, stdin.lock()) { |             match read_till_end_of_header(&mut buf, stdin.lock()) { | ||||||
|                 Some(()) => match req.parse(&buf) { |                 Some(()) => match req.parse(&buf) { | ||||||
|                     Ok(httparse::Status::Complete(_body_start)) => {}, |                     Ok(httparse::Status::Complete(_body_start)) => {} | ||||||
|                     Ok(httparse::Status::Partial) => die_expected_error("read-http", "httparse should have gotten a full header"), |                     Ok(httparse::Status::Partial) => { | ||||||
|                     Err(err) => die_expected_error("read-http", format!("httparse response parsing failed: {:#?}", err)) |                         die_expected_error("read-http", "httparse should have gotten a full header") | ||||||
|  |                     } | ||||||
|  |                     Err(err) => die_expected_error( | ||||||
|  |                         "read-http", | ||||||
|  |                         format!("httparse response parsing failed: {:#?}", err), | ||||||
|  |                     ), | ||||||
|                 }, |                 }, | ||||||
|                 None => die_expected_error("read-http", format!("httparse end of stdin reached before able to parse request headers")) |                 None => die_expected_error( | ||||||
|  |                     "read-http", | ||||||
|  |                     format!("httparse end of stdin reached before able to parse request headers"), | ||||||
|  |                 ), | ||||||
|             } |             } | ||||||
|             let method = req.method.expect("method must be filled on complete parse"); |             let method = req.method.expect("method must be filled on complete parse"); | ||||||
|             let path = req.path.expect("path must be filled on complete parse"); |             let path = req.path.expect("path must be filled on complete parse"); | ||||||
|             write_dict_req(method, path, &normalize_headers(req.headers)) |             write_dict_req(method, path, &normalize_headers(req.headers)) | ||||||
|         }, |         } | ||||||
|         Response => { |         Response => { | ||||||
|             let mut resp = httparse::Response::new(&mut headers); |             let mut resp = httparse::Response::new(&mut headers); | ||||||
|             let mut buf: Vec<u8> = vec![]; |             let mut buf: Vec<u8> = vec![]; | ||||||
|             match read_till_end_of_header(&mut buf, stdin.lock()) { |             match read_till_end_of_header(&mut buf, stdin.lock()) { | ||||||
|                 Some(()) => match resp.parse(&buf) { |                 Some(()) => match resp.parse(&buf) { | ||||||
|                     Ok(httparse::Status::Complete(_body_start)) => {}, |                     Ok(httparse::Status::Complete(_body_start)) => {} | ||||||
|                     Ok(httparse::Status::Partial) => die_expected_error("read-http", "httparse should have gotten a full header"), |                     Ok(httparse::Status::Partial) => { | ||||||
|                     Err(err) => die_expected_error("read-http", format!("httparse response parsing failed: {:#?}", err)) |                         die_expected_error("read-http", "httparse should have gotten a full header") | ||||||
|  |                     } | ||||||
|  |                     Err(err) => die_expected_error( | ||||||
|  |                         "read-http", | ||||||
|  |                         format!("httparse response parsing failed: {:#?}", err), | ||||||
|  |                     ), | ||||||
|                 }, |                 }, | ||||||
|                 None => die_expected_error("read-http", format!("httparse end of stdin reached before able to parse response headers")) |                 None => die_expected_error( | ||||||
|  |                     "read-http", | ||||||
|  |                     format!("httparse end of stdin reached before able to parse response headers"), | ||||||
|  |                 ), | ||||||
|             } |             } | ||||||
|             let code = resp.code.expect("code must be filled on complete parse"); |             let code = resp.code.expect("code must be filled on complete parse"); | ||||||
|             let reason = resp.reason.expect("reason must be filled on complete parse"); |             let reason = resp | ||||||
|  |                 .reason | ||||||
|  |                 .expect("reason must be filled on complete parse"); | ||||||
|             write_dict_resp(code, reason, &normalize_headers(resp.headers)) |             write_dict_resp(code, reason, &normalize_headers(resp.headers)) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn write_dict_req<'a, 'buf>(method: &'buf str, path: &'buf str, headers: &'a HashMap<String, U<'a>>) -> std::io::Result<()> { | fn write_dict_req<'a, 'buf>( | ||||||
|     let mut http = vec![ |     method: &'buf str, | ||||||
|         ("method", U::Text(method)), |     path: &'buf str, | ||||||
|         ("path", U::Text(path)), |     headers: &'a HashMap<String, U<'a>>, | ||||||
|     ].into_iter().collect(); | ) -> std::io::Result<()> { | ||||||
|  |     let mut http = vec![("method", U::Text(method)), ("path", U::Text(path))] | ||||||
|  |         .into_iter() | ||||||
|  |         .collect(); | ||||||
|     write_dict(http, headers) |     write_dict(http, headers) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn write_dict_resp<'a, 'buf>(code: u16, reason: &'buf str, headers: &'a HashMap<String, U<'a>>) -> std::io::Result<()> { | fn write_dict_resp<'a, 'buf>( | ||||||
|  |     code: u16, | ||||||
|  |     reason: &'buf str, | ||||||
|  |     headers: &'a HashMap<String, U<'a>>, | ||||||
|  | ) -> std::io::Result<()> { | ||||||
|     let mut http = vec![ |     let mut http = vec![ | ||||||
|         ("status", U::N6(code as u64)), |         ("status", U::N6(code as u64)), | ||||||
|         ("status-text", U::Text(reason)), |         ("status-text", U::Text(reason)), | ||||||
|     ].into_iter().collect(); |     ] | ||||||
|  |     .into_iter() | ||||||
|  |     .collect(); | ||||||
|     write_dict(http, headers) |     write_dict(http, headers) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | fn write_dict<'buf, 'a>( | ||||||
| fn write_dict<'buf, 'a>(mut http: HashMap<&str, U<'a>>, headers: &'a HashMap<String, U<'a>>) -> std::io::Result<()> { |     mut http: HashMap<&str, U<'a>>, | ||||||
|     match http.insert("headers", U::Record( |     headers: &'a HashMap<String, U<'a>>, | ||||||
|         headers.iter().map(|(k,v)| (k.as_str(), v.clone())).collect() | ) -> std::io::Result<()> { | ||||||
|     )) { |     match http.insert( | ||||||
|  |         "headers", | ||||||
|  |         U::Record( | ||||||
|  |             headers | ||||||
|  |                 .iter() | ||||||
|  |                 .map(|(k, v)| (k.as_str(), v.clone())) | ||||||
|  |                 .collect(), | ||||||
|  |         ), | ||||||
|  |     ) { | ||||||
|         None => (), |         None => (), | ||||||
|         Some(_) => panic!("read-http: headers already in dict"), |         Some(_) => panic!("read-http: headers already in dict"), | ||||||
|     }; |     }; | ||||||
|     netencode::encode( |     netencode::encode(&mut std::io::stdout(), &U::Record(http))?; | ||||||
|         &mut std::io::stdout(), |  | ||||||
|         &U::Record(http) |  | ||||||
|     )?; |  | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| // iter helper
 | // iter helper
 | ||||||
| 
 | 
 | ||||||
| struct Chunkyboi<T> { | struct Chunkyboi<T> { | ||||||
|  | @ -188,10 +229,7 @@ struct Chunkyboi<T> { | ||||||
| impl<R: Read> Chunkyboi<R> { | impl<R: Read> Chunkyboi<R> { | ||||||
|     fn new(inner: R, chunksize: usize) -> Self { |     fn new(inner: R, chunksize: usize) -> Self { | ||||||
|         let buf = vec![0; chunksize]; |         let buf = vec![0; chunksize]; | ||||||
|         Chunkyboi { |         Chunkyboi { inner, buf } | ||||||
|             inner, |  | ||||||
|             buf |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -205,7 +243,7 @@ impl<R: Read> Iterator for Chunkyboi<R> { | ||||||
|                 // clone a new buffer so we can reuse the internal one
 |                 // clone a new buffer so we can reuse the internal one
 | ||||||
|                 Some(Ok(self.buf[..read].to_owned())) |                 Some(Ok(self.buf[..read].to_owned())) | ||||||
|             } |             } | ||||||
|             Err(err) => Some(Err(err)) |             Err(err) => Some(Err(err)), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,10 +1,6 @@ | ||||||
| use clap::Clap; | use clap::Clap; | ||||||
| 
 | 
 | ||||||
| use crate::codegen; | use crate::{codegen, interpreter, parser, tc, Result}; | ||||||
| use crate::interpreter; |  | ||||||
| use crate::parser; |  | ||||||
| use crate::tc; |  | ||||||
| use crate::Result; |  | ||||||
| 
 | 
 | ||||||
| /// Evaluate an expression and print its result
 | /// Evaluate an expression and print its result
 | ||||||
| #[derive(Clap)] | #[derive(Clap)] | ||||||
|  |  | ||||||
|  | @ -1,9 +1,8 @@ | ||||||
| use std::borrow::Cow; | use std::borrow::Cow; | ||||||
| 
 | 
 | ||||||
| use nom::alt; |  | ||||||
| use nom::character::complete::{digit1, multispace0, multispace1}; | use nom::character::complete::{digit1, multispace0, multispace1}; | ||||||
| use nom::{ | use nom::{ | ||||||
|     call, char, complete, delimited, do_parse, flat_map, many0, map, named, opt, parse_to, |     alt, call, char, complete, delimited, do_parse, flat_map, many0, map, named, opt, parse_to, | ||||||
|     preceded, separated_list0, separated_list1, tag, tuple, |     preceded, separated_list0, separated_list1, tag, tuple, | ||||||
| }; | }; | ||||||
| use pratt::{Affix, Associativity, PrattParser, Precedence}; | use pratt::{Affix, Associativity, PrattParser, Precedence}; | ||||||
|  |  | ||||||
|  | @ -13,11 +13,8 @@ use futures::Future; | ||||||
| use metrics_exporter_prometheus::PrometheusBuilder; | use metrics_exporter_prometheus::PrometheusBuilder; | ||||||
| use nix::pty::Winsize; | use nix::pty::Winsize; | ||||||
| use pty::ChildHandle; | use pty::ChildHandle; | ||||||
| use thrussh::ChannelId; | use thrussh::server::{self, Auth, Session}; | ||||||
| use thrussh::{ | use thrussh::{ChannelId, CryptoVec}; | ||||||
|     server::{self, Auth, Session}, |  | ||||||
|     CryptoVec, |  | ||||||
| }; |  | ||||||
| use thrussh_keys::decode_secret_key; | use thrussh_keys::decode_secret_key; | ||||||
| use thrussh_keys::key::KeyPair; | use thrussh_keys::key::KeyPair; | ||||||
| use tokio::fs::File; | use tokio::fs::File; | ||||||
|  |  | ||||||
|  | @ -6,8 +6,7 @@ use std::task::{Context, Poll}; | ||||||
| 
 | 
 | ||||||
| use eyre::{bail, Result}; | use eyre::{bail, Result}; | ||||||
| use futures::Future; | use futures::Future; | ||||||
| use nix::pty::forkpty; | use nix::pty::{forkpty, Winsize}; | ||||||
| use nix::pty::Winsize; |  | ||||||
| use nix::sys::termios::Termios; | use nix::sys::termios::Termios; | ||||||
| use nix::sys::wait::{waitpid, WaitPidFlag, WaitStatus}; | use nix::sys::wait::{waitpid, WaitPidFlag, WaitStatus}; | ||||||
| use nix::unistd::{ForkResult, Pid}; | use nix::unistd::{ForkResult, Pid}; | ||||||
|  |  | ||||||
|  | @ -28,14 +28,19 @@ fn main() { | ||||||
| 
 | 
 | ||||||
|     // Otherwise ask Nix to build it and inject the result.
 |     // Otherwise ask Nix to build it and inject the result.
 | ||||||
|     let output = Command::new("nix-build") |     let output = Command::new("nix-build") | ||||||
|         .arg("-A").arg("web.atward.indexHtml") |         .arg("-A") | ||||||
|  |         .arg("web.atward.indexHtml") | ||||||
|         // ... assuming atward is at //web/atward ...
 |         // ... assuming atward is at //web/atward ...
 | ||||||
|         .arg("../..") |         .arg("../..") | ||||||
|         .output() |         .output() | ||||||
|         .expect(ERROR_MESSAGE); |         .expect(ERROR_MESSAGE); | ||||||
| 
 | 
 | ||||||
|     if !output.status.success() { |     if !output.status.success() { | ||||||
|         eprintln!("{}\nNix output: {}", ERROR_MESSAGE, String::from_utf8_lossy(&output.stderr)); |         eprintln!( | ||||||
|  |             "{}\nNix output: {}", | ||||||
|  |             ERROR_MESSAGE, | ||||||
|  |             String::from_utf8_lossy(&output.stderr) | ||||||
|  |         ); | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -19,13 +19,13 @@ | ||||||
| //! This module implements the database executor, which holds the
 | //! This module implements the database executor, which holds the
 | ||||||
| //! database connection and performs queries on it.
 | //! database connection and performs queries on it.
 | ||||||
| 
 | 
 | ||||||
| use actix::prelude::*; |  | ||||||
| use diesel::{self, sql_query}; |  | ||||||
| use diesel::sql_types::Text; |  | ||||||
| use diesel::prelude::*; |  | ||||||
| use diesel::r2d2::{Pool, ConnectionManager}; |  | ||||||
| use crate::models::*; |  | ||||||
| use crate::errors::{ConverseError, Result}; | use crate::errors::{ConverseError, Result}; | ||||||
|  | use crate::models::*; | ||||||
|  | use actix::prelude::*; | ||||||
|  | use diesel::prelude::*; | ||||||
|  | use diesel::r2d2::{ConnectionManager, Pool}; | ||||||
|  | use diesel::sql_types::Text; | ||||||
|  | use diesel::{self, sql_query}; | ||||||
| 
 | 
 | ||||||
| /// Raw PostgreSQL query used to perform full-text search on posts
 | /// Raw PostgreSQL query used to perform full-text search on posts
 | ||||||
| /// with a supplied phrase. For now, the query language is hardcoded
 | /// with a supplied phrase. For now, the query language is hardcoded
 | ||||||
|  | @ -50,14 +50,12 @@ pub struct DbExecutor(pub Pool<ConnectionManager<PgConnection>>); | ||||||
| 
 | 
 | ||||||
| impl DbExecutor { | impl DbExecutor { | ||||||
|     /// Request a list of threads.
 |     /// Request a list of threads.
 | ||||||
|     //
 |  | ||||||
|     // TODO(tazjin): This should support pagination.
 |     // TODO(tazjin): This should support pagination.
 | ||||||
|     pub fn list_threads(&self) -> Result<Vec<ThreadIndex>> { |     pub fn list_threads(&self) -> Result<Vec<ThreadIndex>> { | ||||||
|         use crate::schema::thread_index::dsl::*; |         use crate::schema::thread_index::dsl::*; | ||||||
| 
 | 
 | ||||||
|         let conn = self.0.get()?; |         let conn = self.0.get()?; | ||||||
|         let results = thread_index |         let results = thread_index.load::<ThreadIndex>(&conn)?; | ||||||
|             .load::<ThreadIndex>(&conn)?; |  | ||||||
|         Ok(results) |         Ok(results) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -69,9 +67,7 @@ impl DbExecutor { | ||||||
| 
 | 
 | ||||||
|         let conn = self.0.get()?; |         let conn = self.0.get()?; | ||||||
| 
 | 
 | ||||||
|         let opt_user = users |         let opt_user = users.filter(email.eq(email)).first(&conn).optional()?; | ||||||
|             .filter(email.eq(email)) |  | ||||||
|             .first(&conn).optional()?; |  | ||||||
| 
 | 
 | ||||||
|         if let Some(user) = opt_user { |         if let Some(user) = opt_user { | ||||||
|             Ok(user) |             Ok(user) | ||||||
|  | @ -93,12 +89,11 @@ impl DbExecutor { | ||||||
| 
 | 
 | ||||||
|     /// Fetch a specific thread and return it with its posts.
 |     /// Fetch a specific thread and return it with its posts.
 | ||||||
|     pub fn get_thread(&self, thread_id: i32) -> Result<(Thread, Vec<SimplePost>)> { |     pub fn get_thread(&self, thread_id: i32) -> Result<(Thread, Vec<SimplePost>)> { | ||||||
|         use crate::schema::threads::dsl::*; |  | ||||||
|         use crate::schema::simple_posts::dsl::id; |         use crate::schema::simple_posts::dsl::id; | ||||||
|  |         use crate::schema::threads::dsl::*; | ||||||
| 
 | 
 | ||||||
|         let conn = self.0.get()?; |         let conn = self.0.get()?; | ||||||
|         let thread_result: Thread = threads |         let thread_result: Thread = threads.find(thread_id).first(&conn)?; | ||||||
|             .find(thread_id).first(&conn)?; |  | ||||||
| 
 | 
 | ||||||
|         let post_list = SimplePost::belonging_to(&thread_result) |         let post_list = SimplePost::belonging_to(&thread_result) | ||||||
|             .order_by(id.asc()) |             .order_by(id.asc()) | ||||||
|  | @ -127,8 +122,7 @@ impl DbExecutor { | ||||||
| 
 | 
 | ||||||
|     /// Create a new thread.
 |     /// Create a new thread.
 | ||||||
|     pub fn create_thread(&self, new_thread: NewThread, post_text: String) -> Result<Thread> { |     pub fn create_thread(&self, new_thread: NewThread, post_text: String) -> Result<Thread> { | ||||||
|                 use crate::schema::threads; |         use crate::schema::{posts, threads}; | ||||||
|         use crate::schema::posts; |  | ||||||
| 
 | 
 | ||||||
|         let conn = self.0.get()?; |         let conn = self.0.get()?; | ||||||
| 
 | 
 | ||||||
|  | @ -161,15 +155,16 @@ impl DbExecutor { | ||||||
| 
 | 
 | ||||||
|         let closed: bool = { |         let closed: bool = { | ||||||
|             use crate::schema::threads::dsl::*; |             use crate::schema::threads::dsl::*; | ||||||
|             threads.select(closed) |             threads | ||||||
|  |                 .select(closed) | ||||||
|                 .find(new_post.thread_id) |                 .find(new_post.thread_id) | ||||||
|                 .first(&conn)? |                 .first(&conn)? | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         if closed { |         if closed { | ||||||
|             return Err(ConverseError::ThreadClosed { |             return Err(ConverseError::ThreadClosed { | ||||||
|                 id: new_post.thread_id |                 id: new_post.thread_id, | ||||||
|             }) |             }); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         Ok(diesel::insert_into(posts::table) |         Ok(diesel::insert_into(posts::table) | ||||||
|  | @ -197,7 +192,6 @@ impl DbExecutor { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| // Old actor implementation:
 | // Old actor implementation:
 | ||||||
| 
 | 
 | ||||||
| impl Actor for DbExecutor { | impl Actor for DbExecutor { | ||||||
|  | @ -216,9 +210,7 @@ message!(LookupOrCreateUser, Result<User>); | ||||||
| impl Handler<LookupOrCreateUser> for DbExecutor { | impl Handler<LookupOrCreateUser> for DbExecutor { | ||||||
|     type Result = <LookupOrCreateUser as Message>::Result; |     type Result = <LookupOrCreateUser as Message>::Result; | ||||||
| 
 | 
 | ||||||
|     fn handle(&mut self, |     fn handle(&mut self, _: LookupOrCreateUser, _: &mut Self::Context) -> Self::Result { | ||||||
|               _: LookupOrCreateUser, |  | ||||||
|               _: &mut Self::Context) -> Self::Result { |  | ||||||
|         unimplemented!() |         unimplemented!() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -238,7 +230,9 @@ impl Handler<GetThread> for DbExecutor { | ||||||
| 
 | 
 | ||||||
| /// Message used to fetch a specific post.
 | /// Message used to fetch a specific post.
 | ||||||
| #[derive(Deserialize, Debug)] | #[derive(Deserialize, Debug)] | ||||||
| pub struct GetPost { pub id: i32 } | pub struct GetPost { | ||||||
|  |     pub id: i32, | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| message!(GetPost, Result<SimplePost>); | message!(GetPost, Result<SimplePost>); | ||||||
| 
 | 
 | ||||||
|  | @ -296,7 +290,9 @@ impl Handler<CreatePost> for DbExecutor { | ||||||
| 
 | 
 | ||||||
| /// Message used to search for posts
 | /// Message used to search for posts
 | ||||||
| #[derive(Deserialize)] | #[derive(Deserialize)] | ||||||
| pub struct SearchPosts { pub query: String } | pub struct SearchPosts { | ||||||
|  |     pub query: String, | ||||||
|  | } | ||||||
| message!(SearchPosts, Result<Vec<SearchResult>>); | message!(SearchPosts, Result<Vec<SearchResult>>); | ||||||
| 
 | 
 | ||||||
| impl Handler<SearchPosts> for DbExecutor { | impl Handler<SearchPosts> for DbExecutor { | ||||||
|  |  | ||||||
|  | @ -21,17 +21,12 @@ | ||||||
| //! are established in a similar way as was tradition in
 | //! are established in a similar way as was tradition in
 | ||||||
| //! `error_chain`, albeit manually.
 | //! `error_chain`, albeit manually.
 | ||||||
| 
 | 
 | ||||||
| use std::result; |  | ||||||
| use actix_web::{ResponseError, HttpResponse}; |  | ||||||
| use actix_web::http::StatusCode; | use actix_web::http::StatusCode; | ||||||
|  | use actix_web::{HttpResponse, ResponseError}; | ||||||
|  | use std::result; | ||||||
| 
 | 
 | ||||||
| // Modules with foreign errors:
 | // Modules with foreign errors:
 | ||||||
| use actix; | use {actix, actix_web, askama, diesel, r2d2, tokio_timer}; | ||||||
| use actix_web; |  | ||||||
| use askama; |  | ||||||
| use diesel; |  | ||||||
| use r2d2; |  | ||||||
| use tokio_timer; |  | ||||||
| 
 | 
 | ||||||
| pub type Result<T> = result::Result<T, ConverseError>; | pub type Result<T> = result::Result<T, ConverseError>; | ||||||
| pub type ConverseResult<T> = result::Result<T, ConverseError>; | pub type ConverseResult<T> = result::Result<T, ConverseError>; | ||||||
|  | @ -96,7 +91,9 @@ impl From<askama::Error> for ConverseError { | ||||||
| 
 | 
 | ||||||
| impl From<actix::MailboxError> for ConverseError { | impl From<actix::MailboxError> for ConverseError { | ||||||
|     fn from(error: actix::MailboxError) -> ConverseError { |     fn from(error: actix::MailboxError) -> ConverseError { | ||||||
|         ConverseError::Actix { error: Box::new(error) } |         ConverseError::Actix { | ||||||
|  |             error: Box::new(error), | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -136,7 +133,7 @@ impl ResponseError for ConverseError { | ||||||
|                 .header("Location", format!("/thread/{}#post-reply", id)) |                 .header("Location", format!("/thread/{}#post-reply", id)) | ||||||
|                 .finish(), |                 .finish(), | ||||||
|             _ => HttpResponse::build(StatusCode::INTERNAL_SERVER_ERROR) |             _ => HttpResponse::build(StatusCode::INTERNAL_SERVER_ERROR) | ||||||
|                 .body(format!("An error occured: {}", self)) |                 .body(format!("An error occured: {}", self)), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -23,18 +23,18 @@ | ||||||
| //! the tera templates stored in the `/templates` directory in the
 | //! the tera templates stored in the `/templates` directory in the
 | ||||||
| //! project root.
 | //! project root.
 | ||||||
| 
 | 
 | ||||||
| use actix::prelude::*; |  | ||||||
| use actix_web::*; |  | ||||||
| use actix_web::http::Method; |  | ||||||
| use actix_web::middleware::identity::RequestIdentity; |  | ||||||
| use actix_web::middleware::{Started, Middleware}; |  | ||||||
| use actix_web; |  | ||||||
| use crate::db::*; | use crate::db::*; | ||||||
| use crate::errors::{ConverseResult, ConverseError}; | use crate::errors::{ConverseError, ConverseResult}; | ||||||
| use futures::Future; |  | ||||||
| use crate::models::*; | use crate::models::*; | ||||||
| use crate::oidc::*; | use crate::oidc::*; | ||||||
| use crate::render::*; | use crate::render::*; | ||||||
|  | use actix::prelude::*; | ||||||
|  | use actix_web; | ||||||
|  | use actix_web::http::Method; | ||||||
|  | use actix_web::middleware::identity::RequestIdentity; | ||||||
|  | use actix_web::middleware::{Middleware, Started}; | ||||||
|  | use actix_web::*; | ||||||
|  | use futures::Future; | ||||||
| 
 | 
 | ||||||
| use rouille::{Request, Response}; | use rouille::{Request, Response}; | ||||||
| 
 | 
 | ||||||
|  | @ -84,23 +84,31 @@ pub fn get_user_id_rouille(_req: &Request) -> i32 { | ||||||
|     ANONYMOUS |     ANONYMOUS | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub fn forum_thread_rouille(req: &Request, db: &DbExecutor, thread_id: i32) | pub fn forum_thread_rouille( | ||||||
|                             -> ConverseResult<Response> { |     req: &Request, | ||||||
|  |     db: &DbExecutor, | ||||||
|  |     thread_id: i32, | ||||||
|  | ) -> ConverseResult<Response> { | ||||||
|     let user = get_user_id_rouille(&req); |     let user = get_user_id_rouille(&req); | ||||||
|     let thread = db.get_thread(thread_id)?; |     let thread = db.get_thread(thread_id)?; | ||||||
|     Ok(Response::html(thread_page(user, thread.0, thread.1)?)) |     Ok(Response::html(thread_page(user, thread.0, thread.1)?)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// This handler retrieves and displays a single forum thread.
 | /// This handler retrieves and displays a single forum thread.
 | ||||||
| pub fn forum_thread(_: State<AppState>, | pub fn forum_thread( | ||||||
|  |     _: State<AppState>, | ||||||
|     _: HttpRequest<AppState>, |     _: HttpRequest<AppState>, | ||||||
|                     _: Path<i32>) -> ConverseResponse { |     _: Path<i32>, | ||||||
|  | ) -> ConverseResponse { | ||||||
|     unimplemented!() |     unimplemented!() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// This handler presents the user with the "New Thread" form.
 | /// This handler presents the user with the "New Thread" form.
 | ||||||
| pub fn new_thread(state: State<AppState>) -> ConverseResponse { | pub fn new_thread(state: State<AppState>) -> ConverseResponse { | ||||||
|     state.renderer.send(NewThreadPage::default()).flatten() |     state | ||||||
|  |         .renderer | ||||||
|  |         .send(NewThreadPage::default()) | ||||||
|  |         .flatten() | ||||||
|         .map(|res| HttpResponse::Ok().content_type(HTML).body(res)) |         .map(|res| HttpResponse::Ok().content_type(HTML).body(res)) | ||||||
|         .responder() |         .responder() | ||||||
| } | } | ||||||
|  | @ -113,9 +121,9 @@ pub struct NewThreadForm { | ||||||
| 
 | 
 | ||||||
| /// This handler receives a "New thread"-form and redirects the user
 | /// This handler receives a "New thread"-form and redirects the user
 | ||||||
| /// to the new thread after creation.
 | /// to the new thread after creation.
 | ||||||
| pub fn submit_thread((state, input, req): (State<AppState>, | pub fn submit_thread( | ||||||
|                                            Form<NewThreadForm>, |     (state, input, req): (State<AppState>, Form<NewThreadForm>, HttpRequest<AppState>), | ||||||
|                                            HttpRequest<AppState>)) -> ConverseResponse { | ) -> ConverseResponse { | ||||||
|     // Trim whitespace out of inputs:
 |     // Trim whitespace out of inputs:
 | ||||||
|     let input = NewThreadForm { |     let input = NewThreadForm { | ||||||
|         title: input.title.trim().into(), |         title: input.title.trim().into(), | ||||||
|  | @ -124,7 +132,8 @@ pub fn submit_thread((state, input, req): (State<AppState>, | ||||||
| 
 | 
 | ||||||
|     // Perform simple validation and abort here if it fails:
 |     // Perform simple validation and abort here if it fails:
 | ||||||
|     if input.title.is_empty() || input.post.is_empty() { |     if input.title.is_empty() || input.post.is_empty() { | ||||||
|         return state.renderer |         return state | ||||||
|  |             .renderer | ||||||
|             .send(NewThreadPage { |             .send(NewThreadPage { | ||||||
|                 alerts: vec![NEW_THREAD_LENGTH_ERR], |                 alerts: vec![NEW_THREAD_LENGTH_ERR], | ||||||
|                 title: Some(input.title), |                 title: Some(input.title), | ||||||
|  | @ -147,11 +156,16 @@ pub fn submit_thread((state, input, req): (State<AppState>, | ||||||
|         post: input.post, |         post: input.post, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     state.db.send(msg) |     state | ||||||
|  |         .db | ||||||
|  |         .send(msg) | ||||||
|         .from_err() |         .from_err() | ||||||
|         .and_then(move |res| { |         .and_then(move |res| { | ||||||
|             let thread = res?; |             let thread = res?; | ||||||
|             info!("Created new thread \"{}\" with ID {}", thread.title, thread.id); |             info!( | ||||||
|  |                 "Created new thread \"{}\" with ID {}", | ||||||
|  |                 thread.title, thread.id | ||||||
|  |             ); | ||||||
|             Ok(HttpResponse::SeeOther() |             Ok(HttpResponse::SeeOther() | ||||||
|                 .header("Location", format!("/thread/{}", thread.id)) |                 .header("Location", format!("/thread/{}", thread.id)) | ||||||
|                 .finish()) |                 .finish()) | ||||||
|  | @ -167,9 +181,11 @@ pub struct NewPostForm { | ||||||
| 
 | 
 | ||||||
| /// This handler receives a "Reply"-form and redirects the user to the
 | /// This handler receives a "Reply"-form and redirects the user to the
 | ||||||
| /// new post after creation.
 | /// new post after creation.
 | ||||||
| pub fn reply_thread(state: State<AppState>, | pub fn reply_thread( | ||||||
|  |     state: State<AppState>, | ||||||
|     input: Form<NewPostForm>, |     input: Form<NewPostForm>, | ||||||
|                     req: HttpRequest<AppState>) -> ConverseResponse { |     req: HttpRequest<AppState>, | ||||||
|  | ) -> ConverseResponse { | ||||||
|     let user_id = get_user_id(&req); |     let user_id = get_user_id(&req); | ||||||
| 
 | 
 | ||||||
|     let new_post = NewPost { |     let new_post = NewPost { | ||||||
|  | @ -178,13 +194,18 @@ pub fn reply_thread(state: State<AppState>, | ||||||
|         body: input.post.trim().into(), |         body: input.post.trim().into(), | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     state.db.send(CreatePost(new_post)) |     state | ||||||
|  |         .db | ||||||
|  |         .send(CreatePost(new_post)) | ||||||
|         .flatten() |         .flatten() | ||||||
|         .from_err() |         .from_err() | ||||||
|         .and_then(move |post| { |         .and_then(move |post| { | ||||||
|             info!("Posted reply {} to thread {}", post.id, post.thread_id); |             info!("Posted reply {} to thread {}", post.id, post.thread_id); | ||||||
|             Ok(HttpResponse::SeeOther() |             Ok(HttpResponse::SeeOther() | ||||||
|                .header("Location", format!("/thread/{}#post-{}", post.thread_id, post.id)) |                 .header( | ||||||
|  |                     "Location", | ||||||
|  |                     format!("/thread/{}#post-{}", post.thread_id, post.id), | ||||||
|  |                 ) | ||||||
|                 .finish()) |                 .finish()) | ||||||
|         }) |         }) | ||||||
|         .responder() |         .responder() | ||||||
|  | @ -194,12 +215,16 @@ pub fn reply_thread(state: State<AppState>, | ||||||
| /// the user attempts to edit a post that they do not have access to,
 | /// the user attempts to edit a post that they do not have access to,
 | ||||||
| /// they are currently ungracefully redirected back to the post
 | /// they are currently ungracefully redirected back to the post
 | ||||||
| /// itself.
 | /// itself.
 | ||||||
| pub fn edit_form(state: State<AppState>, | pub fn edit_form( | ||||||
|  |     state: State<AppState>, | ||||||
|     req: HttpRequest<AppState>, |     req: HttpRequest<AppState>, | ||||||
|                  query: Path<GetPost>) -> ConverseResponse { |     query: Path<GetPost>, | ||||||
|  | ) -> ConverseResponse { | ||||||
|     let user_id = get_user_id(&req); |     let user_id = get_user_id(&req); | ||||||
| 
 | 
 | ||||||
|     state.db.send(query.into_inner()) |     state | ||||||
|  |         .db | ||||||
|  |         .send(query.into_inner()) | ||||||
|         .flatten() |         .flatten() | ||||||
|         .from_err() |         .from_err() | ||||||
|         .and_then(move |post| { |         .and_then(move |post| { | ||||||
|  | @ -227,12 +252,16 @@ pub fn edit_form(state: State<AppState>, | ||||||
| 
 | 
 | ||||||
| /// This handler "executes" an edit to a post if the current user owns
 | /// This handler "executes" an edit to a post if the current user owns
 | ||||||
| /// the edited post.
 | /// the edited post.
 | ||||||
| pub fn edit_post(state: State<AppState>, | pub fn edit_post( | ||||||
|  |     state: State<AppState>, | ||||||
|     req: HttpRequest<AppState>, |     req: HttpRequest<AppState>, | ||||||
|                  update: Form<UpdatePost>) -> ConverseResponse { |     update: Form<UpdatePost>, | ||||||
|  | ) -> ConverseResponse { | ||||||
|     let user_id = get_user_id(&req); |     let user_id = get_user_id(&req); | ||||||
| 
 | 
 | ||||||
|     state.db.send(GetPost { id: update.post_id }) |     state | ||||||
|  |         .db | ||||||
|  |         .send(GetPost { id: update.post_id }) | ||||||
|         .flatten() |         .flatten() | ||||||
|         .from_err() |         .from_err() | ||||||
|         .and_then(move |post| { |         .and_then(move |post| { | ||||||
|  | @ -247,24 +276,34 @@ pub fn edit_post(state: State<AppState>, | ||||||
|         }) |         }) | ||||||
|         .and_then(move |_| state.db.send(update.0).from_err()) |         .and_then(move |_| state.db.send(update.0).from_err()) | ||||||
|         .flatten() |         .flatten() | ||||||
|         .map(|updated| HttpResponse::SeeOther() |         .map(|updated| { | ||||||
|              .header("Location", format!("/thread/{}#post-{}", |             HttpResponse::SeeOther() | ||||||
|                                          updated.thread_id, updated.id)) |                 .header( | ||||||
|              .finish()) |                     "Location", | ||||||
|  |                     format!("/thread/{}#post-{}", updated.thread_id, updated.id), | ||||||
|  |                 ) | ||||||
|  |                 .finish() | ||||||
|  |         }) | ||||||
|         .responder() |         .responder() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// This handler executes a full-text search on the forum database and
 | /// This handler executes a full-text search on the forum database and
 | ||||||
| /// displays the results to the user.
 | /// displays the results to the user.
 | ||||||
| pub fn search_forum(state: State<AppState>, | pub fn search_forum(state: State<AppState>, query: Query<SearchPosts>) -> ConverseResponse { | ||||||
|                     query: Query<SearchPosts>) -> ConverseResponse { |  | ||||||
|     let query_string = query.query.clone(); |     let query_string = query.query.clone(); | ||||||
|     state.db.send(query.into_inner()) |     state | ||||||
|  |         .db | ||||||
|  |         .send(query.into_inner()) | ||||||
|         .flatten() |         .flatten() | ||||||
|         .and_then(move |results| state.renderer.send(SearchResultPage { |         .and_then(move |results| { | ||||||
|  |             state | ||||||
|  |                 .renderer | ||||||
|  |                 .send(SearchResultPage { | ||||||
|                     results, |                     results, | ||||||
|                     query: query_string, |                     query: query_string, | ||||||
|         }).from_err()) |                 }) | ||||||
|  |                 .from_err() | ||||||
|  |         }) | ||||||
|         .flatten() |         .flatten() | ||||||
|         .map(|res| HttpResponse::Ok().content_type(HTML).body(res)) |         .map(|res| HttpResponse::Ok().content_type(HTML).body(res)) | ||||||
|         .responder() |         .responder() | ||||||
|  | @ -272,11 +311,15 @@ pub fn search_forum(state: State<AppState>, | ||||||
| 
 | 
 | ||||||
| /// This handler initiates an OIDC login.
 | /// This handler initiates an OIDC login.
 | ||||||
| pub fn login(state: State<AppState>) -> ConverseResponse { | pub fn login(state: State<AppState>) -> ConverseResponse { | ||||||
|     state.oidc.send(GetLoginUrl) |     state | ||||||
|  |         .oidc | ||||||
|  |         .send(GetLoginUrl) | ||||||
|         .from_err() |         .from_err() | ||||||
|         .and_then(|url| Ok(HttpResponse::TemporaryRedirect() |         .and_then(|url| { | ||||||
|  |             Ok(HttpResponse::TemporaryRedirect() | ||||||
|                 .header("Location", url) |                 .header("Location", url) | ||||||
|                            .finish())) |                 .finish()) | ||||||
|  |         }) | ||||||
|         .responder() |         .responder() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -286,21 +329,26 @@ pub fn login(state: State<AppState>) -> ConverseResponse { | ||||||
| /// provider and a user lookup is performed. If a user with a matching
 | /// provider and a user lookup is performed. If a user with a matching
 | ||||||
| /// email-address is found in the database, it is logged in -
 | /// email-address is found in the database, it is logged in -
 | ||||||
| /// otherwise a new user is created.
 | /// otherwise a new user is created.
 | ||||||
| pub fn callback(state: State<AppState>, | pub fn callback( | ||||||
|  |     state: State<AppState>, | ||||||
|     data: Form<CodeResponse>, |     data: Form<CodeResponse>, | ||||||
|                 req: HttpRequest<AppState>) -> ConverseResponse { |     req: HttpRequest<AppState>, | ||||||
|     state.oidc.send(RetrieveToken(data.0)).flatten() | ) -> ConverseResponse { | ||||||
|  |     state | ||||||
|  |         .oidc | ||||||
|  |         .send(RetrieveToken(data.0)) | ||||||
|  |         .flatten() | ||||||
|         .map(|author| LookupOrCreateUser { |         .map(|author| LookupOrCreateUser { | ||||||
|             email: author.email, |             email: author.email, | ||||||
|             name: author.name, |             name: author.name, | ||||||
|         }) |         }) | ||||||
|         .and_then(move |msg| state.db.send(msg).from_err()).flatten() |         .and_then(move |msg| state.db.send(msg).from_err()) | ||||||
|  |         .flatten() | ||||||
|         .and_then(move |user| { |         .and_then(move |user| { | ||||||
|             info!("Completed login for user {} ({})", user.email, user.id); |             info!("Completed login for user {} ({})", user.email, user.id); | ||||||
|             req.remember(user.id.to_string()); |             req.remember(user.id.to_string()); | ||||||
|             Ok(HttpResponse::SeeOther() |             Ok(HttpResponse::SeeOther().header("Location", "/").finish()) | ||||||
|                .header("Location", "/") |         }) | ||||||
|                .finish())}) |  | ||||||
|         .responder() |         .responder() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -317,9 +365,7 @@ impl EmbeddedFile for App<AppState> { | ||||||
|     fn static_file(self, path: &'static str, content: &'static [u8]) -> Self { |     fn static_file(self, path: &'static str, content: &'static [u8]) -> Self { | ||||||
|         self.route(path, Method::GET, move |_: HttpRequest<_>| { |         self.route(path, Method::GET, move |_: HttpRequest<_>| { | ||||||
|             let mime = format!("{}", mime_guess::from_path(path).first_or_octet_stream()); |             let mime = format!("{}", mime_guess::from_path(path).first_or_octet_stream()); | ||||||
|             HttpResponse::Ok() |             HttpResponse::Ok().content_type(mime.as_str()).body(content) | ||||||
|                 .content_type(mime.as_str()) |  | ||||||
|                 .body(content) |  | ||||||
|         }) |         }) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -336,7 +382,7 @@ impl <S> Middleware<S> for RequireLogin { | ||||||
|             Ok(Started::Response( |             Ok(Started::Response( | ||||||
|                 HttpResponse::SeeOther() |                 HttpResponse::SeeOther() | ||||||
|                     .header("Location", "/oidc/login") |                     .header("Location", "/oidc/login") | ||||||
|                     .finish() |                     .finish(), | ||||||
|             )) |             )) | ||||||
|         } else { |         } else { | ||||||
|             Ok(Started::Done) |             Ok(Started::Done) | ||||||
|  |  | ||||||
|  | @ -30,7 +30,6 @@ extern crate log; | ||||||
| #[macro_use] | #[macro_use] | ||||||
| extern crate serde_derive; | extern crate serde_derive; | ||||||
| 
 | 
 | ||||||
| extern crate rouille; |  | ||||||
| extern crate actix; | extern crate actix; | ||||||
| extern crate actix_web; | extern crate actix_web; | ||||||
| extern crate chrono; | extern crate chrono; | ||||||
|  | @ -44,6 +43,7 @@ extern crate md5; | ||||||
| extern crate mime_guess; | extern crate mime_guess; | ||||||
| extern crate r2d2; | extern crate r2d2; | ||||||
| extern crate rand; | extern crate rand; | ||||||
|  | extern crate rouille; | ||||||
| extern crate serde; | extern crate serde; | ||||||
| extern crate serde_json; | extern crate serde_json; | ||||||
| extern crate tokio; | extern crate tokio; | ||||||
|  | @ -58,7 +58,7 @@ macro_rules! message { | ||||||
|         impl Message for $t { |         impl Message for $t { | ||||||
|             type Result = $r; |             type Result = $r; | ||||||
|         } |         } | ||||||
|     } |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub mod db; | pub mod db; | ||||||
|  | @ -69,18 +69,18 @@ pub mod oidc; | ||||||
| pub mod render; | pub mod render; | ||||||
| pub mod schema; | pub mod schema; | ||||||
| 
 | 
 | ||||||
| use actix::prelude::*; |  | ||||||
| use actix_web::*; |  | ||||||
| use actix_web::http::Method; |  | ||||||
| use actix_web::middleware::Logger; |  | ||||||
| use actix_web::middleware::identity::{IdentityService, CookieIdentityPolicy}; |  | ||||||
| use crate::db::*; | use crate::db::*; | ||||||
| use diesel::pg::PgConnection; |  | ||||||
| use diesel::r2d2::{ConnectionManager, Pool}; |  | ||||||
| use crate::handlers::*; | use crate::handlers::*; | ||||||
| use crate::oidc::OidcExecutor; | use crate::oidc::OidcExecutor; | ||||||
| use rand::{OsRng, Rng}; |  | ||||||
| use crate::render::Renderer; | use crate::render::Renderer; | ||||||
|  | use actix::prelude::*; | ||||||
|  | use actix_web::http::Method; | ||||||
|  | use actix_web::middleware::identity::{CookieIdentityPolicy, IdentityService}; | ||||||
|  | use actix_web::middleware::Logger; | ||||||
|  | use actix_web::*; | ||||||
|  | use diesel::pg::PgConnection; | ||||||
|  | use diesel::r2d2::{ConnectionManager, Pool}; | ||||||
|  | use rand::{OsRng, Rng}; | ||||||
| use std::env; | use std::env; | ||||||
| 
 | 
 | ||||||
| fn config(name: &str) -> String { | fn config(name: &str) -> String { | ||||||
|  | @ -96,16 +96,18 @@ fn start_db_executor() -> Addr<DbExecutor> { | ||||||
|     let db_url = config("DATABASE_URL"); |     let db_url = config("DATABASE_URL"); | ||||||
| 
 | 
 | ||||||
|     let manager = ConnectionManager::<PgConnection>::new(db_url); |     let manager = ConnectionManager::<PgConnection>::new(db_url); | ||||||
|     let pool = Pool::builder().build(manager).expect("Failed to initialise DB pool"); |     let pool = Pool::builder() | ||||||
|  |         .build(manager) | ||||||
|  |         .expect("Failed to initialise DB pool"); | ||||||
| 
 | 
 | ||||||
|     SyncArbiter::start(2, move || DbExecutor(pool.clone())) |     SyncArbiter::start(2, move || DbExecutor(pool.clone())) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn schedule_search_refresh(db: Addr<DbExecutor>) { | fn schedule_search_refresh(db: Addr<DbExecutor>) { | ||||||
|  |     use std::thread; | ||||||
|  |     use std::time::{Duration, Instant}; | ||||||
|     use tokio::prelude::*; |     use tokio::prelude::*; | ||||||
|     use tokio::timer::Interval; |     use tokio::timer::Interval; | ||||||
|     use std::time::{Duration, Instant}; |  | ||||||
|     use std::thread; |  | ||||||
| 
 | 
 | ||||||
|     let task = Interval::new(Instant::now(), Duration::from_secs(60)) |     let task = Interval::new(Instant::now(), Duration::from_secs(60)) | ||||||
|         .from_err() |         .from_err() | ||||||
|  | @ -118,8 +120,8 @@ fn schedule_search_refresh(db: Addr<DbExecutor>) { | ||||||
| fn start_oidc_executor(base_url: &str) -> Addr<OidcExecutor> { | fn start_oidc_executor(base_url: &str) -> Addr<OidcExecutor> { | ||||||
|     info!("Initialising OIDC integration ..."); |     info!("Initialising OIDC integration ..."); | ||||||
|     let oidc_url = config("OIDC_DISCOVERY_URL"); |     let oidc_url = config("OIDC_DISCOVERY_URL"); | ||||||
|     let oidc_config = oidc::load_oidc(&oidc_url) |     let oidc_config = | ||||||
|         .expect("Failed to retrieve OIDC discovery document"); |         oidc::load_oidc(&oidc_url).expect("Failed to retrieve OIDC discovery document"); | ||||||
| 
 | 
 | ||||||
|     let oidc = oidc::OidcExecutor { |     let oidc = oidc::OidcExecutor { | ||||||
|         oidc_config, |         oidc_config, | ||||||
|  | @ -148,17 +150,18 @@ fn start_renderer() -> Addr<Renderer> { | ||||||
| 
 | 
 | ||||||
| fn gen_session_key() -> [u8; 64] { | fn gen_session_key() -> [u8; 64] { | ||||||
|     let mut key_bytes = [0; 64]; |     let mut key_bytes = [0; 64]; | ||||||
|     let mut rng = OsRng::new() |     let mut rng = OsRng::new().expect("Failed to retrieve RNG for key generation"); | ||||||
|         .expect("Failed to retrieve RNG for key generation"); |  | ||||||
|     rng.fill_bytes(&mut key_bytes); |     rng.fill_bytes(&mut key_bytes); | ||||||
| 
 | 
 | ||||||
|     key_bytes |     key_bytes | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn start_http_server(base_url: String, | fn start_http_server( | ||||||
|  |     base_url: String, | ||||||
|     db_addr: Addr<DbExecutor>, |     db_addr: Addr<DbExecutor>, | ||||||
|     oidc_addr: Addr<OidcExecutor>, |     oidc_addr: Addr<OidcExecutor>, | ||||||
|                      renderer_addr: Addr<Renderer>) { |     renderer_addr: Addr<Renderer>, | ||||||
|  | ) { | ||||||
|     info!("Initialising HTTP server ..."); |     info!("Initialising HTTP server ..."); | ||||||
|     let bind_host = config_default("CONVERSE_BIND_HOST", "127.0.0.1:4567"); |     let bind_host = config_default("CONVERSE_BIND_HOST", "127.0.0.1:4567"); | ||||||
|     let key = gen_session_key(); |     let key = gen_session_key(); | ||||||
|  | @ -175,7 +178,7 @@ fn start_http_server(base_url: String, | ||||||
|             CookieIdentityPolicy::new(&key) |             CookieIdentityPolicy::new(&key) | ||||||
|                 .name("converse_auth") |                 .name("converse_auth") | ||||||
|                 .path("/") |                 .path("/") | ||||||
|                 .secure(base_url.starts_with("https")) |                 .secure(base_url.starts_with("https")), | ||||||
|         ); |         ); | ||||||
| 
 | 
 | ||||||
|         let app = App::with_state(state) |         let app = App::with_state(state) | ||||||
|  | @ -183,24 +186,36 @@ fn start_http_server(base_url: String, | ||||||
|             .middleware(identity) |             .middleware(identity) | ||||||
|             .resource("/", |r| r.method(Method::GET).with(forum_index)) |             .resource("/", |r| r.method(Method::GET).with(forum_index)) | ||||||
|             .resource("/thread/new", |r| r.method(Method::GET).with(new_thread)) |             .resource("/thread/new", |r| r.method(Method::GET).with(new_thread)) | ||||||
|             .resource("/thread/submit", |r| r.method(Method::POST).with(submit_thread)) |             .resource("/thread/submit", |r| { | ||||||
|             .resource("/thread/reply", |r| r.method(Method::POST).with(reply_thread)) |                 r.method(Method::POST).with(submit_thread) | ||||||
|  |             }) | ||||||
|  |             .resource("/thread/reply", |r| { | ||||||
|  |                 r.method(Method::POST).with(reply_thread) | ||||||
|  |             }) | ||||||
|             .resource("/thread/{id}", |r| r.method(Method::GET).with(forum_thread)) |             .resource("/thread/{id}", |r| r.method(Method::GET).with(forum_thread)) | ||||||
|             .resource("/post/{id}/edit", |r| r.method(Method::GET).with(edit_form)) |             .resource("/post/{id}/edit", |r| r.method(Method::GET).with(edit_form)) | ||||||
|             .resource("/post/edit", |r| r.method(Method::POST).with(edit_post)) |             .resource("/post/edit", |r| r.method(Method::POST).with(edit_post)) | ||||||
|             .resource("/search", |r| r.method(Method::GET).with(search_forum)) |             .resource("/search", |r| r.method(Method::GET).with(search_forum)) | ||||||
|             .resource("/oidc/login", |r| r.method(Method::GET).with(login)) |             .resource("/oidc/login", |r| r.method(Method::GET).with(login)) | ||||||
|             .resource("/oidc/callback", |r| r.method(Method::POST).with(callback)) |             .resource("/oidc/callback", |r| r.method(Method::POST).with(callback)) | ||||||
|             .static_file("/static/highlight.css", include_bytes!("../static/highlight.css")) |             .static_file( | ||||||
|             .static_file("/static/highlight.js", include_bytes!("../static/highlight.js")) |                 "/static/highlight.css", | ||||||
|  |                 include_bytes!("../static/highlight.css"), | ||||||
|  |             ) | ||||||
|  |             .static_file( | ||||||
|  |                 "/static/highlight.js", | ||||||
|  |                 include_bytes!("../static/highlight.js"), | ||||||
|  |             ) | ||||||
|             .static_file("/static/styles.css", include_bytes!("../static/styles.css")); |             .static_file("/static/styles.css", include_bytes!("../static/styles.css")); | ||||||
| 
 | 
 | ||||||
|         if require_login { |         if require_login { | ||||||
|             app.middleware(RequireLogin) |             app.middleware(RequireLogin) | ||||||
|         } else { |         } else { | ||||||
|             app |             app | ||||||
|         }}) |         } | ||||||
|         .bind(&bind_host).expect(&format!("Could not bind on '{}'", bind_host)) |     }) | ||||||
|  |     .bind(&bind_host) | ||||||
|  |     .expect(&format!("Could not bind on '{}'", bind_host)) | ||||||
|     .start(); |     .start(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -16,9 +16,9 @@ | ||||||
| // along with this program. If not, see
 | // along with this program. If not, see
 | ||||||
| // <https://www.gnu.org/licenses/>.
 | // <https://www.gnu.org/licenses/>.
 | ||||||
| 
 | 
 | ||||||
|  | use crate::schema::{posts, simple_posts, threads, users}; | ||||||
| use chrono::prelude::{DateTime, Utc}; | use chrono::prelude::{DateTime, Utc}; | ||||||
| use crate::schema::{users, threads, posts, simple_posts}; | use diesel::sql_types::{Integer, Text}; | ||||||
| use diesel::sql_types::{Text, Integer}; |  | ||||||
| 
 | 
 | ||||||
| /// Represents a single user in the Converse database. Converse does
 | /// Represents a single user in the Converse database. Converse does
 | ||||||
| /// not handle logins itself, but rather looks them up based on the
 | /// not handle logins itself, but rather looks them up based on the
 | ||||||
|  |  | ||||||
|  | @ -22,12 +22,12 @@ | ||||||
| //! Currently Converse only supports a single OIDC provider. Note that
 | //! Currently Converse only supports a single OIDC provider. Note that
 | ||||||
| //! this has so far only been tested with Office365.
 | //! this has so far only been tested with Office365.
 | ||||||
| 
 | 
 | ||||||
| use actix::prelude::*; |  | ||||||
| use crate::errors::*; | use crate::errors::*; | ||||||
|  | use actix::prelude::*; | ||||||
| use crimp::Request; | use crimp::Request; | ||||||
|  | use curl::easy::Form; | ||||||
| use url::Url; | use url::Url; | ||||||
| use url_serde; | use url_serde; | ||||||
| use curl::easy::Form; |  | ||||||
| 
 | 
 | ||||||
| /// This structure represents the contents of an OIDC discovery
 | /// This structure represents the contents of an OIDC discovery
 | ||||||
| /// document.
 | /// document.
 | ||||||
|  | @ -114,20 +114,30 @@ impl Handler<RetrieveToken> for OidcExecutor { | ||||||
|         debug!("Received OAuth2 code, requesting access_token"); |         debug!("Received OAuth2 code, requesting access_token"); | ||||||
| 
 | 
 | ||||||
|         let mut form = Form::new(); |         let mut form = Form::new(); | ||||||
|         form.part("client_id").contents(&self.client_id.as_bytes()) |         form.part("client_id") | ||||||
|             .add().expect("critical error: invalid form data"); |             .contents(&self.client_id.as_bytes()) | ||||||
|  |             .add() | ||||||
|  |             .expect("critical error: invalid form data"); | ||||||
| 
 | 
 | ||||||
|         form.part("client_secret").contents(&self.client_secret.as_bytes()) |         form.part("client_secret") | ||||||
|             .add().expect("critical error: invalid form data"); |             .contents(&self.client_secret.as_bytes()) | ||||||
|  |             .add() | ||||||
|  |             .expect("critical error: invalid form data"); | ||||||
| 
 | 
 | ||||||
|         form.part("grant_type").contents("authorization_code".as_bytes()) |         form.part("grant_type") | ||||||
|             .add().expect("critical error: invalid form data"); |             .contents("authorization_code".as_bytes()) | ||||||
|  |             .add() | ||||||
|  |             .expect("critical error: invalid form data"); | ||||||
| 
 | 
 | ||||||
|         form.part("code").contents(&msg.0.code.as_bytes()) |         form.part("code") | ||||||
|             .add().expect("critical error: invalid form data"); |             .contents(&msg.0.code.as_bytes()) | ||||||
|  |             .add() | ||||||
|  |             .expect("critical error: invalid form data"); | ||||||
| 
 | 
 | ||||||
|         form.part("redirect_uri").contents(&self.redirect_uri.as_bytes()) |         form.part("redirect_uri") | ||||||
|             .add().expect("critical error: invalid form data"); |             .contents(&self.redirect_uri.as_bytes()) | ||||||
|  |             .add() | ||||||
|  |             .expect("critical error: invalid form data"); | ||||||
| 
 | 
 | ||||||
|         let response = Request::post(&self.oidc_config.token_endpoint) |         let response = Request::post(&self.oidc_config.token_endpoint) | ||||||
|             .user_agent(concat!("converse-", env!("CARGO_PKG_VERSION")))? |             .user_agent(concat!("converse-", env!("CARGO_PKG_VERSION")))? | ||||||
|  | @ -142,7 +152,8 @@ impl Handler<RetrieveToken> for OidcExecutor { | ||||||
|             .user_agent(concat!("converse-", env!("CARGO_PKG_VERSION")))? |             .user_agent(concat!("converse-", env!("CARGO_PKG_VERSION")))? | ||||||
|             .header("Authorization", &bearer)? |             .header("Authorization", &bearer)? | ||||||
|             .send()? |             .send()? | ||||||
|             .as_json()?.body; |             .as_json()? | ||||||
|  |             .body; | ||||||
| 
 | 
 | ||||||
|         Ok(Author { |         Ok(Author { | ||||||
|             name: user.name, |             name: user.name, | ||||||
|  |  | ||||||
|  | @ -20,14 +20,14 @@ | ||||||
| //! data into whatever format is needed by the templates and rendering
 | //! data into whatever format is needed by the templates and rendering
 | ||||||
| //! them.
 | //! them.
 | ||||||
| 
 | 
 | ||||||
|  | use crate::errors::*; | ||||||
|  | use crate::models::*; | ||||||
| use actix::prelude::*; | use actix::prelude::*; | ||||||
| use askama::Template; | use askama::Template; | ||||||
| use crate::errors::*; |  | ||||||
| use std::fmt; |  | ||||||
| use md5; |  | ||||||
| use crate::models::*; |  | ||||||
| use chrono::prelude::{DateTime, Utc}; | use chrono::prelude::{DateTime, Utc}; | ||||||
| use comrak::{ComrakOptions, markdown_to_html}; | use comrak::{markdown_to_html, ComrakOptions}; | ||||||
|  | use md5; | ||||||
|  | use std::fmt; | ||||||
| 
 | 
 | ||||||
| pub struct Renderer { | pub struct Renderer { | ||||||
|     pub comrak: ComrakOptions, |     pub comrak: ComrakOptions, | ||||||
|  | @ -101,7 +101,9 @@ pub enum EditingMode { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Default for EditingMode { | impl Default for EditingMode { | ||||||
|     fn default() -> EditingMode { EditingMode::NewThread } |     fn default() -> EditingMode { | ||||||
|  |         EditingMode::NewThread | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// This is the template used for rendering the new thread, edit post
 | /// This is the template used for rendering the new thread, edit post
 | ||||||
|  | @ -215,7 +217,9 @@ pub fn index_page(threads: Vec<ThreadIndex>) -> Result<String> { | ||||||
| 
 | 
 | ||||||
| // Render the page of a given thread.
 | // Render the page of a given thread.
 | ||||||
| pub fn thread_page(user: i32, thread: Thread, posts: Vec<SimplePost>) -> Result<String> { | pub fn thread_page(user: i32, thread: Thread, posts: Vec<SimplePost>) -> Result<String> { | ||||||
|     let posts = posts.into_iter().map(|post| { |     let posts = posts | ||||||
|  |         .into_iter() | ||||||
|  |         .map(|post| { | ||||||
|             let editable = user != 1 && post.user_id == user; |             let editable = user != 1 && post.user_id == user; | ||||||
| 
 | 
 | ||||||
|             let comrak = ComrakOptions::default(); // TODO(tazjin): cheddar
 |             let comrak = ComrakOptions::default(); // TODO(tazjin): cheddar
 | ||||||
|  | @ -227,7 +231,8 @@ pub fn thread_page(user: i32, thread: Thread, posts: Vec<SimplePost>) -> Result< | ||||||
|                 author_gravatar: md5_hex(post.author_email.as_bytes()), |                 author_gravatar: md5_hex(post.author_email.as_bytes()), | ||||||
|                 editable, |                 editable, | ||||||
|             } |             } | ||||||
|     }).collect(); |         }) | ||||||
|  |         .collect(); | ||||||
| 
 | 
 | ||||||
|     let renderable = RenderableThreadPage { |     let renderable = RenderableThreadPage { | ||||||
|         posts, |         posts, | ||||||
|  |  | ||||||
|  | @ -80,9 +80,4 @@ joinable!(posts -> users (user_id)); | ||||||
| joinable!(threads -> users (user_id)); | joinable!(threads -> users (user_id)); | ||||||
| joinable!(simple_posts -> threads (thread_id)); | joinable!(simple_posts -> threads (thread_id)); | ||||||
| 
 | 
 | ||||||
| allow_tables_to_appear_in_same_query!( | allow_tables_to_appear_in_same_query!(posts, threads, users, simple_posts,); | ||||||
|     posts, |  | ||||||
|     threads, |  | ||||||
|     users, |  | ||||||
|     simple_posts, |  | ||||||
| ); |  | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue