1
/*
2
 * Hurl (https://hurl.dev)
3
 * Copyright (C) 2026 Orange
4
 *
5
 * Licensed under the Apache License, Version 2.0 (the "License");
6
 * you may not use this file except in compliance with the License.
7
 * You may obtain a copy of the License at
8
 *
9
 *          http://www.apache.org/licenses/LICENSE-2.0
10
 *
11
 * Unless required by applicable law or agreed to in writing, software
12
 * distributed under the License is distributed on an "AS IS" BASIS,
13
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
 * See the License for the specific language governing permissions and
15
 * limitations under the License.
16
 *
17
 */
18
use base64::Engine;
19
use base64::engine::general_purpose;
20
use hurl_core::ast::{
21
    Assert, Base64, Body, BooleanOption, Bytes, Capture, CertificateAttributeName, Comment, Cookie,
22
    CountOption, Duration, DurationOption, Entry, EntryOption, File, FilenameParam, Filter,
23
    FilterValue, Hex, HurlFile, JsonListElement, JsonValue, KeyValue, MultilineString,
24
    MultilineStringKind, MultipartParam, NaturalOption, OptionKind, Placeholder, Predicate,
25
    PredicateFuncValue, PredicateValue, Query, QueryValue, Regex, RegexValue, Request, Response,
26
    StatusValue, VersionValue,
27
};
28
use hurl_core::types::{Count, ToSource};
29

            
30
use crate::format::serialize_json::JValue;
31

            
32
57
pub fn format(hurl_file: &HurlFile) -> String {
33
57
    hurl_file.to_json().format()
34
}
35

            
36
pub trait ToJson {
37
    fn to_json(&self) -> JValue;
38
}
39

            
40
impl ToJson for HurlFile {
41
57
    fn to_json(&self) -> JValue {
42
57
        JValue::Object(vec![(
43
57
            "entries".to_string(),
44
211
            JValue::List(self.entries.iter().map(|e| e.to_json()).collect()),
45
        )])
46
    }
47
}
48

            
49
impl ToJson for Entry {
50
192
    fn to_json(&self) -> JValue {
51
192
        let mut attributes = vec![("request".to_string(), self.request.to_json())];
52
192
        if let Some(response) = &self.response {
53
87
            attributes.push(("response".to_string(), response.to_json()));
54
        }
55
192
        JValue::Object(attributes)
56
    }
57
}
58

            
59
impl ToJson for Request {
60
192
    fn to_json(&self) -> JValue {
61
192
        let mut attributes = vec![
62
192
            (
63
192
                "method".to_string(),
64
192
                JValue::String(self.method.to_string()),
65
192
            ),
66
192
            ("url".to_string(), JValue::String(self.url.to_string())),
67
        ];
68
192
        add_headers(&mut attributes, &self.headers);
69

            
70
192
        if !self.querystring_params().is_empty() {
71
6
            let params = self
72
6
                .querystring_params()
73
6
                .iter()
74
17
                .map(|p| p.to_json())
75
6
                .collect();
76
6
            attributes.push(("query_string_params".to_string(), JValue::List(params)));
77
        }
78
192
        if !self.form_params().is_empty() {
79
13
            let params = self.form_params().iter().map(|p| p.to_json()).collect();
80
3
            attributes.push(("form_params".to_string(), JValue::List(params)));
81
        }
82
192
        if !self.multipart_form_data().is_empty() {
83
3
            let params = self
84
3
                .multipart_form_data()
85
3
                .iter()
86
10
                .map(|p| p.to_json())
87
3
                .collect();
88
3
            attributes.push(("multipart_form_data".to_string(), JValue::List(params)));
89
        }
90
192
        if !self.cookies().is_empty() {
91
7
            let cookies = self.cookies().iter().map(|c| c.to_json()).collect();
92
3
            attributes.push(("cookies".to_string(), JValue::List(cookies)));
93
        }
94
192
        if !self.options().is_empty() {
95
322
            let options = self.options().iter().map(|c| c.to_json()).collect();
96
12
            attributes.push(("options".to_string(), JValue::List(options)));
97
        }
98
192
        if let Some(body) = &self.body {
99
48
            attributes.push(("body".to_string(), body.to_json()));
100
        }
101

            
102
        // Request comments (can be used to check custom commands)
103
192
        let comments: Vec<_> = self
104
192
            .line_terminators
105
192
            .iter()
106
233
            .filter_map(|l| l.comment.as_ref())
107
192
            .collect();
108
192
        if !comments.is_empty() {
109
26
            let comments = comments.iter().map(|c| c.to_json()).collect();
110
15
            attributes.push(("comments".to_string(), JValue::List(comments)));
111
        }
112

            
113
192
        JValue::Object(attributes)
114
    }
115
}
116

            
117
impl ToJson for Response {
118
    /// Transforms this response to a JSON object.
119
87
    fn to_json(&self) -> JValue {
120
87
        let mut attributes = vec![];
121
87
        if let Some(v) = get_json_version(&self.version.value) {
122
12
            attributes.push(("version".to_string(), JValue::String(v)));
123
        }
124
87
        if let StatusValue::Specific(n) = self.status.value {
125
87
            attributes.push(("status".to_string(), JValue::Number(n.to_string())));
126
        }
127
87
        add_headers(&mut attributes, &self.headers);
128
87
        if !self.captures().is_empty() {
129
21
            let captures = self.captures().iter().map(|c| c.to_json()).collect();
130
9
            attributes.push(("captures".to_string(), JValue::List(captures)));
131
        }
132
87
        if !self.asserts().is_empty() {
133
387
            let asserts = self.asserts().iter().map(|a| a.to_json()).collect();
134
36
            attributes.push(("asserts".to_string(), JValue::List(asserts)));
135
        }
136
87
        if let Some(body) = &self.body {
137
24
            attributes.push(("body".to_string(), body.to_json()));
138
        }
139
87
        JValue::Object(attributes)
140
    }
141
}
142

            
143
279
fn add_headers(attributes: &mut Vec<(String, JValue)>, headers: &[KeyValue]) {
144
279
    if !headers.is_empty() {
145
62
        let headers = JValue::List(headers.iter().map(|h| h.to_json()).collect());
146
24
        attributes.push(("headers".to_string(), headers));
147
    }
148
}
149

            
150
impl ToJson for Body {
151
72
    fn to_json(&self) -> JValue {
152
72
        self.value.to_json()
153
    }
154
}
155

            
156
impl ToJson for Bytes {
157
72
    fn to_json(&self) -> JValue {
158
72
        match self {
159
6
            Bytes::Base64(value) => value.to_json(),
160
3
            Bytes::Hex(value) => value.to_json(),
161
3
            Bytes::File(value) => value.to_json(),
162
6
            Bytes::Json(value) => JValue::Object(vec![
163
6
                ("type".to_string(), JValue::String("json".to_string())),
164
6
                ("value".to_string(), value.to_json()),
165
6
            ]),
166
3
            Bytes::Xml(value) => JValue::Object(vec![
167
3
                ("type".to_string(), JValue::String("xml".to_string())),
168
3
                ("value".to_string(), JValue::String(value.clone())),
169
3
            ]),
170
9
            Bytes::OnelineString(value) => JValue::Object(vec![
171
9
                ("type".to_string(), JValue::String("text".to_string())),
172
9
                ("value".to_string(), JValue::String(value.to_string())),
173
9
            ]),
174
42
            Bytes::MultilineString(multi) => {
175
                // TODO: check these values. Maybe we want to have the same
176
                // export when using:
177
                //
178
                // ~~~
179
                // GET https://foo.com
180
                // ```base64
181
                // SGVsbG8gd29ybGQ=
182
                // ```
183
                //
184
                // or
185
                //
186
                // ~~~
187
                // GET https://foo.com
188
                // base64,SGVsbG8gd29ybGQ=;
189
                // ~~~
190
42
                let lang = match multi {
191
                    MultilineString {
192
                        kind: MultilineStringKind::Text(_),
193
                        ..
194
9
                    } => "text",
195
                    MultilineString {
196
                        kind: MultilineStringKind::Raw(_),
197
                        ..
198
9
                    } => "raw",
199
                    MultilineString {
200
                        kind: MultilineStringKind::Json(_),
201
                        ..
202
9
                    } => "json",
203
                    MultilineString {
204
                        kind: MultilineStringKind::Xml(_),
205
                        ..
206
6
                    } => "xml",
207
                    MultilineString {
208
                        kind: MultilineStringKind::GraphQl(_),
209
                        ..
210
9
                    } => "graphql",
211
                };
212
42
                JValue::Object(vec![
213
42
                    ("type".to_string(), JValue::String(lang.to_string())),
214
42
                    ("value".to_string(), JValue::String(multi.to_string())),
215
42
                ])
216
            }
217
        }
218
    }
219
}
220

            
221
impl ToJson for Base64 {
222
6
    fn to_json(&self) -> JValue {
223
6
        let value = general_purpose::STANDARD.encode(&self.value);
224
6
        JValue::Object(vec![
225
6
            ("encoding".to_string(), JValue::String("base64".to_string())),
226
6
            ("value".to_string(), JValue::String(value)),
227
6
        ])
228
    }
229
}
230

            
231
impl ToJson for Hex {
232
3
    fn to_json(&self) -> JValue {
233
3
        let value = general_purpose::STANDARD.encode(&self.value);
234
3
        JValue::Object(vec![
235
3
            ("encoding".to_string(), JValue::String("base64".to_string())),
236
3
            ("value".to_string(), JValue::String(value)),
237
3
        ])
238
    }
239
}
240

            
241
impl ToJson for File {
242
6
    fn to_json(&self) -> JValue {
243
6
        JValue::Object(vec![
244
6
            ("type".to_string(), JValue::String("file".to_string())),
245
6
            (
246
6
                "filename".to_string(),
247
6
                JValue::String(self.filename.to_string()),
248
6
            ),
249
6
        ])
250
    }
251
}
252

            
253
87
fn get_json_version(version_value: &VersionValue) -> Option<String> {
254
87
    match version_value {
255
3
        VersionValue::Version1 => Some("HTTP/1.0".to_string()),
256
3
        VersionValue::Version11 => Some("HTTP/1.1".to_string()),
257
3
        VersionValue::Version2 => Some("HTTP/2".to_string()),
258
3
        VersionValue::Version3 => Some("HTTP/3".to_string()),
259
75
        VersionValue::VersionAny => None,
260
    }
261
}
262

            
263
impl ToJson for KeyValue {
264
84
    fn to_json(&self) -> JValue {
265
84
        let attributes = vec![
266
84
            ("name".to_string(), JValue::String(self.key.to_string())),
267
84
            ("value".to_string(), JValue::String(self.value.to_string())),
268
        ];
269
84
        JValue::Object(attributes)
270
    }
271
}
272

            
273
impl ToJson for MultipartParam {
274
9
    fn to_json(&self) -> JValue {
275
9
        match self {
276
3
            MultipartParam::Param(param) => param.to_json(),
277
6
            MultipartParam::FilenameParam(param) => param.to_json(),
278
        }
279
    }
280
}
281

            
282
impl ToJson for FilenameParam {
283
6
    fn to_json(&self) -> JValue {
284
6
        let mut attributes = vec![
285
6
            ("name".to_string(), JValue::String(self.key.to_string())),
286
6
            (
287
6
                "filename".to_string(),
288
6
                JValue::String(self.value.filename.to_string()),
289
6
            ),
290
        ];
291
6
        if let Some(content_type) = &self.value.content_type {
292
3
            attributes.push((
293
3
                "content_type".to_string(),
294
3
                JValue::String(content_type.to_string()),
295
3
            ));
296
        }
297
6
        JValue::Object(attributes)
298
    }
299
}
300

            
301
impl ToJson for Cookie {
302
6
    fn to_json(&self) -> JValue {
303
6
        let attributes = vec![
304
6
            ("name".to_string(), JValue::String(self.name.to_string())),
305
6
            ("value".to_string(), JValue::String(self.value.to_string())),
306
        ];
307
6
        JValue::Object(attributes)
308
    }
309
}
310

            
311
impl ToJson for EntryOption {
312
318
    fn to_json(&self) -> JValue {
313
318
        let value = match &self.kind {
314
6
            OptionKind::AwsSigV4(value) => JValue::String(value.to_string()),
315
6
            OptionKind::CaCertificate(filename) => JValue::String(filename.to_string()),
316
9
            OptionKind::ClientCert(filename) => JValue::String(filename.to_string()),
317
6
            OptionKind::ClientKey(filename) => JValue::String(filename.to_string()),
318
6
            OptionKind::Compressed(value) => value.to_json(),
319
6
            OptionKind::ConnectTo(value) => JValue::String(value.to_string()),
320
6
            OptionKind::ConnectTimeout(value) => value.to_json(),
321
18
            OptionKind::Delay(value) => value.to_json(),
322
6
            OptionKind::Digest(value) => value.to_json(),
323
6
            OptionKind::FailWithBody(value) => value.to_json(),
324
6
            OptionKind::FollowLocation(value) => value.to_json(),
325
6
            OptionKind::FollowLocationTrusted(value) => value.to_json(),
326
6
            OptionKind::Header(value) => JValue::String(value.to_string()),
327
6
            OptionKind::Http10(value) => value.to_json(),
328
6
            OptionKind::Http11(value) => value.to_json(),
329
6
            OptionKind::Http2(value) => value.to_json(),
330
6
            OptionKind::Http3(value) => value.to_json(),
331
6
            OptionKind::Insecure(value) => value.to_json(),
332
6
            OptionKind::IpV4(value) => value.to_json(),
333
6
            OptionKind::IpV6(value) => value.to_json(),
334
6
            OptionKind::LimitRate(value) => value.to_json(),
335
6
            OptionKind::MaxRedirect(value) => value.to_json(),
336
6
            OptionKind::MaxTime(value) => value.to_json(),
337
6
            OptionKind::Negotiate(value) => value.to_json(),
338
6
            OptionKind::NetRc(value) => value.to_json(),
339
6
            OptionKind::NetRcFile(filename) => JValue::String(filename.to_string()),
340
6
            OptionKind::NetRcOptional(value) => value.to_json(),
341
6
            OptionKind::NoHeader(value) => JValue::String(value.to_string()),
342
6
            OptionKind::Ntlm(value) => value.to_json(),
343
6
            OptionKind::Output(filename) => JValue::String(filename.to_string()),
344
6
            OptionKind::PathAsIs(value) => value.to_json(),
345
6
            OptionKind::PinnedPublicKey(value) => JValue::String(value.to_string()),
346
6
            OptionKind::Proxy(value) => JValue::String(value.to_string()),
347
9
            OptionKind::Repeat(value) => value.to_json(),
348
6
            OptionKind::Resolve(value) => JValue::String(value.to_string()),
349
12
            OptionKind::Retry(value) => value.to_json(),
350
12
            OptionKind::RetryInterval(value) => value.to_json(),
351
6
            OptionKind::Skip(value) => value.to_json(),
352
6
            OptionKind::UnixSocket(value) => JValue::String(value.to_string()),
353
6
            OptionKind::User(value) => JValue::String(value.to_string()),
354
27
            OptionKind::Variable(value) => {
355
27
                JValue::String(format!("{}={}", value.name, value.value.to_source()))
356
            }
357
9
            OptionKind::Verbose(value) => value.to_json(),
358
6
            OptionKind::Verbosity(value) => JValue::String(value.to_string()),
359
6
            OptionKind::VeryVerbose(value) => value.to_json(),
360
        };
361

            
362
        // If the value contains the unit such as `{ "value": 10, "unit": "second" }`
363
        // The JSON for this option should still have one level
364
        // for example: { "name": "delay", "value": 10, "unit", "second" }
365
318
        let attributes = if let JValue::Object(mut attributes) = value {
366
24
            attributes.push((
367
24
                "name".to_string(),
368
24
                JValue::String(self.kind.identifier().to_string()),
369
24
            ));
370
24
            attributes
371
        } else {
372
294
            vec![
373
294
                (
374
294
                    "name".to_string(),
375
294
                    JValue::String(self.kind.identifier().to_string()),
376
294
                ),
377
294
                ("value".to_string(), value),
378
            ]
379
        };
380
318
        JValue::Object(attributes)
381
    }
382
}
383

            
384
impl ToJson for BooleanOption {
385
123
    fn to_json(&self) -> JValue {
386
123
        match self {
387
63
            BooleanOption::Literal(value) => JValue::Boolean(*value),
388
60
            BooleanOption::Placeholder(placeholder) => placeholder.to_json(),
389
        }
390
    }
391
}
392

            
393
impl ToJson for CountOption {
394
27
    fn to_json(&self) -> JValue {
395
27
        match self {
396
18
            CountOption::Literal(value) => value.to_json(),
397
9
            CountOption::Placeholder(placeholder) => placeholder.to_json(),
398
        }
399
    }
400
}
401

            
402
impl ToJson for Count {
403
18
    fn to_json(&self) -> JValue {
404
18
        match self {
405
12
            Count::Finite(n) => JValue::Number(n.to_string()),
406
6
            Count::Infinite => JValue::Number("-1".to_string()),
407
        }
408
    }
409
}
410

            
411
impl ToJson for DurationOption {
412
42
    fn to_json(&self) -> JValue {
413
42
        match self {
414
30
            DurationOption::Literal(value) => value.to_json(),
415
12
            DurationOption::Placeholder(placeholder) => placeholder.to_json(),
416
        }
417
    }
418
}
419

            
420
impl ToJson for Duration {
421
30
    fn to_json(&self) -> JValue {
422
30
        if let Some(unit) = self.unit {
423
24
            let mut attributes =
424
24
                vec![("value".to_string(), JValue::Number(self.value.to_string()))];
425
24
            attributes.push(("unit".to_string(), JValue::String(unit.to_string())));
426
24
            JValue::Object(attributes)
427
        } else {
428
6
            JValue::Number(self.value.to_string())
429
        }
430
    }
431
}
432

            
433
impl ToJson for Capture {
434
18
    fn to_json(&self) -> JValue {
435
18
        let mut attributes = vec![
436
18
            ("name".to_string(), JValue::String(self.name.to_string())),
437
18
            ("query".to_string(), self.query.to_json()),
438
        ];
439
18
        if !self.filters.is_empty() {
440
4
            let filters = JValue::List(self.filters.iter().map(|(_, f)| f.to_json()).collect());
441
3
            attributes.push(("filters".to_string(), filters));
442
        }
443
18
        if self.redacted {
444
6
            attributes.push(("redact".to_string(), JValue::Boolean(true)));
445
        }
446
18
        JValue::Object(attributes)
447
    }
448
}
449

            
450
impl ToJson for Assert {
451
375
    fn to_json(&self) -> JValue {
452
375
        let mut attributes = vec![("query".to_string(), self.query.to_json())];
453
375
        if !self.filters.is_empty() {
454
183
            let filters = JValue::List(self.filters.iter().map(|(_, f)| f.to_json()).collect());
455
117
            attributes.push(("filters".to_string(), filters));
456
        }
457
375
        attributes.push(("predicate".to_string(), self.predicate.to_json()));
458
375
        JValue::Object(attributes)
459
    }
460
}
461

            
462
impl ToJson for Query {
463
393
    fn to_json(&self) -> JValue {
464
393
        let attributes = query_value_attributes(&self.value);
465
393
        JValue::Object(attributes)
466
    }
467
}
468

            
469
393
fn query_value_attributes(query_value: &QueryValue) -> Vec<(String, JValue)> {
470
393
    let mut attributes = vec![];
471
393
    let att_type = JValue::String(query_value.identifier().to_string());
472
393
    attributes.push(("type".to_string(), att_type));
473

            
474
393
    match query_value {
475
243
        QueryValue::Jsonpath { expr, .. } => {
476
243
            attributes.push(("expr".to_string(), JValue::String(expr.to_string())));
477
        }
478
6
        QueryValue::Header { name, .. } => {
479
6
            attributes.push(("name".to_string(), JValue::String(name.to_string())));
480
        }
481
9
        QueryValue::Cookie { expr, .. } => {
482
9
            attributes.push(("expr".to_string(), JValue::String(expr.to_string())));
483
        }
484
3
        QueryValue::Xpath { expr, .. } => {
485
3
            attributes.push(("expr".to_string(), JValue::String(expr.to_string())));
486
        }
487
3
        QueryValue::Regex { value, .. } => {
488
3
            attributes.push(("expr".to_string(), value.to_json()));
489
        }
490
9
        QueryValue::Variable { name, .. } => {
491
9
            attributes.push(("name".to_string(), JValue::String(name.to_string())));
492
        }
493
        QueryValue::Certificate {
494
33
            attribute_name: field,
495
            ..
496
33
        } => {
497
33
            attributes.push(("expr".to_string(), field.to_json()));
498
        }
499
87
        _ => {}
500
    };
501
393
    attributes
502
}
503

            
504
impl ToJson for RegexValue {
505
9
    fn to_json(&self) -> JValue {
506
9
        match self {
507
3
            RegexValue::Template(template) => JValue::String(template.to_string()),
508
6
            RegexValue::Regex(regex) => regex.to_json(),
509
        }
510
    }
511
}
512

            
513
impl ToJson for Regex {
514
6
    fn to_json(&self) -> JValue {
515
6
        let attributes = vec![
516
6
            ("type".to_string(), JValue::String("regex".to_string())),
517
6
            ("value".to_string(), JValue::String(self.to_string())),
518
        ];
519
6
        JValue::Object(attributes)
520
    }
521
}
522

            
523
impl ToJson for CertificateAttributeName {
524
33
    fn to_json(&self) -> JValue {
525
33
        JValue::String(self.identifier().to_string())
526
    }
527
}
528

            
529
impl ToJson for Predicate {
530
375
    fn to_json(&self) -> JValue {
531
375
        let mut attributes = vec![];
532
375
        if self.not {
533
3
            attributes.push(("not".to_string(), JValue::Boolean(true)));
534
        }
535
375
        let identifier = self.predicate_func.value.identifier();
536
375
        attributes.push(("type".to_string(), JValue::String(identifier.to_string())));
537

            
538
375
        match &self.predicate_func.value {
539
252
            PredicateFuncValue::Equal { value, .. } => add_predicate_value(&mut attributes, value),
540
9
            PredicateFuncValue::NotEqual { value, .. } => {
541
9
                add_predicate_value(&mut attributes, value);
542
            }
543
9
            PredicateFuncValue::GreaterThan { value, .. } => {
544
9
                add_predicate_value(&mut attributes, value);
545
            }
546
3
            PredicateFuncValue::GreaterThanOrEqual { value, .. } => {
547
3
                add_predicate_value(&mut attributes, value);
548
            }
549
9
            PredicateFuncValue::LessThan { value, .. } => {
550
9
                add_predicate_value(&mut attributes, value);
551
            }
552
3
            PredicateFuncValue::LessThanOrEqual { value, .. } => {
553
3
                add_predicate_value(&mut attributes, value);
554
            }
555
9
            PredicateFuncValue::StartWith { value, .. } => {
556
9
                add_predicate_value(&mut attributes, value);
557
            }
558
6
            PredicateFuncValue::EndWith { value, .. } => {
559
6
                add_predicate_value(&mut attributes, value);
560
            }
561
9
            PredicateFuncValue::Contain { value, .. } => {
562
9
                add_predicate_value(&mut attributes, value);
563
            }
564
3
            PredicateFuncValue::Include { value, .. } => {
565
3
                add_predicate_value(&mut attributes, value);
566
            }
567
9
            PredicateFuncValue::Match { value, .. } => {
568
9
                add_predicate_value(&mut attributes, value);
569
            }
570
            PredicateFuncValue::Exist
571
            | PredicateFuncValue::IsBoolean
572
            | PredicateFuncValue::IsCollection
573
            | PredicateFuncValue::IsDate
574
            | PredicateFuncValue::IsEmpty
575
            | PredicateFuncValue::IsFloat
576
            | PredicateFuncValue::IsInteger
577
            | PredicateFuncValue::IsIpv4
578
            | PredicateFuncValue::IsIpv6
579
            | PredicateFuncValue::IsIsoDate
580
            | PredicateFuncValue::IsList
581
            | PredicateFuncValue::IsNumber
582
            | PredicateFuncValue::IsObject
583
            | PredicateFuncValue::IsString
584
54
            | PredicateFuncValue::IsUuid => {}
585
        }
586
375
        JValue::Object(attributes)
587
    }
588
}
589

            
590
321
fn add_predicate_value(attributes: &mut Vec<(String, JValue)>, predicate_value: &PredicateValue) {
591
321
    let (value, encoding) = json_predicate_value(predicate_value);
592
321
    attributes.push(("value".to_string(), value));
593
321
    if let Some(encoding) = encoding {
594
36
        attributes.push(("encoding".to_string(), JValue::String(encoding)));
595
    }
596
}
597

            
598
321
fn json_predicate_value(predicate_value: &PredicateValue) -> (JValue, Option<String>) {
599
321
    match predicate_value {
600
150
        PredicateValue::String(value) => (JValue::String(value.to_string()), None),
601
18
        PredicateValue::MultilineString(value) => (JValue::String(value.value().to_string()), None),
602
3
        PredicateValue::Bool(value) => (JValue::Boolean(*value), None),
603
3
        PredicateValue::Null => (JValue::Null, None),
604
105
        PredicateValue::Number(value) => (JValue::Number(value.to_string()), None),
605
3
        PredicateValue::File(value) => (value.to_json(), None),
606
27
        PredicateValue::Hex(value) => {
607
27
            let base64_string = general_purpose::STANDARD.encode(value.value.clone());
608
27
            (JValue::String(base64_string), Some("base64".to_string()))
609
        }
610
3
        PredicateValue::Base64(value) => {
611
3
            let base64_string = general_purpose::STANDARD.encode(value.value.clone());
612
3
            (JValue::String(base64_string), Some("base64".to_string()))
613
        }
614
3
        PredicateValue::Placeholder(value) => (JValue::String(value.to_string()), None),
615
6
        PredicateValue::Regex(value) => {
616
6
            (JValue::String(value.to_string()), Some("regex".to_string()))
617
        }
618
    }
619
}
620

            
621
impl ToJson for JsonValue {
622
102
    fn to_json(&self) -> JValue {
623
102
        match self {
624
3
            JsonValue::Null => JValue::Null,
625
45
            JsonValue::Number(s) => JValue::Number(s.to_string()),
626
18
            JsonValue::String(s) => JValue::String(s.to_string()),
627
3
            JsonValue::Boolean(v) => JValue::Boolean(*v),
628
15
            JsonValue::List { elements, .. } => {
629
56
                JValue::List(elements.iter().map(|e| e.to_json()).collect())
630
            }
631
15
            JsonValue::Object { elements, .. } => JValue::Object(
632
15
                elements
633
15
                    .iter()
634
50
                    .map(|elem| (elem.name.to_string(), elem.value.to_json()))
635
15
                    .collect(),
636
            ),
637
3
            JsonValue::Placeholder(exp) => JValue::String(format!("{{{{{exp}}}}}")),
638
        }
639
    }
640
}
641

            
642
impl ToJson for JsonListElement {
643
51
    fn to_json(&self) -> JValue {
644
51
        self.value.to_json()
645
    }
646
}
647

            
648
impl ToJson for Filter {
649
147
    fn to_json(&self) -> JValue {
650
147
        self.value.to_json()
651
    }
652
}
653

            
654
impl ToJson for FilterValue {
655
147
    fn to_json(&self) -> JValue {
656
147
        let mut attributes = vec![];
657
147
        let att_name = "type".to_string();
658
147
        let att_value = JValue::String(self.identifier().to_string());
659
147
        attributes.push((att_name, att_value));
660

            
661
147
        match self {
662
3
            FilterValue::Decode { encoding, .. } => {
663
3
                attributes.push(("encoding".to_string(), JValue::String(encoding.to_string())));
664
            }
665
6
            FilterValue::Format { fmt, .. } => {
666
6
                attributes.push(("fmt".to_string(), JValue::String(fmt.to_string())));
667
            }
668
9
            FilterValue::DateFormat { fmt, .. } => {
669
9
                attributes.push(("fmt".to_string(), JValue::String(fmt.to_string())));
670
            }
671
9
            FilterValue::JsonPath { expr, .. } => {
672
9
                attributes.push(("expr".to_string(), JValue::String(expr.to_string())));
673
            }
674
3
            FilterValue::Nth { n, .. } => {
675
3
                attributes.push(("n".to_string(), JValue::Number(n.to_string())));
676
            }
677
3
            FilterValue::Regex { value, .. } => {
678
3
                attributes.push(("expr".to_string(), value.to_json()));
679
            }
680
            FilterValue::Replace {
681
15
                old_value,
682
15
                new_value,
683
                ..
684
15
            } => {
685
15
                attributes.push((
686
15
                    "old_value".to_string(),
687
15
                    JValue::String(old_value.to_string()),
688
15
                ));
689
15
                attributes.push((
690
15
                    "new_value".to_string(),
691
15
                    JValue::String(new_value.to_string()),
692
15
                ));
693
            }
694
            FilterValue::ReplaceRegex {
695
3
                pattern, new_value, ..
696
3
            } => {
697
3
                attributes.push(("pattern".to_string(), pattern.to_json()));
698
3
                attributes.push((
699
3
                    "new_value".to_string(),
700
3
                    JValue::String(new_value.to_string()),
701
3
                ));
702
            }
703
3
            FilterValue::Split { sep, .. } => {
704
3
                attributes.push(("sep".to_string(), JValue::String(sep.to_string())));
705
            }
706
3
            FilterValue::ToDate { fmt, .. } => {
707
3
                attributes.push(("fmt".to_string(), JValue::String(fmt.to_string())));
708
            }
709
3
            FilterValue::UrlQueryParam { param, .. } => {
710
3
                attributes.push(("param".to_string(), JValue::String(param.to_string())));
711
            }
712
3
            FilterValue::XPath { expr, .. } => {
713
3
                attributes.push(("expr".to_string(), JValue::String(expr.to_string())));
714
            }
715
84
            _ => {}
716
        }
717
147
        JValue::Object(attributes)
718
    }
719
}
720

            
721
impl ToJson for Placeholder {
722
84
    fn to_json(&self) -> JValue {
723
84
        JValue::String(format!("{{{{{self}}}}}"))
724
    }
725
}
726

            
727
impl ToJson for Comment {
728
21
    fn to_json(&self) -> JValue {
729
21
        JValue::String(self.value.to_string())
730
    }
731
}
732

            
733
impl ToJson for NaturalOption {
734
6
    fn to_json(&self) -> JValue {
735
6
        match self {
736
3
            NaturalOption::Literal(value) => JValue::Number(value.to_string()),
737
3
            NaturalOption::Placeholder(placeholder) => placeholder.to_json(),
738
        }
739
    }
740
}
741
#[cfg(test)]
742
pub mod tests {
743
    use hurl_core::ast::{
744
        I64, LineTerminator, Method, Number, PredicateFunc, SourceInfo, Status, Template,
745
        TemplateElement, Version, Whitespace,
746
    };
747
    use hurl_core::reader::Pos;
748
    use hurl_core::types::ToSource;
749

            
750
    use super::*;
751

            
752
    fn whitespace() -> Whitespace {
753
        Whitespace {
754
            value: String::new(),
755
            source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
756
        }
757
    }
758

            
759
    fn line_terminator() -> LineTerminator {
760
        LineTerminator {
761
            space0: whitespace(),
762
            comment: None,
763
            newline: whitespace(),
764
        }
765
    }
766

            
767
    #[test]
768
    pub fn test_request() {
769
        assert_eq!(
770
            Request {
771
                line_terminators: vec![],
772
                space0: whitespace(),
773
                method: Method::new("GET"),
774
                space1: whitespace(),
775
                url: Template::new(
776
                    None,
777
                    vec![TemplateElement::String {
778
                        value: "http://example.com".to_string(),
779
                        source: "not_used".to_source(),
780
                    }],
781
                    SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
782
                ),
783
                line_terminator0: line_terminator(),
784
                headers: vec![KeyValue {
785
                    line_terminators: vec![],
786
                    space0: whitespace(),
787
                    key: Template::new(
788
                        None,
789
                        vec![TemplateElement::String {
790
                            value: "Foo".to_string(),
791
                            source: "unused".to_source(),
792
                        }],
793
                        SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0))
794
                    ),
795
                    space1: whitespace(),
796
                    space2: whitespace(),
797
                    value: Template::new(
798
                        None,
799
                        vec![TemplateElement::String {
800
                            value: "Bar".to_string(),
801
                            source: "unused".to_source(),
802
                        }],
803
                        SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0))
804
                    ),
805
                    line_terminator0: line_terminator(),
806
                }],
