シェルスクリプト、メイクファイル、または別のプログラムを実行して(リアルタイムで)出力するc ++プログラムを書いています。ただし、エラーが発生した場合、またはエラーが発生しなかった場合に、プログラムの戻り方を変えたいと思います。
#include "execxi.h"
using namespace std;
int execXI::run(string command)
{
FILE *in;
char buff[512];
// is this the check for command execution exited with not 0?
if(!(in = popen(command.c_str(), "r"))){
// I want to return the exit code and error message too if any
return 1;
}
// this part echoes the output of the command that's executed
while(fgets(buff, sizeof(buff), in)!=NULL){
cout << buff;
}
pclose(in);
return 0;
}
これまでのところです。
このスクリプトがmake
を実行してプログラムをビルドし、そのようなエラーが発生したとしましょう
on_target_webkit_version out/Release/obj/gen/webkit_version.h
Traceback (most recent call last):
File "../build/webkit_version.py", line 107, in <module>
sys.exit(main())
File "../build/webkit_version.py", line 103, in main
return EmitVersionHeader(*sys.argv[1:])
File "../build/webkit_version.py", line 86, in EmitVersionHeader
webkit_revision = GetWebKitRevision(webkit_dir, version_file)
File "../build/webkit_version.py", line 60, in GetWebKitRevision
version_info = lastchange.FetchVersionInfo(
AttributeError: 'module' object has no attribute 'FetchVersionInfo'
make: *** [out/Release/obj/gen/webkit_version.h] Error 1
これがエラーで終了したことを私が知ることは可能ですか?
エラーのため、コードelse than 0
で終了しますか?
最後の部分はstderr
に出力されますか?
make
が0
ではなくコードで終了したことを考慮して、1
とすると、stderr
に出力されるため、これらの終了コードとエラーメッセージを取得できません。最終的には?
プログラムの結果を出力した後、どのようにして終了コードとstderr
を取得し、exit code
/stderr
を関数に返すことができますか?
エラーコードに興味がある場合、これは256で除算するのではなく、よりポータブルな方法で取得できます。
printf("Exit code: %i\n", WEXITSTATUS(pclose(fp)));
ただし、popen
は1つの方法であるため、シェルの通常のリダイレクトスタイルでさらに回避策を作成するか、次のテストされていないコードに従って正しく実行します。
#include <unistd.h>
#include <stdio.h>
/* since pipes are unidirectional, we need two pipes.
one for data to flow from parent's stdout to child's
stdin and the other for child's stdout to flow to
parent's stdin */
#define NUM_PIPES 2
#define PARENT_WRITE_PIPE 0
#define PARENT_READ_PIPE 1
int pipes[NUM_PIPES][2];
/* always in a pipe[], pipe[0] is for read and
pipe[1] is for write */
#define READ_FD 0
#define WRITE_FD 1
#define PARENT_READ_FD ( pipes[PARENT_READ_PIPE][READ_FD] )
#define PARENT_WRITE_FD ( pipes[PARENT_WRITE_PIPE][WRITE_FD] )
#define CHILD_READ_FD ( pipes[PARENT_WRITE_PIPE][READ_FD] )
#define CHILD_WRITE_FD ( pipes[PARENT_READ_PIPE][WRITE_FD] )
void
main()
{
int outfd[2];
int infd[2];
// pipes for parent to write and read
pipe(pipes[PARENT_READ_PIPE]);
pipe(pipes[PARENT_WRITE_PIPE]);
if(!fork()) {
char *argv[]={ "/usr/bin/bc", "-q", 0};
dup2(CHILD_READ_FD, STDIN_FILENO);
dup2(CHILD_WRITE_FD, STDOUT_FILENO);
/* Close fds not required by child. Also, we don't
want the exec'ed program to know these existed */
close(CHILD_READ_FD);
close(CHILD_WRITE_FD);
close(PARENT_READ_FD);
close(PARENT_WRITE_FD);
execv(argv[0], argv);
} else {
char buffer[100];
int count;
/* close fds not required by parent */
close(CHILD_READ_FD);
close(CHILD_WRITE_FD);
// Write to child’s stdin
write(PARENT_WRITE_FD, "2^32\n", 5);
// Read from child’s stdout
count = read(PARENT_READ_FD, buffer, sizeof(buffer)-1);
if (count >= 0) {
buffer[count] = 0;
printf("%s", buffer);
} else {
printf("IO Error\n");
}
}
}
コードはここからです:
http://jineshkj.wordpress.com/2006/12/22/how-to-capture-stdin-stdout-and-stderr-of-child-program/
子プロセスの戻り値が一番上にあります
168ビット。 pcloseの戻り値を256で除算する必要があります。その後、子プロセスの検索された戻り値を取得します。
http://bytes.com/topic/c/answers/131694-pclose-returning-termination-status-command から取得
私の答えはpclose(in)/256
は終了コードです。
Stderrまたはsdtoutを別の方法でキャプチャする方法はまだわかりませんが、その答えが見つかるまで、これを私の答えとして受け入れます。
終了コードLoganについて返信いただきありがとうございます。
Stderrを取得するためのラウンドトリップは、一時ファイルにリダイレクトすることだと思います。
FILE* f = popen("cmd 2>/tmp/tmpfile.txt", "r");