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
  
73
  
74
  
75
  
76
  
77
  
78
  
79
  
80
  
81
  
82
  
83
  
84
  
85
  
86
  
87
  
88
  
89
  
90
  
91
  
92
  
93
  
94
  
95
  
96
  
97
  
98
  
99
  
100
  
101
  
102
  
103
  
104
  
105
  
106
  
107
  
108
  
109
  
110
  
111
  
112
  
113
  
114
  
115
  
116
  
117
  
118
  
119
  
120
  
121
  
void perror(string format,mixed ... args); 
 
string popen(string s, void|mapping env, int|void uid, int|void gid) 
{ 
  object p,p2; 
 
  p2 = File(); 
  p=p2->pipe(); 
  if(!p) error("Popen failed. (couldn't create pipe)\n"); 
 
  if(!fork()) 
  { 
    array (int) olduid = ({ -1, -1 }); 
    catch { 
      if(p->query_fd() < 0) 
      { 
        perror("File to dup2 to closed!\n"); 
        exit(99); 
      } 
      p->dup2(File("stdout")); 
      if(uid || gid) 
      { 
        object privs = ((program)"privs")("Executing script as non-www user"); 
        olduid = ({ uid, gid }); 
        setgid(olduid[1]); 
        setuid(olduid[0]); 
#if efun(initgroups) 
      array pw = getpwuid((int)uid); 
        if(pw) initgroups(pw[0], (int)olduid[0]); 
#endif 
      } 
      catch(exece("/bin/sh", ({ "-c", s }), (env||environment))); 
    }; 
    exit(69); 
  }else{ 
    string t; 
    destruct(p); 
    t=p2->read(0x7fffffff); 
    destruct(p2); 
    return t; 
  } 
} 
 
 
mapping make_mapping(string *f) 
{ 
  mapping foo=([ ]); 
  string s, a, b; 
  foreach(f, s) 
  { 
    sscanf(s, "%s=%s", a, b); 
    foo[a]=b; 
  } 
  return foo; 
} 
 
int low_spawne(string s,string *args, mapping|array env, object stdin,  
           object stdout, object stderr, void|string wd) 
{ 
  object p; 
  int pid; 
  string t; 
 
  if(arrayp(env)) 
    env = make_mapping(env); 
  if(!mappingp(env))  
    env=([]); 
   
   
  stdin->dup2(File("stdin")); 
  stdout->dup2(File("stdout")); 
  stderr->dup2(File("stderr")); 
  if(stringp(wd) && sizeof(wd)) 
    cd(wd); 
  exece(s, args, env); 
  perror(sprintf("Spawne: Failed to exece %s\n", s)); 
  exit(0); 
} 
 
int spawne(string s,string *args, mapping|array env, object stdin,  
           object stdout, object stderr, void|string wd, void|array (int) uid) 
{ 
  int pid, *olduid = allocate(2, "int"); 
  object privs; 
 
  if(pid=fork()) return pid; 
 
  if(arrayp(uid) && sizeof(uid) == 2) 
  { 
    privs = ((program)"privs")("Executing program as non-www user (outside roxen)"); 
    setgid(uid[1]); 
    setuid(uid[0]); 
  }  
  catch(low_spawne(s, args, env, stdin, stdout, stderr, wd)); 
  exit(0);  
} 
 
private static int perror_last_was_newline=1; 
 
void perror(string format,mixed ... args) 
{ 
   string s; 
   int lwn; 
   s=((args==({}))?format:sprintf(format,@args)); 
   if (s=="") return; 
   if ( (lwn = s[-1]=="\n") ) 
      s=s[0..strlen(s)-2]; 
   werror((perror_last_was_newline?getpid()+": ":"") 
          +replace(s,"\n","\n"+getpid()+": ") 
          +(lwn?"\n":"")); 
   perror_last_was_newline=lwn; 
} 
 
void create() 
{ 
   add_constant("spawne",spawne); 
   add_constant("perror",perror); 
   add_constant("popen",popen); 
}