807
                sections: vec![],
808
                body: None,
809
                source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
810
            }
811
            .to_json(),
812
            JValue::Object(vec![
813
                ("method".to_string(), JValue::String("GET".to_string())),
814
                (
815
                    "url".to_string(),
816
                    JValue::String("http://example.com".to_string())
817
                ),
818
                (
819
                    "headers".to_string(),
820
                    JValue::List(vec![JValue::Object(vec![
821
                        ("name".to_string(), JValue::String("Foo".to_string())),
822
                        ("value".to_string(), JValue::String("Bar".to_string()))
823
                    ])])
824
                )
825
            ])
826
        );
827
    }
828

            
829
    #[test]
830
    pub fn test_response() {
831
        assert_eq!(
832
            Response {
833
                line_terminators: vec![],
834
                version: Version {
835
                    value: VersionValue::Version11,
836
                    source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
837
                },
838
                space0: whitespace(),
839
                status: Status {
840
                    value: StatusValue::Specific(200),
841
                    source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
842
                },
843
                space1: whitespace(),
844
                line_terminator0: line_terminator(),
845
                headers: vec![],
846
                sections: vec![],
847
                body: None,
848
                source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
849
            }
850
            .to_json(),
851
            JValue::Object(vec![
852
                (
853
                    "version".to_string(),
854
                    JValue::String("HTTP/1.1".to_string())
855
                ),
856
                ("status".to_string(), JValue::Number("200".to_string()))
857
            ])
858
        );
859
        assert_eq!(
860
            Response {
861
                line_terminators: vec![],
862
                version: Version {
863
                    value: VersionValue::VersionAny,
864
                    source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
865
                },
866
                space0: whitespace(),
867
                status: Status {
868
                    value: StatusValue::Any,
869
                    source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
870
                },
871
                space1: whitespace(),
872
                line_terminator0: line_terminator(),
873
                headers: vec![],
874
                sections: vec![],
875
                body: None,
876
                source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
877
            }
878
            .to_json(),
879
            JValue::Object(vec![])
880
        );
881
    }
