RInside:parseEvalQ'Parse Error'导致对parseEvalQ的每次后续调用都给出'Parse Error',即使处理了异常

时间:2011-04-14 19:29:53

标签: r rcpp rinside

我的代码尝试通过C ++模拟R shell,允许用户通过tcp连接发送R命令,然后在运行时通过RInside :: parseEvalQ函数传递给R实例。我必须能够处理格式错误的命令。每当一个错误的命令作为parseEvalQ的参数给出时,我会捕获抛出的运行时错误(查看RInside.cpp,我的特定错误在parseEval(const string&,SEXP)函数中标记为'PARSE_ERROR''status'),( )给出了“St9exception”异常。

我有两个问题,第一个比第二个更紧迫:

1a。1a。在初始解析错误之后,即使参数有效,对parseEvalQ的任何后续调用都会导致另一个解析错误。嵌入式R实例是否因解析错误而以某种方式损坏?

1b。 RInside文档建议使用Rcpp :: Evaluator :: run来处理C ++中的R异常(我怀疑在调用parseEval(const string&,SEXP)期间,在它返回错误状态之前,它被抛出在R实例中的某个地方)解析错误')。我已经尝试过尝试使用它,但是在网上找不到如何实际使用Rcpp :: Evaluator :: run的例子。

2。在我的程序中,我将stdout和stderr(在C ++级别)重新路由到我的tcp连接的文件描述符,来自RInside实例的任何错误消息都被发送到控制台,但是常规输出不会。我发送了RInside命令'sink(stderr(),type =“output”)',以便将stdout重新路由到stderr(因为stderr似乎出现在我的控制台中)但是仍然没有显示常规输出。 'print(command)'工作,但我想要一个更简洁的方式将stdout直接传递到控制台,就像普通的R shell一样。

非常感谢任何帮助和/或想法。我的代码的提炼版本如下所示:

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

using namespace std;

string request_cpp;

ostringstream oss;

int read(FILE* tcp_fd)  
{
  /* function to read input from FILE* into the 'request_cpp' string */
}  

int write(FILE* tcp_fd, const string& response)  
{
  /* function to write a string to FILE* */
}

int main(int argc, char* argv[])  
{
  // create RInside object
  RInside R(argc,argv);

  //socket  
  int sd = socket(PF_INET, SOCK_STREAM, 0);  
  addr.sin_family = AF_INET;  
  addr.sin_port = htons(40650);

  // set and accept connection on socket  
  inet_pton(AF_INET, "127.0.0.1", &addr.sin_addr);  
  bind(sd,(struct sockaddr*)&addr, sizeof(addr));  
  listen(sd,1);  
  int sd_i = accept(sd, 0, 0);  

  //re-route stdout and stderr to socket  
  close(1);  
  dup(sd_i);  
  close(2);  
  dup(sd_i);  

  // open read/write file descriptor to socket

  FILE* fp = fdopen(sd_i,"r+");  

  // emulate R prompt  

  write(fp,"> ");  

  // (attempt to) redirect R's stdout to stderr
  R.parseEvalQ("sink(stderr(),type=\"output\");");

  // read from socket and pass commands to RInside  
  while( read(fp) )  
  {
    try  
    {
      // skip empty input  
      if(request_cpp == "")  
      {  
        write(fp, "> ");  
        continue;  
      }
      else if(request_cpp == "q()")  
      {  
        break;
      }  
      else  
      {  
        // clear string stream  
        oss.str("");  

        // wrap command in try  
        oss << "try(" << request_cpp << ");" << endl;  

        // send command  
        R.parseEvalQ(oss.str());  
      }  
    }  
    catch(exception e)  
    {  
      // print exception to console  
      write(fp, e.what());  
    }    
 write(fp, "> ");
 }

fclose(fp);  
close(sd_i);  
exit(0);  
}

1 个答案:

答案 0 :(得分:1)

我几周前因为你没有使用'r'标签而错过了。

好像你正在重新实施Simon的受信任rserver。为什么不直接使用它?

否则,对于Rcpp问题,请考虑询问我们的rcpp-devel列表。