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
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
// Copyright 2021 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.

use {
    fidl_fuchsia_component_runner as fcrunner, fuchsia_async as fasync,
    fuchsia_component::server::ServiceFs,
    futures::prelude::*,
    test_runners_lib::elf,
    test_runners_lib::{elf::SuiteServer, errors::*},
    thiserror::Error,
    tracing::{info, warn},
};

pub fn add_runner_service<F, U, S>(
    get_test_server: F,
    validate_args: U,
) -> Result<(), anyhow::Error>
where
    F: 'static + Fn() -> S + Send + Copy,
    U: 'static + Fn(&Vec<String>) -> Result<(), ArgumentError> + Copy,
    S: SuiteServer,
{
    info!("started");
    let mut executor = fasync::LocalExecutor::new();

    let mut fs = ServiceFs::new_local();
    fs.dir("svc").add_fidl_service(move |stream| {
        fasync::Task::local(async move {
            start_runner(stream, get_test_server, validate_args)
                .await
                .expect("failed to start runner.")
        })
        .detach();
    });
    fs.take_and_serve_directory_handle()?;
    executor.run_singlethreaded(fs.collect::<()>());
    Ok(())
}

/// Error encountered by runner.
#[derive(Debug, Error)]
pub enum RunnerError {
    #[error("Cannot read request: {:?}", _0)]
    RequestRead(fidl::Error),
}

async fn start_runner<F, U, S>(
    mut stream: fcrunner::ComponentRunnerRequestStream,
    get_test_server: F,
    validate_args: U,
) -> Result<(), RunnerError>
where
    F: 'static + Fn() -> S + Send + Copy,
    U: 'static + Fn(&Vec<String>) -> Result<(), ArgumentError> + Copy,
    S: SuiteServer,
{
    while let Some(event) = stream.try_next().await.map_err(RunnerError::RequestRead)? {
        match event {
            fcrunner::ComponentRunnerRequest::Start { start_info, controller, .. } => {
                let url = start_info.resolved_url.clone().unwrap_or("".to_owned());
                if let Err(e) =
                    elf::start_component(start_info, controller, get_test_server, validate_args)
                        .await
                {
                    warn!("Cannot start component '{}': {:?}", url, e)
                };
            }
        }
    }
    Ok(())
}