882

            
883
    fn header_query() -> Query {
884
        Query {
885
            source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
886
            value: QueryValue::Header {
887
                space0: whitespace(),
888
                name: Template::new(
889
                    None,
890
                    vec![TemplateElement::String {
891
                        value: "Content-Length".to_string(),
892
                        source: "Content-Length".to_source(),
893
                    }],
894
                    SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
895
                ),
896
            },
897
        }
898
    }
899

            
900
    fn header_capture() -> Capture {
901
        Capture {
902
            line_terminators: vec![],
903
            space0: whitespace(),
904
            name: Template::new(
905
                None,
906
                vec![TemplateElement::String {
907
                    value: "size".to_string(),
908
                    source: "unused".to_source(),
909
                }],
910
                SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
911
            ),
912
            space1: whitespace(),
913
            space2: whitespace(),
914
            query: header_query(),
915
            filters: vec![],
916
            space3: whitespace(),
917
            redacted: false,
918
            line_terminator0: line_terminator(),
919
        }
920
    }
921

            
922
    fn header_assert() -> Assert {
923
        Assert {
924
            line_terminators: vec![],
925
            space0: whitespace(),
926
            query: header_query(),
927
            filters: vec![],
928
            space1: whitespace(),
929
            predicate: equal_int_predicate(10),
930
            line_terminator0: line_terminator(),
931
        }
932
    }
