X-Git-Url: https://git.llucax.com/software/sercom.git/blobdiff_plain/332f04295e93700fb9e60df7d6eabcd864702057..4b7437a8033c0bc66eae8be038000a86dc54c35b:/sercom/tester.py diff --git a/sercom/tester.py b/sercom/tester.py index d610781..e278417 100644 --- a/sercom/tester.py +++ b/sercom/tester.py @@ -103,6 +103,24 @@ def unzip(bytes, default_dst='.', specific_dst=dict()): # {{{ zfile.close() #}}} +class Multizip(object): #{{{ + def __init__(self, *zip_streams): + self.zips = [ZipFile(StringIO(z), 'r') for z in zip_streams + if z is not None] + self.names = set() + for z in self.zips: + self.names |= set(z.namelist()) + def read(self, name): + for z in self.zips: + try: + return z.read(name) + except KeyError: + pass + raise KeyError(name) + def namelist(self): + return self.names +#}}} + class SecureProcess(object): #{{{ default = dict( max_tiempo_cpu = 120, @@ -270,7 +288,7 @@ def ejecutar_caso_de_prueba(self, path, entrega): #{{{ entrega.shortrepr()) tareas = [t for t in entrega.instancia.ejercicio.enunciado.tareas if isinstance(t, TareaPrueba)] - prueba = entrega.add_prueba(self) + prueba = entrega.add_prueba(self, inicio=datetime.now()) try: try: for tarea in tareas: @@ -301,7 +319,7 @@ Tarea.ejecutar = ejecutar_tarea def ejecutar_comando_fuente(self, path, entrega): #{{{ log.debug(_(u'ComandoFuente.ejecutar(path=%s, entrega=%s)'), path, entrega.shortrepr()) - comando_ejecutado = entrega.add_comando_ejecutado(self) # TODO debería rodear solo la ejecución del comando + comando_ejecutado = entrega.add_comando_ejecutado(self) basetmp = '/tmp/sercom.tester.fuente' # FIXME TODO /var/run/sercom? unzip(self.archivos_entrada, path, # TODO try/except {self.STDIN: '%s.%s.stdin' % (basetmp, comando_ejecutado.id)}) @@ -316,12 +334,8 @@ def ejecutar_comando_fuente(self, path, entrega): #{{{ else: options['preexec_fn'].close_stdin = True a_guardar = set(self.archivos_a_guardar) - if self.archivos_a_comparar: - zip_a_comparar = ZipFile(StringIO(self.archivos_a_comparar), 'r') - a_comparar = set(zip_a_comparar.namelist()) - else: - zip_a_comparar = None - a_comparar = frozenset() + zip_a_comparar = Multizip(self.archivos_a_comparar) + a_comparar = set(zip_a_comparar.namelist()) a_usar = frozenset(a_guardar | a_comparar) if self.STDOUTERR in a_usar: options['stdout'] = file('%s.%s.stdouterr' % (basetmp, @@ -338,12 +352,14 @@ def ejecutar_comando_fuente(self, path, entrega): #{{{ comando_ejecutado.id), 'w') else: options['preexec_fn'].close_stderr = True - log.debug(_(u'Ejecutando como root: %s'), self.comando) + comando = self.comando # FIXME Acá tiene que diferenciarse de ComandoPrueba + comando_ejecutado.inicio = datetime.now() + log.debug(_(u'Ejecutando como root: %s'), comando) os.seteuid(0) # Dios! (para chroot) os.setegid(0) try: try: - proc = sp.Popen(self.comando, **options) + proc = sp.Popen(comando, **options) finally: os.setegid(user_info.gid) # Mortal de nuevo os.seteuid(user_info.uid) @@ -354,7 +370,7 @@ def ejecutar_comando_fuente(self, path, entrega): #{{{ log.error(_(u'Error en el hijo: %s'), e.child_traceback) raise proc.wait() #TODO un sleep grande nos caga todo, ver sercom viejo - comando_ejecutado.fin = datetime.now() # TODO debería rodear solo la ejecución del comando + comando_ejecutado.fin = datetime.now() retorno = self.retorno if retorno != self.RET_ANY: if retorno == self.RET_FAIL: @@ -488,7 +504,8 @@ def ejecutar_comando_prueba(self, path, prueba): #{{{ # y setup/clean de test. log.debug(_(u'ComandoPrueba.ejecutar(path=%s, prueba=%s)'), path, prueba.shortrepr()) - comando_ejecutado = prueba.add_comando_ejecutado(self) # TODO debería rodear solo la ejecución del comando + caso_de_prueba = prueba.caso_de_prueba + comando_ejecutado = prueba.add_comando_ejecutado(self) basetmp = '/tmp/sercom.tester.prueba' # FIXME TODO /var/run/sercom? #{{{ Código que solo va en ComandoPrueba (setup de directorio) rsync = ('rsync', '--stats', '--itemize-changes', '--human-readable', @@ -504,11 +521,11 @@ def ejecutar_comando_prueba(self, path, prueba): #{{{ os.seteuid(user_info.uid) log.debug(_(u'Usuario y grupo efectivos cambiados a %s:%s (%s:%s)'), user_info.user, user_info.group, user_info.uid, user_info.gid) - unzip(prueba.caso_de_prueba.archivos_entrada, path, # TODO try/except - {self.STDIN: '%s.%s.stdin' % (basetmp, comando_ejecutado.id)}) #}}} unzip(self.archivos_entrada, path, # TODO try/except {self.STDIN: '%s.%s.stdin' % (basetmp, comando_ejecutado.id)}) + unzip(caso_de_prueba.archivos_entrada, path, # TODO try/except # FIXME Esto es propio de ComandoPrueba + {self.STDIN: '%s.%s.stdin' % (basetmp, comando_ejecutado.id)}) # FIXME Esto es propio de ComandoPrueba options = dict( close_fds=True, shell=True, @@ -520,12 +537,10 @@ def ejecutar_comando_prueba(self, path, prueba): #{{{ else: options['preexec_fn'].close_stdin = True a_guardar = set(self.archivos_a_guardar) - if self.archivos_a_comparar: - zip_a_comparar = ZipFile(StringIO(self.archivos_a_comparar), 'r') - a_comparar = set(zip_a_comparar.namelist()) - else: - zip_a_comparar = None - a_comparar = frozenset() + a_guardar |= set(caso_de_prueba.archivos_a_guardar) # FIXME Esto es propio de ComandoPrueba + zip_a_comparar = Multizip(caso_de_prueba.archivos_a_comparar, # FIXME Esto es propio de ComandoPrueba + self.archivos_a_comparar) # FIXME Esto es propio de ComandoPrueba + a_comparar = set(zip_a_comparar.namelist()) a_usar = frozenset(a_guardar | a_comparar) if self.STDOUTERR in a_usar: options['stdout'] = file('%s.%s.stdouterr' % (basetmp, @@ -542,12 +557,14 @@ def ejecutar_comando_prueba(self, path, prueba): #{{{ comando_ejecutado.id), 'w') else: options['preexec_fn'].close_stderr = True - log.debug(_(u'Ejecutando como root: %s'), self.comando) + comando = self.comando + ' ' + caso_de_prueba.comando # FIXME Esto es propio de ComandoPrueba + comando_ejecutado.inicio = datetime.now() + log.debug(_(u'Ejecutando como root: %s'), comando) os.seteuid(0) # Dios! (para chroot) os.setegid(0) try: try: - proc = sp.Popen(self.comando, **options) + proc = sp.Popen(comando, **options) finally: os.setegid(user_info.gid) # Mortal de nuevo os.seteuid(user_info.uid) @@ -558,10 +575,10 @@ def ejecutar_comando_prueba(self, path, prueba): #{{{ log.error(_(u'Error en el hijo: %s'), e.child_traceback) raise proc.wait() #TODO un sleep grande nos caga todo, ver sercom viejo - comando_ejecutado.fin_tareas = datetime.now() # TODO debería rodear solo la ejecución del comando + comando_ejecutado.fin = datetime.now() retorno = self.retorno if retorno == self.RET_PRUEBA: # FIXME Esto es propio de ComandoPrueba - retorno = prueba.caso_de_prueba.retorno # FIXME Esto es propio de ComandoPrueba + retorno = caso_de_prueba.retorno # FIXME Esto es propio de ComandoPrueba if retorno != self.RET_ANY: if retorno == self.RET_FAIL: if proc.returncode == 0: