1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
// Copyright 2020 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

// This implementation will only reply back after every 5 requests. It tests that test cases can be
// executed in parallel. IF tests are not executed in parallel, the tests will hang as this
// implementation will never reply back

use {
    fidl_fidl_examples_routing_echo as fecho, fuchsia_async as fasync,
    fuchsia_component::server::ServiceFs,
    futures::{StreamExt, TryStreamExt},
    std::sync::{Arc, Mutex},
};

#[fasync::run_singlethreaded]
async fn main() {
    let mut fs = ServiceFs::new_local();
    let responders = Arc::new(Mutex::new(vec![]));
    fs.dir("svc").add_fidl_service(move |stream| {
        let responders = responders.clone();
        fasync::Task::spawn(async move {
            run_echo_service(stream, responders).await;
        })
        .detach();
    });
    fs.take_and_serve_directory_handle().expect("failed to serve outgoing directory");
    fs.collect::<()>().await;
}

async fn run_echo_service(
    mut stream: fecho::EchoRequestStream,
    responders: Arc<Mutex<Vec<(fecho::EchoEchoStringResponder, Option<String>)>>>,
) {
    while let Some(event) = stream.try_next().await.expect("failed to serve echo service") {
        let fecho::EchoRequest::EchoString { value, responder } = event;
        let mut responders = responders.lock().unwrap();
        responders.push((responder, value));
        if responders.len() == 5 {
            while let Some((r, v)) = responders.pop() {
                r.send(v.as_ref().map(|s| &**s)).expect("failed to send echo response");
            }
        }
    }
}