- // comienza la ejecución de casos de prueba
- while ($prueba = $intento->pedir_caso_de_prueba()) {
- logs('Prueba: ' . $prueba->to_line(), DEBUG);
- // ejecuta con fork
- $pid = pcntl_fork();
- if ($pid == -1) {
- // Error al forkear
- $intento->resultado_de_prueba($prueba, false, 'Error al forkear proceso');
- enviar_respuesta_error_log($mail, 'Error al forkear proceso', $intento);
- continue 2;
- } elseif ($pid) {
- // Estamos en el padre, controlamos el tiempo.
- logs("En el padre (hijo: $pid)", DEBUG);
- // TODO controlar tiempo.
- pcntl_waitpid($pid, $exitcode);
- logs("Fin de ejecución de caso de prueba (hijo: $pid, ret: $exitcode)", DEBUG);
- $stderr = false; // FIXME ver si salida de error es vacia.
- if ($exitcode or $stderr) {
- logs('Comando ejecutado ERROR', DEBUG);
- if ($ret) {
- $msg = "El programa salió con código de error $ret";
- $msgs[] = $msg;
- logs($msg, DEBUG);
- }
- if ($stderr) {
- $msg = "El programa imprimió mensajes en la salida de error: '$stderr'";
- $msgs[] = $msg;
- logs($msg, DEBUG);
- }
- $msg = join("\n", $msgs);
- $intento->resultado_de_prueba($prueba, false, $msg);
- enviar_respuesta(R_ERR, $mail, $msg, $intento);
- exit(0); // salgo ok del hijo
- } else { // Sin errores en la ejecución.
- logs('Comando ejecutado OK', DEBUG);
- // TODO
- $salidas = array();
- foreach ($prueba->salidas as $salida) {
- // TODO hacer diffs.
- if ($salida == 'stdout') {
- $salidas[$salida] = $stdout;
- } else {
- $salidas[$salida] = file_get_contents($salida);
- }
- }
- logs('Salidas: ' . var_export($salidas, true), DEBUG);
- $intento->resultado_de_prueba($prueba, true, 'Salidas: ' . var_export($salidas, true));
- enviar_respuesta(R_ERR, $mail, 'Salidas: ' . var_export($salidas, true), $intento);
- exit(0); // salgo ok del hijo
- }
- } else {
- // Estamos en el hijo, corremos en chroot.
- logs('En el hijo', DEBUG);
- // Hago chroot.
- if (!@chroot('chroot')) {
- $intento->resultado_de_prueba($prueba, false, 'Error al hacer chroot');
- enviar_respuesta_error_log($mail, 'Error al hacer chroot', $intento);
- exit(1); // salgo del hijo
- }
- logs('Chrooteado, cwd = '.getcwd(), DEBUG);
- if (!@posix_setuid($usrinfo['uid'])) {
- $intento->resultado_de_prueba($prueba, false, "Error al cambiar al uid '{$usrinfo['uid']}'");
- enviar_respuesta_error_log($mail, "Error al cambiar al uid '{$usrinfo['uid']}'", $intento);
- exit(2); // salgo del hijo
- }
- // TODO poner stdout y stderr en archivos para hacer diff o hacer diffs en memoria
- $stdout = null;
- if (in_array('stdout', $prueba->salidas)) $stdout = '';
- logs('Se ejecutará: chroot chroot /tp ' . escapeshellarg($prueba->params), DEBUG);
- if (!pcntl_exec('/redir', array('tp', 'stdin', 'stdout', 'stderr', '/tp', '0', $prueba->params))) {
- logserr('No se pudo ejecutar el comando');
- $intento->resultado_de_prueba($prueba, false, 'No se pudo ejecutar el tp');
- enviar_respuesta_error_log($mail, 'No se pudo ejecutar el tp', $intento);
- exit(4); // salgo del hijo
- }
- exit(100); // salgo del hijo (no deberia llegar nunca aca).
- }
- break; // FIXME
- }
+def params2seq(params):
+ r"""Parsea un string de forma similar al bash, separando por espacios y
+ teniendo en cuenta comillas simples y dobles para agrupar. Para poner
+ comillas se puede usar el \ como caracter de escape (\' y \") y también
+ interpreta \n y \t. Devuelve una lista con los parámetros encontrados."""
+ # Constantes
+ SEP, TOKEN, DQUOTE, SQUOTE = ' ', None, '"', "'"
+ seq = []
+ buff = ''
+ escape = False
+ state = SEP
+ for c in params:
+ # Es un caracter escapado
+ if escape:
+ if c == 'n':
+ buff += '\n'
+ elif c == 't':
+ buff += '\t'
+ else:
+ buff += c
+ escape = False
+ continue
+ # Es una secuencia de escape
+ if c == '\\':
+ escape = True
+ continue
+ # Si está buscando espacios
+ if state == SEP:
+ if c == SEP:
+ continue
+ else:
+ state = TOKEN # Encontró
+ if state == TOKEN:
+ if c == DQUOTE:
+ state = DQUOTE
+ continue
+ if c == SQUOTE:
+ state = SQUOTE
+ continue
+ if c == SEP:
+ state = SEP
+ seq.append(buff)
+ buff = ''
+ continue
+ buff += c
+ continue
+ if state == DQUOTE:
+ if c == DQUOTE:
+ state = TOKEN
+ continue
+ buff += c
+ continue
+ if state == SQUOTE:
+ if c == SQUOTE:
+ state = TOKEN
+ continue
+ buff += c
+ continue
+ raise Exception, 'No tiene sentido'
+ if state == DQUOTE or state == SQUOTE:
+ raise Exception, 'Parse error, falta cerrar comilla (%s)' % state
+ if buff:
+ seq.append(buff)
+ return seq