mirror of
https://github.com/dlang/phobos.git
synced 2025-04-29 22:50:38 +03:00

* std.contracts: Added file and line information to enforce. Added errnoEnforce that reads and formats a message according to errno. Added corresponding ErrnoException class. * std.encoding: For now commented out std.encoding.to. * std.file: Fixed bug 2065 * std.format: Fixed bug in raw write for arrays * std.getopt: Added new option stopOnFirstNonOption. Also automatically expand dubious option groups with embedded in them (useful for shebang scripts) * std.math: improved integral powers * std.md5: Improved signature of sum so it takes multiple arrays. Added getDigestString. * std.path: changed signatures of test functions from bool to int. Implemented rel2abs for Windows. Improved join so that it accepts multiple paths. Got rid of some gotos with the help of scope statements. * std.process: added getenv and setenv. Improved system() so it returns the exit code correctly on Linux. * std.random: added the dice function - a handy (possibly biased) dice. * std.file: added support for opening large files (not yet tested) * std.utf: added the codeLength function. Got rid of some gotos.
423 lines
11 KiB
D
423 lines
11 KiB
D
// Written in the D programming language
|
|
|
|
/*
|
|
* Copyright (C) 2003-2004 by Digital Mars, www.digitalmars.com
|
|
* Written by Matthew Wilson and Walter Bright
|
|
*
|
|
* Incorporating idea (for execvpe() on Linux) from Russ Lewis
|
|
*
|
|
* Updated: 21st August 2004
|
|
*
|
|
* This software is provided 'as-is', without any express or implied
|
|
* warranty. In no event will the authors be held liable for any damages
|
|
* arising from the use of this software.
|
|
*
|
|
* Permission is granted to anyone to use this software for any purpose,
|
|
* including commercial applications, and to alter it and redistribute it
|
|
* freely, subject to the following restrictions:
|
|
*
|
|
* o The origin of this software must not be misrepresented; you must not
|
|
* claim that you wrote the original software. If you use this software
|
|
* in a product, an acknowledgment in the product documentation would be
|
|
* appreciated but is not required.
|
|
* o Altered source versions must be plainly marked as such, and must not
|
|
* be misrepresented as being the original software.
|
|
* o This notice may not be removed or altered from any source
|
|
* distribution.
|
|
*/
|
|
|
|
/**
|
|
Authors:
|
|
|
|
$(WEB digitalmars.com, Walter Bright), $(WEB erdani.org, Andrei
|
|
Alexandrescu)
|
|
|
|
Macros:
|
|
|
|
WIKI=Phobos/StdProcess
|
|
*/
|
|
|
|
module std.process;
|
|
|
|
private import std.c.stdlib;
|
|
private import std.c.string;
|
|
private import std.string;
|
|
private import std.c.process;
|
|
private import std.contracts;
|
|
version (Windows)
|
|
{
|
|
private import std.stdio : readln, fclose;
|
|
private import std.c.windows.windows:GetCurrentProcessId;
|
|
}
|
|
version (linux)
|
|
{
|
|
private import std.stdio : popen, readln, fclose;
|
|
}
|
|
|
|
/**
|
|
Execute $(D command) in a _command shell.
|
|
|
|
Returns: If $(D command) is null, returns nonzero if the _command
|
|
interpreter is found, and zero otherwise. If $(D command) is not
|
|
null, returns -1 on error, or the exit status of command (which may
|
|
in turn signal an error in command's execution).
|
|
|
|
Note: On Unix systems, the homonym C function (which is accessible
|
|
to D programs as $(LINK2 std_c_process.html, std.c._system))
|
|
returns a code in the same format as
|
|
$(WEB www.scit.wlv.ac.uk/cgi-bin/mansec?2+waitpid, waitpid),
|
|
meaning that C programs must use the $(D WEXITSTATUS) macro to
|
|
extract the actual exit code from the $(D system) call. D's $(D
|
|
system) automatically extracts the exit status.
|
|
|
|
*/
|
|
|
|
int system(string command)
|
|
{
|
|
if (!command) return std.c.process.system(null);
|
|
const commandz = toStringz(command);
|
|
invariant status = std.c.process.system(commandz);
|
|
if (status == -1) return status;
|
|
version (linux)
|
|
return (status & 0x0000ff00) >>> 8;
|
|
else
|
|
return status;
|
|
}
|
|
|
|
private void toAStringz(in string[] a, const(char)**az)
|
|
{
|
|
foreach(string s; a)
|
|
{
|
|
*az++ = toStringz(s);
|
|
}
|
|
*az = null;
|
|
}
|
|
|
|
|
|
/* ========================================================== */
|
|
|
|
//version (Windows)
|
|
//{
|
|
// int spawnvp(int mode, string pathname, string[] argv)
|
|
// {
|
|
// char** argv_ = cast(char**)alloca((char*).sizeof * (1 + argv.length));
|
|
//
|
|
// toAStringz(argv, argv_);
|
|
//
|
|
// return std.c.process.spawnvp(mode, toStringz(pathname), argv_);
|
|
// }
|
|
//}
|
|
|
|
// Incorporating idea (for spawnvp() on linux) from Dave Fladebo
|
|
|
|
alias std.c.process._P_WAIT P_WAIT;
|
|
alias std.c.process._P_NOWAIT P_NOWAIT;
|
|
|
|
int spawnvp(int mode, string pathname, string[] argv)
|
|
{
|
|
auto argv_ = cast(const(char)**)alloca((char*).sizeof * (1 + argv.length));
|
|
|
|
toAStringz(argv, argv_);
|
|
|
|
version(linux)
|
|
{
|
|
return _spawnvp(mode, toStringz(pathname), argv_);
|
|
}
|
|
else
|
|
{
|
|
return std.c.process.spawnvp(mode, toStringz(pathname), argv_);
|
|
}
|
|
}
|
|
|
|
version(linux)
|
|
{
|
|
private import std.c.linux.linux;
|
|
int _spawnvp(int mode, in char *pathname, in char **argv)
|
|
{
|
|
int retval = 0;
|
|
pid_t pid = fork();
|
|
|
|
if(!pid)
|
|
{ // child
|
|
std.c.process.execvp(pathname, argv);
|
|
goto Lerror;
|
|
}
|
|
else if(pid > 0)
|
|
{ // parent
|
|
if(mode == _P_NOWAIT)
|
|
{
|
|
retval = pid; // caller waits
|
|
}
|
|
else
|
|
{
|
|
while(1)
|
|
{
|
|
int status;
|
|
pid_t wpid = waitpid(pid, &status, 0);
|
|
if(exited(status))
|
|
{
|
|
retval = exitstatus(status);
|
|
break;
|
|
}
|
|
else if(signaled(status))
|
|
{
|
|
retval = -termsig(status);
|
|
break;
|
|
}
|
|
else if(stopped(status)) // ptrace support
|
|
continue;
|
|
else
|
|
goto Lerror;
|
|
}
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
Lerror:
|
|
retval = getErrno;
|
|
char[80] buf = void;
|
|
throw new Exception(
|
|
"Cannot spawn " ~ toString(pathname) ~ "; "
|
|
~ toString(strerror_r(retval, buf.ptr, buf.length))
|
|
~ " [errno " ~ toString(retval) ~ "]");
|
|
} // _spawnvp
|
|
private
|
|
{
|
|
bool stopped(int status) { return cast(bool)((status & 0xff) == 0x7f); }
|
|
bool signaled(int status) { return cast(bool)((cast(char)((status & 0x7f) + 1) >> 1) > 0); }
|
|
int termsig(int status) { return status & 0x7f; }
|
|
bool exited(int status) { return cast(bool)((status & 0x7f) == 0); }
|
|
int exitstatus(int status) { return (status & 0xff00) >> 8; }
|
|
} // private
|
|
} // version(linux)
|
|
|
|
/* ========================================================== */
|
|
|
|
/**
|
|
* Execute program specified by pathname, passing it the arguments (argv)
|
|
* and the environment (envp), returning the exit status.
|
|
* The 'p' versions of exec search the PATH environment variable
|
|
* setting for the program.
|
|
*/
|
|
|
|
int execv(in string pathname, in string[] argv)
|
|
{
|
|
auto argv_ = cast(const(char)**)alloca((char*).sizeof * (1 + argv.length));
|
|
|
|
toAStringz(argv, argv_);
|
|
|
|
return std.c.process.execv(toStringz(pathname), argv_);
|
|
}
|
|
|
|
/** ditto */
|
|
int execve(in string pathname, in string[] argv, in string[] envp)
|
|
{
|
|
auto argv_ = cast(const(char)**)alloca((char*).sizeof * (1 + argv.length));
|
|
auto envp_ = cast(const(char)**)alloca((char*).sizeof * (1 + envp.length));
|
|
|
|
toAStringz(argv, argv_);
|
|
toAStringz(envp, envp_);
|
|
|
|
return std.c.process.execve(toStringz(pathname), argv_, envp_);
|
|
}
|
|
|
|
/** ditto */
|
|
int execvp(in string pathname, in string[] argv)
|
|
{
|
|
auto argv_ = cast(const(char)**)alloca((char*).sizeof * (1 + argv.length));
|
|
|
|
toAStringz(argv, argv_);
|
|
|
|
return std.c.process.execvp(toStringz(pathname), argv_);
|
|
}
|
|
|
|
/** ditto */
|
|
int execvpe(in string pathname, in string[] argv, in string[] envp)
|
|
{
|
|
version(linux)
|
|
{
|
|
// Is pathname rooted?
|
|
if(pathname[0] == '/')
|
|
{
|
|
// Yes, so just call execve()
|
|
return execve(pathname, argv, envp);
|
|
}
|
|
else
|
|
{
|
|
// No, so must traverse PATHs, looking for first match
|
|
string[] envPaths = std.string.split(std.string.toString(std.c.stdlib.getenv("PATH")), ":");
|
|
int iRet = 0;
|
|
|
|
// Note: if any call to execve() succeeds, this process will cease
|
|
// execution, so there's no need to check the execve() result through
|
|
// the loop.
|
|
|
|
foreach(string pathDir; envPaths)
|
|
{
|
|
string composite = cast(string) (pathDir ~ "/" ~ pathname);
|
|
|
|
iRet = execve(composite, argv, envp);
|
|
}
|
|
if(0 != iRet)
|
|
{
|
|
iRet = execve(pathname, argv, envp);
|
|
}
|
|
|
|
return iRet;
|
|
}
|
|
}
|
|
else version(Windows)
|
|
{
|
|
char** argv_ = cast(char**)alloca((char*).sizeof * (1 + argv.length));
|
|
char** envp_ = cast(char**)alloca((char*).sizeof * (1 + envp.length));
|
|
|
|
toAStringz(argv, argv_);
|
|
toAStringz(envp, envp_);
|
|
|
|
return std.c.process.execvpe(toStringz(pathname), argv_, envp_);
|
|
}
|
|
else
|
|
{
|
|
static assert(0);
|
|
} // version
|
|
}
|
|
|
|
version(linux)
|
|
{
|
|
alias std.c.process.getpid getpid;
|
|
}
|
|
else version (Windows)
|
|
{
|
|
alias std.c.windows.windows.GetCurrentProcessId getpid;
|
|
}
|
|
|
|
/**
|
|
Runs $(D_PARAM cmd) in a shell and returns its standard output. If
|
|
the process could not be started or exits with an error code,
|
|
throws an exception.
|
|
|
|
Example:
|
|
|
|
----
|
|
auto tempFilename = chomp(shell("mcookie"));
|
|
auto f = enforce(fopen(tempFilename), "w");
|
|
scope(exit)
|
|
{
|
|
fclose(f) == 0 || assert(false);
|
|
system("rm " ~ tempFilename);
|
|
}
|
|
... use f ...
|
|
----
|
|
*/
|
|
string shell(string cmd)
|
|
{
|
|
version (linux)
|
|
{
|
|
auto f = enforce(popen(cmd, "r"), "Could not execute: "~cmd);
|
|
scope(failure) f is null || fclose(f);
|
|
char[] line;
|
|
string result;
|
|
while (readln(f, line))
|
|
{
|
|
result ~= line;
|
|
}
|
|
auto error = fclose(f) != 0;
|
|
f = null;
|
|
enforce(!error, "Process \""~cmd~"\" finished in error.");
|
|
return result;
|
|
}
|
|
else
|
|
{
|
|
enforce(false, "shell() function not yet implemented on Windows");
|
|
return null;
|
|
}
|
|
}
|
|
|
|
unittest
|
|
{
|
|
auto x = shell("echo wyda");
|
|
assert(x == "wyda\n");
|
|
}
|
|
|
|
/**
|
|
Gets the value of environment variable $(D name) as a string. Calls
|
|
$(LINK2 std_c_stdlib.html#_getenv, std.c.stdlib._getenv)
|
|
internally. */
|
|
|
|
string getenv(in char[] name)
|
|
{
|
|
auto p = std.c.stdlib.getenv(toStringz(name));
|
|
if (!p) return null;
|
|
return p[0 .. strlen(p)].idup;
|
|
}
|
|
|
|
/**
|
|
Sets the value of environment variable $(D name) to $(D value). If the
|
|
value was written, or the variable was already present and $(D
|
|
overwrite) is false, returns normally. Otherwise, it throws an
|
|
exception. Calls $(LINK2 std_c_stdlib.html#_setenv,
|
|
std.c.stdlib._setenv) internally. */
|
|
|
|
void setenv(in char[] name, in char[] value, bool overwrite)
|
|
{
|
|
errnoEnforce(
|
|
std.c.stdlib.setenv(toStringz(name), toStringz(value), overwrite) == 0);
|
|
}
|
|
|
|
/**
|
|
Removes variable $(D name) from the environment. Calls $(LINK2
|
|
std_c_stdlib.html#_unsetenv, std.c.stdlib._unsetenv) internally. */
|
|
|
|
void unsetenv(in char[] name)
|
|
{
|
|
errnoEnforce(std.c.stdlib.unsetenv(toStringz(name)) == 0);
|
|
}
|
|
|
|
unittest
|
|
{
|
|
setenv("wyda", "geeba", true);
|
|
assert(getenv("wyda") == "geeba");
|
|
unsetenv("wyda");
|
|
assert(getenv("wyda") is null);
|
|
}
|
|
|
|
/* ////////////////////////////////////////////////////////////////////////// */
|
|
|
|
version(MainTest)
|
|
{
|
|
int main(string[] args)
|
|
{
|
|
if(args.length < 2)
|
|
{
|
|
printf("Must supply executable (and optional arguments)\n");
|
|
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
string[] dummy_env;
|
|
|
|
dummy_env ~= "VAL0=value";
|
|
dummy_env ~= "VAL1=value";
|
|
|
|
/+
|
|
foreach(string arg; args)
|
|
{
|
|
printf("%.*s\n", arg);
|
|
}
|
|
+/
|
|
|
|
// int i = execv(args[1], args[1 .. args.length]);
|
|
// int i = execvp(args[1], args[1 .. args.length]);
|
|
int i = execvpe(args[1], args[1 .. args.length], dummy_env);
|
|
|
|
printf("exec??() has returned! Error code: %d; errno: %d\n", i, /* std.c.stdlib.getErrno() */-1);
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* ////////////////////////////////////////////////////////////////////////// */
|
|
|