diff options
author | Bryan Newbold <bnewbold@archive.org> | 2020-08-25 18:09:18 -0700 |
---|---|---|
committer | Bryan Newbold <bnewbold@archive.org> | 2020-08-25 18:15:53 -0700 |
commit | 55f9b88ba1d63db75ef9cde3cf94e5c98526ad2f (patch) | |
tree | 68d31143996afce99dfa346042c1123dd944f766 /src | |
parent | a66caf9b7ae476f526a706d31ed5ef4f8d361b00 (diff) | |
download | es-public-proxy-55f9b88ba1d63db75ef9cde3cf94e5c98526ad2f.tar.gz es-public-proxy-55f9b88ba1d63db75ef9cde3cf94e5c98526ad2f.zip |
significant increase in parse and test coverage
Diffstat (limited to 'src')
-rw-r--r-- | src/parse.rs | 494 |
1 files changed, 481 insertions, 13 deletions
diff --git a/src/parse.rs b/src/parse.rs index 0bd1eeb..2a4c0a5 100644 --- a/src/parse.rs +++ b/src/parse.rs @@ -12,46 +12,85 @@ pub struct ApiRequest { #[derive(Serialize, Deserialize, Debug, Default)] #[serde(deny_unknown_fields)] pub struct UrlQueryParams { + #[serde(skip_serializing_if="Option::is_none")] pub allow_no_indices: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub allow_partial_search_results: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub batched_reduce_size: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] pub ccs_minimize_roundtrips: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub docvalue_fields: Option<String>, // array of strings, comma-separated + #[serde(skip_serializing_if="Option::is_none")] pub expand_wildcards: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] pub explain: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub from: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] pub ignore_throttled: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub ignore_unavailable: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub max_concurrent_shard_requests: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] pub pre_filter_shard_size: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] pub preference: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] pub q: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] pub request_cache: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub rest_total_hits_as_int: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub routing: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] pub scroll: Option<String>, // string is "time value" + #[serde(skip_serializing_if="Option::is_none")] pub search_type: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] pub seq_no_primary_term: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub size: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] pub sort: Option<String>, // array of strings, comma-separated + #[serde(skip_serializing_if="Option::is_none")] pub _source: Option<bool>, // TODO: bool or string + #[serde(skip_serializing_if="Option::is_none")] pub _source_excludes: Option<String>, // array of strings, comma-separated + #[serde(skip_serializing_if="Option::is_none")] pub _source_includes: Option<String>, // array of strings, comma-separated + #[serde(skip_serializing_if="Option::is_none")] pub stats: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] pub stored_fields: Option<String>, // array of strings, comma-separated + #[serde(skip_serializing_if="Option::is_none")] pub suggest_field: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] pub suggest_text: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] pub terminate_after: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] pub timeout: Option<String>, // string is "time units" + #[serde(skip_serializing_if="Option::is_none")] pub track_scores: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub track_total_hits: Option<bool>, // XXX: bool or integer + #[serde(skip_serializing_if="Option::is_none")] pub typed_keys: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub version: Option<bool>, // additional generic params + #[serde(skip_serializing_if="Option::is_none")] pub human: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub pretty: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub filter_path: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] pub error_trace: Option<bool>, } @@ -59,74 +98,142 @@ pub struct UrlQueryParams { #[derive(Serialize, Deserialize, Debug)] #[serde(deny_unknown_fields)] pub struct SearchBody { + #[serde(skip_serializing_if="Option::is_none")] pub query: Option<ApiQuery>, + #[serde(skip_serializing_if="Option::is_none")] pub highlight: Option<ApiHighlight>, + #[serde(skip_serializing_if="Option::is_none")] pub collapse: Option<ApiCollapse>, + #[serde(skip_serializing_if="Option::is_none")] pub post_filter: Option<ApiQuery>, // TODO: leaf query only? + #[serde(skip_serializing_if="Option::is_none")] pub rescore: Option<ApiRescore>, // TODO: single or an array of rescore objects // script_fields disabled + #[serde(skip_serializing_if="Option::is_none")] + pub aggs: Option<HashMap<String, ApiAggregation>>, + #[serde(skip_serializing_if="Option::is_none")] + pub aggregations: Option<HashMap<String, ApiAggregation>>, // https://www.elastic.co/guide/en/elasticsearch/reference/current/sort-search-results.html + #[serde(skip_serializing_if="Option::is_none")] pub sort: Option<Vec<SortElement>>, + #[serde(skip_serializing_if="Option::is_none")] pub slice: Option<ApiSlice>, + #[serde(skip_serializing_if="Option::is_none")] pub stored_fields: Option<String>, // array of strings, or "_none_" // overlap with URL query parameters + #[serde(skip_serializing_if="Option::is_none")] pub docvalue_fields: Option<Vec<DocValOrString>>, + #[serde(skip_serializing_if="Option::is_none")] pub explain: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub from: Option<u32>, - pub min_score: Option<f64>, + #[serde(skip_serializing_if="Option::is_none")] + pub min_score: Option<Num>, + #[serde(skip_serializing_if="Option::is_none")] pub seq_no_primary_term: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] pub size: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] pub _source: Option<bool>, // XXX: bool, string, or object + #[serde(skip_serializing_if="Option::is_none")] pub terminate_after: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] pub timeout: Option<String>, // string is "time units" } #[derive(Serialize, Deserialize, Debug)] #[serde(deny_unknown_fields)] pub struct ScrollBody { - pub scroll_id: String, + pub scroll_id: StringOrArray, + #[serde(skip_serializing_if="Option::is_none")] pub scroll: Option<String>, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct ApiSlice { id: u32, max: u32, + #[serde(skip_serializing_if="Option::is_none")] field: Option<String>, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct ApiRescore{ + #[serde(skip_serializing_if="Option::is_none")] pub query: Option<ApiQuery>, + #[serde(skip_serializing_if="Option::is_none")] pub window_size: Option<u32>, } // TODO: could revert to having query types as an enum, with flattening #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct ApiQuery { // compound queries #[serde(rename = "bool")] + #[serde(skip_serializing_if="Option::is_none")] bool_query: Option<BoolQuery>, + #[serde(skip_serializing_if="Option::is_none")] boosting: Option<BoostingQuery>, + #[serde(skip_serializing_if="Option::is_none")] constant_score: Option<ConstantScoreQuery>, // fulltext (leaf) queries - - // term-level (leaf) queries #[serde(rename = "match")] - match_query: Option<HashMap<String, QueryFieldOrString>>, + #[serde(skip_serializing_if="Option::is_none")] + match_query: Option<HashMap<String, MatchQueryOrString>>, + #[serde(skip_serializing_if="Option::is_none")] match_phrase: Option<HashMap<String, QueryFieldOrString>>, - query_string: Option<QueryField>, + #[serde(skip_serializing_if="Option::is_none")] + multi_match: Option<MultiMatchQuery>, + #[serde(skip_serializing_if="Option::is_none")] + query_string: Option<QueryStringQuery>, + #[serde(skip_serializing_if="Option::is_none")] + simple_query_string: Option<QueryStringQuery>, + + // term-level (leaf) queries + #[serde(skip_serializing_if="Option::is_none")] + exists: Option<SimpleFieldOrString>, + #[serde(skip_serializing_if="Option::is_none")] + match_all: Option<SimpleBoost>, + #[serde(skip_serializing_if="Option::is_none")] + match_none: Option<SimpleBoost>, + #[serde(skip_serializing_if="Option::is_none")] + ids: Option<IdsQuery>, + #[serde(skip_serializing_if="Option::is_none")] + wildcard: Option<HashMap<String, TermQueryOrString>>, // also works for wildcard + #[serde(skip_serializing_if="Option::is_none")] + prefix: Option<HashMap<String, TermQueryOrString>>, // also works for prefix query + #[serde(skip_serializing_if="Option::is_none")] + range: Option<HashMap<String, RangeQuery>>, + #[serde(skip_serializing_if="Option::is_none")] + term: Option<HashMap<String, TermQueryOrString>>, + #[serde(skip_serializing_if="Option::is_none")] + // TODO: boost in terms query + terms: Option<HashMap<String, Vec<String>>>, // other + #[serde(skip_serializing_if="Option::is_none")] nested: Option<NestedQuery>, + #[serde(skip_serializing_if="Option::is_none")] rescore_query: Option<Box<ApiQuery>>, + + // fields as part of a rescore query + #[serde(skip_serializing_if="Option::is_none")] + score_mode: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + query_weight: Option<Num>, + #[serde(skip_serializing_if="Option::is_none")] + rescore_query_weight: Option<Num>, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct ApiHighlight{ // TODO: fields could also be an array of strings? fields: HashMap<String, HighlightField>, @@ -136,13 +243,19 @@ pub struct ApiHighlight{ } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] #[serde(untagged)] pub enum SortMapValue { String(String), - Object { order: String, mode: Option<String> }, + Object { + order: String, + #[serde(skip_serializing_if="Option::is_none")] + mode: Option<String>, + }, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] #[serde(untagged)] pub enum SortElement{ String(String), @@ -150,13 +263,208 @@ pub enum SortElement{ } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] #[serde(untagged)] pub enum DocValOrString { String(String), - Object {field: String, format: Option<String>}, + Object { + field: String, + #[serde(skip_serializing_if="Option::is_none")] + format: Option<String>, + }, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +#[serde(untagged)] +pub enum MatchQueryOrString { + Object(MatchQuery), + String(String), +} + +// https://www.elastic.co/guide/en/elasticsearch/reference/7.9/query-dsl-match-query.html +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct MatchQuery { + query: String, + + #[serde(flatten)] + options: MatchOptions, +} + +// https://www.elastic.co/guide/en/elasticsearch/reference/7.9/query-dsl-multi-match-query.html +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct MultiMatchQuery { + query: String, + fields: Vec<String>, + #[serde(skip_serializing_if="Option::is_none")] + #[serde(rename="type")] + query_type: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + tie_breaker: Option<Num>, + #[serde(flatten)] + options: MatchOptions, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct MatchOptions { + #[serde(skip_serializing_if="Option::is_none")] + analyzer: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + auto_generate_synonyms_phrase_query: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + fuzziness: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + max_expansions: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] + prefix_length: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] + fuzzy_transpositions: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + fuzzy_rewrite: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + lenient: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + operator: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + minimum_should_match: Option<StringOrNum>, + #[serde(skip_serializing_if="Option::is_none")] + zero_terms_query: Option<String>, +} + +// https://www.elastic.co/guide/en/elasticsearch/reference/7.9/query-dsl-query-string-query.html +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct QueryStringQuery { + query: String, + #[serde(skip_serializing_if="Option::is_none")] + #[serde(rename="type")] + query_type: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + default_field: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + allow_leading_wildcard: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + analyze_wildcard: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + analyzer: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + auto_generate_synonyms_phrase_query: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + boost: Option<Num>, + #[serde(skip_serializing_if="Option::is_none")] + default_operator: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + enable_position_increments: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + fields: Option<Vec<String>>, + #[serde(skip_serializing_if="Option::is_none")] + fuzziness: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + fuzzy_max_expansions: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] + fuzzy_prefix_length: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] + fuzzy_transpositions: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + lenient: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + max_determinized_states: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] + minimum_should_match: Option<StringOrNum>, + #[serde(skip_serializing_if="Option::is_none")] + quote_analyzer: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + phrase_slop: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] + quote_field_suffix: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + rewrite: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + time_zone: Option<String>, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct SimpleBoost { + #[serde(skip_serializing_if="Option::is_none")] + boost: Option<Num>, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct IdsQuery { + values: Vec<String>, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +#[serde(untagged)] +pub enum TermQueryOrString { + String(String), + Object(TermQuery), +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct TermQuery { + value: String, + #[serde(skip_serializing_if="Option::is_none")] + rewrite: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + boost: Option<Num>, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +#[serde(untagged)] +pub enum StringOrNum { + String(String), + Int(u64), + Float(f64), +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +#[serde(untagged)] +pub enum Num { + Int(u64), + Float(f64), +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +#[serde(untagged)] +pub enum StringOrArray { + String(String), + Array(Vec<String>), +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct RangeQuery { + #[serde(skip_serializing_if="Option::is_none")] + gt: Option<StringOrNum>, + #[serde(skip_serializing_if="Option::is_none")] + gte: Option<StringOrNum>, + #[serde(skip_serializing_if="Option::is_none")] + lt: Option<StringOrNum>, + #[serde(skip_serializing_if="Option::is_none")] + lte: Option<StringOrNum>, + #[serde(skip_serializing_if="Option::is_none")] + format: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + relation: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + timezone: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + boost: Option<Num>, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] #[serde(untagged)] pub enum QueryFieldOrString { Object(QueryField), @@ -164,77 +472,128 @@ pub enum QueryFieldOrString { } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct QueryField{ query: String, + #[serde(skip_serializing_if="Option::is_none")] fuzziness: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] slop: Option<u32>, - boost: Option<f64>, + #[serde(skip_serializing_if="Option::is_none")] + boost: Option<Num>, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct BoolQuery { + #[serde(skip_serializing_if="Option::is_none")] must: Option<Box<ApiQuery>>, + #[serde(skip_serializing_if="Option::is_none")] filter: Option<Box<ApiQuery>>, + #[serde(skip_serializing_if="Option::is_none")] should: Option<Box<ApiQuery>>, + #[serde(skip_serializing_if="Option::is_none")] must_not: Option<Box<ApiQuery>>, - minimum_should_match: Option<u32>, - boost: Option<f64>, + #[serde(skip_serializing_if="Option::is_none")] + minimum_should_match: Option<StringOrNum>, + #[serde(skip_serializing_if="Option::is_none")] + boost: Option<Num>, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct NestedQuery { path: String, query: Box<ApiQuery>, + #[serde(skip_serializing_if="Option::is_none")] score_mode: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] ignore_unmapped: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + inner_hits: Option<InnerHitsOneOrMore>, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct BoostingQuery { positive: Box<ApiQuery>, negative: Box<ApiQuery>, - negative_boost: f64, + negative_boost: Num, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct ConstantScoreQuery { filter: Box<ApiQuery>, - boost: Option<f64>, + #[serde(skip_serializing_if="Option::is_none")] + boost: Option<Num>, } // https://www.elastic.co/guide/en/elasticsearch/reference/current/highlighting.html #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct HighlightField{ + #[serde(skip_serializing_if="Option::is_none")] boundary_chars: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] boundary_max_scan: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] boundary_scanner: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] boundary_scanner_locale: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] encoder: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] force_source: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] fragmenter: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] fragment_offset: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] fragment_size: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] highlight_query: Option<ApiQuery>, + #[serde(skip_serializing_if="Option::is_none")] matched_fields: Option<Vec<String>>, + #[serde(skip_serializing_if="Option::is_none")] no_match_size: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] number_of_fragments: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] order: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] phrase_limit: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] pre_tags: Option<Vec<String>>, + #[serde(skip_serializing_if="Option::is_none")] post_tags: Option<Vec<String>>, + #[serde(skip_serializing_if="Option::is_none")] require_field_match: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] tags_schema: Option<String>, #[serde(rename = "type")] + #[serde(skip_serializing_if="Option::is_none")] highlight_type: Option<String>, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +#[serde(untagged)] +pub enum SimpleFieldOrString { + String(String), + Object { field: String }, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct ApiCollapse{ field: String, + #[serde(skip_serializing_if="Option::is_none")] inner_hits: Option<InnerHitsOneOrMore>, } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] #[serde(untagged)] pub enum InnerHitsOneOrMore { Single(InnerHits), @@ -242,9 +601,118 @@ pub enum InnerHitsOneOrMore { } #[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] pub struct InnerHits { + #[serde(skip_serializing_if="Option::is_none")] from: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] size: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] sort: Option<Vec<SortElement>>, + #[serde(skip_serializing_if="Option::is_none")] name: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + collapse: Option<Box<ApiCollapse>>, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct ApiAggregation { + // bucket type aggregations + #[serde(skip_serializing_if="Option::is_none")] + nested: Option<NestedAggregation>, + #[serde(skip_serializing_if="Option::is_none")] + filter: Option<ApiQuery>, + #[serde(skip_serializing_if="Option::is_none")] + histogram: Option<SimpleAggregation>, + #[serde(skip_serializing_if="Option::is_none")] + terms: Option<TermsAggregation>, + #[serde(skip_serializing_if="Option::is_none")] + significant_terms: Option<SimpleAggregation>, + + // metrics type aggregations + #[serde(skip_serializing_if="Option::is_none")] + avg: Option<SimpleAggregation>, + #[serde(skip_serializing_if="Option::is_none")] + min: Option<SimpleAggregation>, + #[serde(skip_serializing_if="Option::is_none")] + max: Option<SimpleAggregation>, + #[serde(skip_serializing_if="Option::is_none")] + sum: Option<SimpleAggregation>, + #[serde(skip_serializing_if="Option::is_none")] + value_count: Option<SimpleAggregation>, + #[serde(skip_serializing_if="Option::is_none")] + stats: Option<SimpleAggregation>, + #[serde(skip_serializing_if="Option::is_none")] + percentiles: Option<SimpleAggregation>, + + // nested aggregations + #[serde(skip_serializing_if="Option::is_none")] + aggs: Option<HashMap<String, Box<ApiAggregation>>>, + #[serde(skip_serializing_if="Option::is_none")] + aggregations: Option<HashMap<String, Box<ApiAggregation>>>, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct NestedAggregation{ + path: String, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct SimpleAggregation{ + field: String, + #[serde(skip_serializing_if="Option::is_none")] + interval: Option<StringOrNum>, + #[serde(skip_serializing_if="Option::is_none")] + missing: Option<StringOrNum>, + #[serde(skip_serializing_if="Option::is_none")] + keyed: Option<bool>, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct DateHistogramAggregation{ + field: String, + #[serde(skip_serializing_if="Option::is_none")] + fixed_interval: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + calendar_interval: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + format: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + time_zone: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + offset: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + order: Option<HashMap<String, String>>, + #[serde(skip_serializing_if="Option::is_none")] + keyed: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + missing: Option<StringOrNum>, +} + +#[derive(Serialize, Deserialize, Debug)] +#[serde(deny_unknown_fields)] +pub struct TermsAggregation { + field: String, + #[serde(skip_serializing_if="Option::is_none")] + size: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] + shard_size: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] + min_doc_count: Option<u32>, + #[serde(skip_serializing_if="Option::is_none")] + show_term_doc_count_error: Option<bool>, + #[serde(skip_serializing_if="Option::is_none")] + order: Option<HashMap<String, String>>, + #[serde(skip_serializing_if="Option::is_none")] + include: Option<StringOrArray>, + #[serde(skip_serializing_if="Option::is_none")] + exclude: Option<StringOrArray>, + #[serde(skip_serializing_if="Option::is_none")] + execution_hint: Option<String>, + #[serde(skip_serializing_if="Option::is_none")] + missing: Option<StringOrNum>, } |