933

            
934
    fn equal_int_predicate(value: i64) -> Predicate {
935
        Predicate {
936
            not: false,
937
            space0: whitespace(),
938
            predicate_func: PredicateFunc {
939
                source_info: SourceInfo::new(Pos::new(0, 0), Pos::new(0, 0)),
940
                value: PredicateFuncValue::Equal {
941
                    space0: whitespace(),
942
                    value: PredicateValue::Number(Number::Integer(I64::new(
943
                        value,
944
                        value.to_string().to_source(),
945
                    ))),
946
                },
947
            },
948
        }
949
    }
950

            
951
    #[test]
952
    pub fn test_query() {
953
        assert_eq!(
954
            header_query().to_json(),
955
            JValue::Object(vec![
956
                ("type".to_string(), JValue::String("header".to_string())),
957
                (
958
                    "name".to_string(),
959
                    JValue::String("Content-Length".to_string())
960
                ),
961
            ])
962
        );
963
    }
964

            
965
    #[test]
966
    pub fn test_capture() {
967
        assert_eq!(
968
            header_capture().to_json(),
969
            JValue::Object(vec![
970
                ("name".to_string(), JValue::String("size".to_string())),
971
                (
972
                    "query".to_string(),
973
                    JValue::Object(vec![
974
                        ("type".to_string(), JValue::String("header".to_string())),
975
                        (
976
                            "name".to_string(),
977
                            JValue::String("Content-Length".to_string())
978
                        ),
979
                    ])
980
                ),
981
            ])
982
        );
983
    }
984

            
985
    #[test]
986
    pub fn test_predicate() {
987
        assert_eq!(
988
            equal_int_predicate(10).to_json(),
989
            JValue::Object(vec![
990
                ("type".to_string(), JValue::String("==".to_string())),
991
                ("value".to_string(), JValue::Number("10".to_string()))
992
            ]),
993
        );
994
    }
995

            
996
    #[test]
997
    pub fn test_assert() {
998
        assert_eq!(
999
            header_assert().to_json(),
            JValue::Object(vec![
                (
                    "query".to_string(),
                    JValue::Object(vec![
                        ("type".to_string(), JValue::String("header".to_string())),
                        (
                            "name".to_string(),
                            JValue::String("Content-Length".to_string())
                        ),
                    ])
                ),
                (
                    "predicate".to_string(),
                    JValue::Object(vec![
                        ("type".to_string(), JValue::String("==".to_string())),
                        ("value".to_string(), JValue::Number("10".to_string()))
                    ])
                )
            ]),
        );
    }
}