|
| 1 | +use crate::envoy::{RateLimitDescriptor, RateLimitRequest}; |
| 2 | +use crate::service::Service; |
| 3 | +use protobuf::{Message, RepeatedField}; |
| 4 | +use proxy_wasm::hostcalls::dispatch_grpc_call; |
| 5 | +use proxy_wasm::types::Status; |
| 6 | +use std::time::Duration; |
| 7 | + |
| 8 | +const RATELIMIT_SERVICE_NAME: &str = "envoy.service.ratelimit.v3.RateLimitService"; |
| 9 | +const RATELIMIT_METHOD_NAME: &str = "ShouldRateLimit"; |
| 10 | + |
| 11 | +pub struct RateLimitService<'a> { |
| 12 | + endpoint: String, |
| 13 | + metadata: Vec<(&'a str, &'a [u8])>, |
| 14 | +} |
| 15 | + |
| 16 | +impl<'a> RateLimitService<'a> { |
| 17 | + pub fn new(endpoint: &str, metadata: Vec<(&'a str, &'a [u8])>) -> RateLimitService<'a> { |
| 18 | + Self { |
| 19 | + endpoint: String::from(endpoint), |
| 20 | + metadata, |
| 21 | + } |
| 22 | + } |
| 23 | +} |
| 24 | + |
| 25 | +impl Service<RateLimitRequest, RateLimitDescriptor> for RateLimitService<'_> { |
| 26 | + fn message( |
| 27 | + domain: String, |
| 28 | + descriptors: RepeatedField<RateLimitDescriptor>, |
| 29 | + ) -> RateLimitRequest { |
| 30 | + RateLimitRequest { |
| 31 | + domain, |
| 32 | + descriptors, |
| 33 | + hits_addend: 1, |
| 34 | + unknown_fields: Default::default(), |
| 35 | + cached_size: Default::default(), |
| 36 | + } |
| 37 | + } |
| 38 | + fn send(&self, message: RateLimitRequest) -> Result<u32, Status> { |
| 39 | + let msg = Message::write_to_bytes(&message).unwrap(); // TODO(didierofrivia): Error Handling |
| 40 | + dispatch_grpc_call( |
| 41 | + self.endpoint.as_str(), |
| 42 | + RATELIMIT_SERVICE_NAME, |
| 43 | + RATELIMIT_METHOD_NAME, |
| 44 | + self.metadata.clone(), |
| 45 | + Some(&msg), |
| 46 | + Duration::from_secs(5), |
| 47 | + ) |
| 48 | + } |
| 49 | +} |
| 50 | + |
| 51 | +#[cfg(test)] |
| 52 | +mod tests { |
| 53 | + use protobuf::{CachedSize, RepeatedField, UnknownFields}; |
| 54 | + use crate::envoy::{RateLimitDescriptor, RateLimitDescriptor_Entry}; |
| 55 | + use crate::service::rate_limit::RateLimitService; |
| 56 | + use crate::service::Service; |
| 57 | + |
| 58 | + #[test] |
| 59 | + fn builds_message() { |
| 60 | + let domain = "rlp1"; |
| 61 | + let mut field = RateLimitDescriptor::new(); |
| 62 | + let mut entry = RateLimitDescriptor_Entry::new(); |
| 63 | + entry.set_key("key1".to_string()); |
| 64 | + entry.set_value("value1".to_string()); |
| 65 | + field.set_entries(RepeatedField::from_vec(vec![entry])); |
| 66 | + let descriptors = RepeatedField::from_vec(vec![field]); |
| 67 | + |
| 68 | + let msg = RateLimitService::message(domain.to_string(), descriptors.clone()); |
| 69 | + |
| 70 | + assert_eq!(msg.hits_addend, 1); |
| 71 | + assert_eq!(msg.domain, domain.to_string()); |
| 72 | + assert_eq!(msg.descriptors , descriptors); |
| 73 | + assert_eq!(msg.unknown_fields , UnknownFields::default()); |
| 74 | + assert_eq!(msg.cached_size , CachedSize::default()); |
| 75 | + } |
| 76 | +} |
0 commit comments