Mercurial > hg
view rust/chg/src/uihandler.rs @ 41692:ee7b7bd432a1
rust: translated random test of missingancestors
This is a Rust implementation of the random
DAG generator and related incrementalmissingancestors
tests against a naive brute force implementation.
It is provided as an integration test, so that it
won't run by default if any unit test fails.
In case of a failed example, all needed information
for reproduction is included in the panic message,
(this is how
`test_remove_ancestors_from_case1()` has been generated),
as well as the random seed.
The whole test is rerunnable by passing the random seed
in the TEST_RANDOM_SEED environment variable.
The other parameters (numbers of iterations) can be passed
in the TEST_MISSING_ANCESTORS environment variable.
An alternative would have been to expose to Python
MissingAncestors<VecGraphs> but that would have meant
pollution of the release build used from Python,
whereas we do it in this changeset within the tests submodule
Differential Revision: https://phab.mercurial-scm.org/D5417
author | Georges Racinet <gracinet@anybox.fr> |
---|---|
date | Sun, 02 Dec 2018 16:19:22 +0100 |
parents | e70b616a077b |
children | ce088b38f92b |
line wrap: on
line source
// Copyright 2018 Yuya Nishihara <yuya@tcha.org> // // This software may be used and distributed according to the terms of the // GNU General Public License version 2 or any later version. use futures::Future; use futures::future::IntoFuture; use std::io; use std::os::unix::io::AsRawFd; use std::os::unix::process::ExitStatusExt; use std::process::{Command, Stdio}; use tokio; use tokio_process::{ChildStdin, CommandExt}; use super::message::CommandSpec; use super::procutil; /// Callback to process shell command requests received from server. pub trait SystemHandler: Sized { type PagerStdin: AsRawFd; type SpawnPagerResult: IntoFuture<Item = (Self, Self::PagerStdin), Error = io::Error>; type RunSystemResult: IntoFuture<Item = (Self, i32), Error = io::Error>; /// Handles pager command request. /// /// Returns the pipe to be attached to the server if the pager is spawned. fn spawn_pager(self, spec: CommandSpec) -> Self::SpawnPagerResult; /// Handles system command request. /// /// Returns command exit code (positive) or signal number (negative). fn run_system(self, spec: CommandSpec) -> Self::RunSystemResult; } /// Default cHg implementation to process requests received from server. pub struct ChgUiHandler { } impl ChgUiHandler { pub fn new() -> ChgUiHandler { ChgUiHandler {} } } impl SystemHandler for ChgUiHandler { type PagerStdin = ChildStdin; type SpawnPagerResult = io::Result<(Self, Self::PagerStdin)>; type RunSystemResult = Box<dyn Future<Item = (Self, i32), Error = io::Error> + Send>; fn spawn_pager(self, spec: CommandSpec) -> Self::SpawnPagerResult { let mut pager = new_shell_command(&spec) .stdin(Stdio::piped()) .spawn_async()?; let pin = pager.stdin().take().unwrap(); procutil::set_blocking_fd(pin.as_raw_fd())?; // TODO: if pager exits, notify the server with SIGPIPE immediately. // otherwise the server won't get SIGPIPE if it does not write // anything. (issue5278) // kill(peerpid, SIGPIPE); tokio::spawn(pager.map(|_| ()).map_err(|_| ())); // just ignore errors Ok((self, pin)) } fn run_system(self, spec: CommandSpec) -> Self::RunSystemResult { let fut = new_shell_command(&spec) .spawn_async() .into_future() .flatten() .map(|status| { let code = status.code().or_else(|| status.signal().map(|n| -n)) .expect("either exit code or signal should be set"); (self, code) }); Box::new(fut) } } fn new_shell_command(spec: &CommandSpec) -> Command { let mut builder = Command::new("/bin/sh"); builder .arg("-c") .arg(&spec.command) .current_dir(&spec.current_dir) .env_clear() .envs(spec.envs.iter().cloned()); builder }