grupos = MultipleJoin('Grupo')
ejercicios = MultipleJoin('Ejercicio', orderBy='numero')
- def __init__(self, docentes=[], ejercicios=[], **kargs):
- super(Curso, self).__init__(**kargs)
+ def __init__(self, docentes=[], ejercicios=[], alumnos=[], **kw):
+ super(Curso, self).__init__(**kw)
for d in docentes:
self.add_docente(d)
for (n, e) in enumerate(ejercicios):
self.add_ejercicio(n, e)
+ for a in alumnos:
+ self.add_alumno(a)
+
+ def set(self, docentes=None, ejercicios=None, alumnos=None, **kw):
+ super(Curso, self).set(**kw)
+ if docentes is not None:
+ for d in DocenteInscripto.selectBy(curso=self):
+ d.destroySelf()
+ for d in docentes:
+ self.add_docente(d)
+ if ejercicios is not None:
+ for e in Ejercicio.selectBy(curso=self):
+ e.destroySelf()
+ for (n, e) in enumerate(ejercicios):
+ self.add_ejercicio(n, e)
+ if alumnos is not None:
+ for a in AlumnoInscripto.selectBy(curso=self):
+ a.destroySelf()
+ for a in alumnos:
+ self.add_alumno(a)
+
+ def add_docente(self, docente, **kw):
+ if isinstance(docente, Docente):
+ kw['docente'] = docente
+ else:
+ kw['docenteID'] = docente
+ return DocenteInscripto(curso=self, **kw)
+
+ def remove_docente(self, docente):
+ if isinstance(docente, Docente):
+ docente = docente.id
+ # FIXME esto deberian arreglarlo en SQLObject y debería ser
+ # DocenteInscripto.pk.get(self, docente).destroySelf()
+ DocenteInscripto.pk.get(self.id, docente).destroySelf()
+
+ def add_alumno(self, alumno, **kw):
+ if isinstance(alumno, Alumno):
+ kw['alumno'] = alumno
+ else:
+ kw['alumnoID'] = alumno
+ return AlumnoInscripto(curso=self, **kw)
- def add_docente(self, docente, **kargs):
- return DocenteInscripto(curso=self, docente=docente, **kargs)
+ def remove_alumno(self, alumno):
+ if isinstance(alumno, Alumno):
+ alumno = alumno.id
+ # FIXME esto deberian arreglarlo en SQLObject
+ AlumnoInscripto.pk.get(self.id, alumno).destroySelf()
- def add_alumno(self, alumno, **kargs):
- return AlumnoInscripto(curso=self, alumno=alumno, **kargs)
+ def add_grupo(self, nombre, **kw):
+ return Grupo(curso=self, nombre=unicode(nombre), **kw)
- def add_grupo(self, nombre, **kargs):
- return Grupo(curso=self, nombre=unicode(nombre), **kargs)
+ def remove_grupo(self, nombre):
+ # FIXME esto deberian arreglarlo en SQLObject
+ Grupo.pk.get(self.id, nombre).destroySelf()
+
+ def add_ejercicio(self, numero, enunciado, **kw):
+ if isinstance(enunciado, Enunciado):
+ kw['enunciado'] = enunciado
+ else:
+ kw['enunciadoID'] = enunciado
+ return Ejercicio(curso=self, numero=numero, **kw)
- def add_ejercicio(self, numero, enunciado, **kargs):
- return Ejercicio(curso=self, numero=numero, enunciado=enunciado, **kargs)
+ def remove_ejercicio(self, numero):
+ # FIXME esto deberian arreglarlo en SQLObject
+ Ejercicio.pk.get(self.id, numero).destroySelf()
def __repr__(self):
return 'Curso(id=%s, anio=%s, cuatrimestre=%s, numero=%s, ' \
observaciones = UnicodeCol(default=None)
activo = BoolCol(notNone=True, default=True)
# Joins
- roles = RelatedJoin('Rol')
+ roles = RelatedJoin('Rol', addRemoveName='_rol')
- def __init__(self, password=None, roles=[], **kargs):
- passwd = password and encryptpw(password)
- super(Usuario, self).__init__(contrasenia=passwd, **kargs)
+ def __init__(self, password=None, roles=[], **kw):
+ if password is not None:
+ kw['contrasenia'] = encryptpw(password)
+ super(Usuario, self).__init__(**kw)
for r in roles:
- self.addRol(r)
+ self.add_rol(r)
+
+ def set(self, password=None, roles=None, **kw):
+ if password is not None:
+ kw['contrasenia'] = encryptpw(password)
+ super(Usuario, self).set(**kw)
+ if roles is not None:
+ for r in self.roles:
+ self.remove_rol(r)
+ for r in roles:
+ self.add_rol(r)
def _get_user_name(self): # para identity
return self.usuario
class Docente(Usuario): #{{{
_inheritable = False
# Campos
- nombrado = BoolCol(notNone=True, default=True)
+ nombrado = BoolCol(notNone=True, default=True)
# Joins
- enunciados = MultipleJoin('Enunciado', joinColumn='autor_id')
- inscripciones = MultipleJoin('DocenteInscripto')
+ enunciados = MultipleJoin('Enunciado', joinColumn='autor_id')
+ cursos = MultipleJoin('DocenteInscripto')
- def __init__(self, **kargs):
- super(Docente, self).__init__(**kargs)
+ def add_entrega(self, instancia, **kw):
+ return Entrega(instancia=instancia, **kw)
- def add_entrega(self, instancia, **kargs):
- return Entrega(instancia=instancia, **kargs)
-
- def add_enunciado(self, nombre, anio, cuatrimestre, **kargs):
+ def add_enunciado(self, nombre, anio, cuatrimestre, **kw):
return Enunciado(nombre=nombre, anio=anio, cuatrimestre=cuatrimestre,
- autor=self, **kargs)
+ autor=self, **kw)
+
+ def remove_enunciado(self, nombre, anio, cuatrimestre):
+ Enunciado.pk.get(nombre=nombre, anio=anio,
+ cuatrimestre=cuatrimestre).destroySelf()
def __repr__(self):
return 'Docente(id=%s, usuario=%s, nombre=%s, password=%s, email=%s, ' \
# Joins
inscripciones = MultipleJoin('AlumnoInscripto')
- def __init__(self, padron=None, **kargs):
- if padron: kargs['usuario'] = padron
- super(Alumno, self).__init__(**kargs)
+ def __init__(self, padron=None, **kw):
+ if padron: kw['usuario'] = padron
+ super(Alumno, self).__init__(**kw)
+
+ def set(self, padron=None, **kw):
+ if padron: kw['usuario'] = padron
+ super(Alumno, self).set(**kw)
def _get_padron(self): # alias para poder referirse al alumno por padron
return self.usuario
def _set_padron(self, padron):
self.usuario = padron
+ @classmethod
+ def byPadron(cls, padron):
+ return cls.byUsuario(unicode(padron))
+
def __repr__(self):
return 'Alumno(id=%s, padron=%s, nombre=%s, password=%s, email=%s, ' \
'telefono=%s, activo=%s, creado=%s, observaciones=%s)' \
class Tarea(InheritableSQLObject): #{{{
class sqlmeta:
- createSQL = r'''
+ createSQL = dict(sqlite=r'''
CREATE TABLE dependencia (
padre_id INTEGER NOT NULL CONSTRAINT tarea_id_exists
- REFERENCES tarea(id),
+ REFERENCES tarea(id) ON DELETE CASCADE,
hijo_id INTEGER NOT NULL CONSTRAINT tarea_id_exists
- REFERENCES tarea(id),
+ REFERENCES tarea(id) ON DELETE CASCADE,
orden INT,
PRIMARY KEY (padre_id, hijo_id)
-)'''
+)''')
# Clave
nombre = UnicodeCol(length=30, alternateID=True)
# Campos
descripcion = UnicodeCol(length=255, default=None)
# Joins
- def __init__(self, dependencias=(), **kargs):
- super(Tarea, self).__init__(**kargs)
+ def __init__(self, dependencias=(), **kw):
+ super(Tarea, self).__init__(**kw)
if dependencias:
self.dependencias = dependencias
+ def set(self, dependencias=None, **kw):
+ super(Tarea, self).set(**kw)
+ if dependencias is not None:
+ self.dependencias = dependencias
+
def _get_dependencias(self):
OtherTarea = Alias(Tarea, 'other_tarea')
self.__dependencias = tuple(Tarea.select(
class Enunciado(SQLObject): #{{{
class sqlmeta:
- createSQL = r'''
+ createSQL = dict(sqlite=r'''
CREATE TABLE enunciado_tarea (
enunciado_id INTEGER NOT NULL CONSTRAINT enunciado_id_exists
- REFERENCES enunciado(id),
+ REFERENCES enunciado(id) ON DELETE CASCADE,
tarea_id INTEGER NOT NULL CONSTRAINT tarea_id_exists
- REFERENCES tarea(id),
+ REFERENCES tarea(id) ON DELETE CASCADE,
orden INT,
PRIMARY KEY (enunciado_id, tarea_id)
-)'''
+)''')
# Clave
nombre = UnicodeCol(length=60)
anio = IntCol(notNone=True)
cuatrimestre = IntCol(notNone=True)
pk = DatabaseIndex(nombre, anio, cuatrimestre, unique=True)
# Campos
- autor = ForeignKey('Docente')
+ autor = ForeignKey('Docente', cascade='null')
descripcion = UnicodeCol(length=255, default=None)
creado = DateTimeCol(notNone=True, default=DateTimeCol.now)
archivo = BLOBCol(default=None)
ejercicios = MultipleJoin('Ejercicio')
casos_de_prueba = MultipleJoin('CasoDePrueba')
- def __init__(self, tareas=(), **kargs):
- super(Enunciado, self).__init__(**kargs)
+ def __init__(self, tareas=(), **kw):
+ super(Enunciado, self).__init__(**kw)
if tareas:
self.tareas = tareas
+ def set(self, tareas=None, **kw):
+ super(Enunciado, self).set(**kw)
+ if tareas is not None:
+ self.tareas = tareas
+
@classmethod
def selectByCurso(self, curso):
return Enunciado.selectBy(cuatrimestre=curso.cuatrimestre, anio=curso.anio)
- def add_caso_de_prueba(self, nombre, **kargs):
- return CasoDePrueba(enunciado=self, nombre=nombre, **kargs)
+ def add_caso_de_prueba(self, nombre, **kw):
+ return CasoDePrueba(enunciado=self, nombre=nombre, **kw)
def _get_tareas(self):
self.__tareas = tuple(Tarea.select(
class CasoDePrueba(SQLObject): #{{{
# Clave
- enunciado = ForeignKey('Enunciado')
+ enunciado = ForeignKey('Enunciado', cascade=True)
nombre = UnicodeCol(length=40, notNone=True)
pk = DatabaseIndex(enunciado, nombre, unique=True)
# Campos
-# privado = IntCol(default=None) TODO iria en instancia_de_entrega_caso_de_prueba
+ privado = IntCol(default=None) # TODO iria en instancia_de_entrega_caso_de_prueba
parametros = ParamsCol(length=255, default=None)
retorno = IntCol(default=None)
tiempo_cpu = FloatCol(default=None)
descripcion = UnicodeCol(length=255, default=None)
+ activo = BoolCol(notNone=True, default=True)
# Joins
pruebas = MultipleJoin('Prueba')
class Ejercicio(SQLObject): #{{{
# Clave
- curso = ForeignKey('Curso', notNone=True)
+ curso = ForeignKey('Curso', notNone=True, cascade=True)
numero = IntCol(notNone=True)
pk = DatabaseIndex(curso, numero, unique=True)
# Campos
- enunciado = ForeignKey('Enunciado', notNone=True)
+ enunciado = ForeignKey('Enunciado', notNone=True, cascade=False)
grupal = BoolCol(notNone=True, default=False)
# Joins
instancias = MultipleJoin('InstanciaDeEntrega')
- def add_instancia(self, numero, inicio, fin, **kargs):
+ def add_instancia(self, numero, inicio, fin, **kw):
return InstanciaDeEntrega(ejercicio=self, numero=numero, inicio=inicio,
- fin=fin, **kargs)
+ fin=fin, **kw)
+
+ def remove_instancia(self, numero):
+ InstanciaDeEntrega.pk.get(ejercicio=self, numero=numero).destroySelf()
def __repr__(self):
return 'Ejercicio(id=%s, curso=%s, numero=%s, enunciado=%s, ' \
class InstanciaDeEntrega(SQLObject): #{{{
class sqlmeta:
- createSQL = r'''
+ createSQL = dict(sqlite=r'''
CREATE TABLE instancia_tarea (
instancia_id INTEGER NOT NULL CONSTRAINT instancia_id_exists
- REFERENCES instancia_de_entrega(id),
+ REFERENCES instancia_de_entrega(id) ON DELETE CASCADE,
tarea_id INTEGER NOT NULL CONSTRAINT tarea_id_exists
- REFERENCES tarea(id),
+ REFERENCES tarea(id) ON DELETE CASCADE,
orden INT,
PRIMARY KEY (instancia_id, tarea_id)
-)'''
+)''')
# Clave
- ejercicio = ForeignKey('Ejercicio', notNone=True)
+ ejercicio = ForeignKey('Ejercicio', notNone=True, cascade=True)
numero = IntCol(notNone=True)
+ pk = DatabaseIndex(ejercicio, numero, unique=True)
# Campos
inicio = DateTimeCol(notNone=True)
fin = DateTimeCol(notNone=True)
# Joins
entregas = MultipleJoin('Entrega', joinColumn='instancia_id')
correcciones = MultipleJoin('Correccion', joinColumn='instancia_id')
- casos_de_prueba = RelatedJoin('CasoDePrueba') # TODO CasoInstancia -> private
- def __init__(self, tareas=(), **kargs):
- super(InstanciaDeEntrega, self).__init__(**kargs)
+ def __init__(self, tareas=(), **kw):
+ super(InstanciaDeEntrega, self).__init__(**kw)
if tareas:
self.tareas = tareas
+ def set(self, tareas=None, **kw):
+ super(InstanciaDeEntrega, self).set(**kw)
+ if tareas is not None:
+ self.tareas = tareas
+
def _get_tareas(self):
self.__tareas = tuple(Tarea.select(
AND(
class DocenteInscripto(SQLObject): #{{{
# Clave
- curso = ForeignKey('Curso', notNone=True)
- docente = ForeignKey('Docente', notNone=True)
+ curso = ForeignKey('Curso', notNone=True, cascade=True)
+ docente = ForeignKey('Docente', notNone=True, cascade=True)
pk = DatabaseIndex(curso, docente, unique=True)
# Campos
corrige = BoolCol(notNone=True, default=True)
# Joins
alumnos = MultipleJoin('AlumnoInscripto', joinColumn='tutor_id')
tutorias = MultipleJoin('Tutor', joinColumn='docente_id')
- entregas = MultipleJoin('Entrega', joinColumn='instancia_id')
correcciones = MultipleJoin('Correccion', joinColumn='corrector_id')
- def add_correccion(self, entrega, **kargs):
+ def add_correccion(self, entrega, **kw):
return Correccion(instancia=entrega.instancia, entrega=entrega,
- entregador=entrega.entregador, corrector=self, **kargs)
+ entregador=entrega.entregador, corrector=self, **kw)
+
+ def remove_correccion(self, instancia, entregador):
+ Correccion.pk.get(instancia=instancia,
+ entregador=entregador).destroySelf()
def __repr__(self):
return 'DocenteInscripto(id=%s, docente=%s, corrige=%s, ' \
entregas = MultipleJoin('Entrega')
correcciones = MultipleJoin('Correccion')
- def add_entrega(self, instancia, **kargs):
- return Entrega(instancia=instancia, entregador=self, **kargs)
+ def add_entrega(self, instancia, **kw):
+ return Entrega(instancia=instancia, entregador=self, **kw)
def __repr__(self):
raise NotImplementedError, 'Clase abstracta!'
class Grupo(Entregador): #{{{
_inheritable = False
# Clave
- curso = ForeignKey('Curso', notNone=True)
+ curso = ForeignKey('Curso', notNone=True, cascade=True)
nombre = UnicodeCol(length=20, notNone=True)
+ pk = DatabaseIndex(curso, nombre, unique=True)
# Campos
- responsable = ForeignKey('AlumnoInscripto', default=None)
+ responsable = ForeignKey('AlumnoInscripto', default=None, cascade='null')
# Joins
miembros = MultipleJoin('Miembro')
tutores = MultipleJoin('Tutor')
- def __init__(self, miembros=[], tutores=[], **kargs):
- super(Grupo, self).__init__(**kargs)
+ def __init__(self, miembros=[], tutores=[], **kw):
+ super(Grupo, self).__init__(**kw)
for a in miembros:
self.add_miembro(a)
for d in tutores:
self.add_tutor(d)
- def add_miembro(self, alumno, **kargs):
- return Miembro(grupo=self, alumno=alumno, **kargs)
+ def set(self, miembros=None, tutores=None, **kw):
+ super(Grupo, self).set(**kw)
+ if miembros is not None:
+ for m in Miembro.selectBy(grupo=self):
+ m.destroySelf()
+ for m in miembros:
+ self.add_miembro(m)
+ if tutores is not None:
+ for t in Tutor.selectBy(grupo=self):
+ t.destroySelf()
+ for t in tutores:
+ self.add_tutor(t)
+
+ def add_miembro(self, alumno, **kw):
+ if isinstance(alumno, AlumnoInscripto):
+ kw['alumno'] = alumno
+ else:
+ kw['alumnoID'] = alumno
+ return Miembro(grupo=self, **kw)
- def add_tutor(self, docente, **kargs):
- return Tutor(grupo=self, docente=docente, **kargs)
+ def remove_miembro(self, alumno):
+ if isinstance(alumno, AlumnoInscripto):
+ Miembro.pk.get(grupo=self, alumno=alumno).destroySelf()
+ else:
+ Miembro.pk.get(grupo=self, alumnoID=alumno).destroySelf()
+
+ def add_tutor(self, docente, **kw):
+ if isinstance(docente, DocenteInscripto):
+ kw['docente'] = docente
+ else:
+ kw['docenteID'] = docente
+ return Tutor(grupo=self, **kw)
+
+ def remove_tutor(self, docente):
+ if isinstance(docente, DocenteInscripto):
+ Tutor.pk.get(grupo=self, docente=docente).destroySelf()
+ else:
+ Tutor.pk.get(grupo=self, docenteID=docente).destroySelf()
def __repr__(self):
return 'Grupo(id=%s, nombre=%s, responsable=%s, nota=%s, ' \
class AlumnoInscripto(Entregador): #{{{
_inheritable = False
# Clave
- curso = ForeignKey('Curso', notNone=True)
- alumno = ForeignKey('Alumno', notNone=True)
+ curso = ForeignKey('Curso', notNone=True, cascade=True)
+ alumno = ForeignKey('Alumno', notNone=True, cascade=True)
pk = DatabaseIndex(curso, alumno, unique=True)
# Campos
condicional = BoolCol(notNone=True, default=False)
- tutor = ForeignKey('DocenteInscripto', default=None)
+ tutor = ForeignKey('DocenteInscripto', default=None, cascade='null')
# Joins
responsabilidades = MultipleJoin('Grupo', joinColumn='responsable_id')
membresias = MultipleJoin('Miembro', joinColumn='alumno_id')
class Tutor(SQLObject): #{{{
# Clave
- grupo = ForeignKey('Grupo', notNone=True)
- docente = ForeignKey('DocenteInscripto', notNone=True)
+ grupo = ForeignKey('Grupo', notNone=True, cascade=True)
+ docente = ForeignKey('DocenteInscripto', notNone=True, cascade=True)
pk = DatabaseIndex(grupo, docente, unique=True)
# Campos
alta = DateTimeCol(notNone=True, default=DateTimeCol.now)
class Miembro(SQLObject): #{{{
# Clave
- grupo = ForeignKey('Grupo', notNone=True)
- alumno = ForeignKey('AlumnoInscripto', notNone=True)
+ grupo = ForeignKey('Grupo', notNone=True, cascade=True)
+ alumno = ForeignKey('AlumnoInscripto', notNone=True, cascade=True)
pk = DatabaseIndex(grupo, alumno, unique=True)
# Campos
nota = DecimalCol(size=3, precision=1, default=None)
class Entrega(SQLObject): #{{{
# Clave
- instancia = ForeignKey('InstanciaDeEntrega', notNone=True)
- entregador = ForeignKey('Entregador', default=None) # Si es None era un Docente
+ instancia = ForeignKey('InstanciaDeEntrega', notNone=True, cascade=False)
+ entregador = ForeignKey('Entregador', default=None, cascade=False) # Si es None era un Docente
fecha = DateTimeCol(notNone=True, default=DateTimeCol.now)
pk = DatabaseIndex(instancia, entregador, fecha, unique=True)
# Campos
codigo_dict = r'0123456789abcdefghijklmnopqrstuvwxyz_.,*@#+'
codigo_format = r'%m%d%H%M%S'
- def add_tarea_ejecutada(self, tarea, **kargs):
- return TareaEjecutada(tarea=tarea, entrega=self, **kargs)
+ def add_tarea_ejecutada(self, tarea, **kw):
+ return TareaEjecutada(tarea=tarea, entrega=self, **kw)
def _get_codigo(self):
if not hasattr(self, '_codigo'): # cache
class Correccion(SQLObject): #{{{
# Clave
- instancia = ForeignKey('InstanciaDeEntrega', notNone=True)
- entregador = ForeignKey('Entregador', notNone=True) # Docente no tiene
+ instancia = ForeignKey('InstanciaDeEntrega', notNone=True, cascade=False)
+ entregador = ForeignKey('Entregador', notNone=True, cascade=False) # Docente no tiene
pk = DatabaseIndex(instancia, entregador, unique=True)
# Campos
- entrega = ForeignKey('Entrega', notNone=True)
- corrector = ForeignKey('DocenteInscripto', notNone=True)
+ entrega = ForeignKey('Entrega', notNone=True, cascade=False)
+ corrector = ForeignKey('DocenteInscripto', default=None, cascade='null')
asignado = DateTimeCol(notNone=True, default=DateTimeCol.now)
corregido = DateTimeCol(default=None)
nota = DecimalCol(size=3, precision=1, default=None)
observaciones = UnicodeCol(default=None)
+ def _get_entregas(self):
+ return list(Entrega.selectBy(instancia=self.instancia, entregador=self.entregador))
+
def __repr__(self):
return 'Correccion(instancia=%s, entregador=%s, entrega=%s, ' \
'corrector=%s, asignado=%s, corregido=%s, nota=%s, ' \
self.corregido, self.nota, self.observaciones)
def shortrepr(self):
+ if not self.corrector:
+ return '%s' % self.entrega.shortrepr()
return '%s,%s' % (self.entrega.shortrepr(), self.corrector.shortrepr())
#}}}
class TareaEjecutada(InheritableSQLObject): #{{{
# Clave
- tarea = ForeignKey('Tarea', notNone=True)
- entrega = ForeignKey('Entrega', notNone=True)
+ tarea = ForeignKey('Tarea', notNone=True, cascade=False)
+ entrega = ForeignKey('Entrega', notNone=True, cascade=False)
pk = DatabaseIndex(tarea, entrega, unique=True)
# Campos
inicio = DateTimeCol(notNone=True, default=DateTimeCol.now)
# Joins
pruebas = MultipleJoin('Prueba')
- def add_prueba(self, caso_de_prueba, **kargs):
+ def add_prueba(self, caso_de_prueba, **kw):
return Prueba(tarea_ejecutada=self, caso_de_prueba=caso_de_prueba,
- **kargs)
+ **kw)
def __repr__(self):
return 'TareaEjecutada(tarea=%s, entrega=%s, inicio=%s, fin=%s, ' \
class Prueba(SQLObject): #{{{
# Clave
- tarea_ejecutada = ForeignKey('TareaEjecutada', notNone=True)
- caso_de_prueba = ForeignKey('CasoDePrueba', notNone=True)
+ tarea_ejecutada = ForeignKey('TareaEjecutada', notNone=True, cascade=False)
+ caso_de_prueba = ForeignKey('CasoDePrueba', notNone=True, cascade=False)
pk = DatabaseIndex(tarea_ejecutada, caso_de_prueba, unique=True)
# Campos
inicio = DateTimeCol(notNone=True, default=DateTimeCol.now)
class Rol(SQLObject): #{{{
# Clave
nombre = UnicodeCol(length=255, alternateID=True,
- alternateMethodName="by_group_name")
+ alternateMethodName='by_nombre')
# Campos
descripcion = UnicodeCol(length=255, default=None)
creado = DateTimeCol(notNone=True, default=datetime.now)
permisos = TupleCol(notNone=True)
# Joins
- usuarios = RelatedJoin('Usuario')
+ usuarios = RelatedJoin('Usuario', addRemoveName='_usuario')
+
+ def by_group_name(self, name): # para identity
+ return self.by_nombre(name)
#}}}
# No es un SQLObject porque no tiene sentido agregar/sacar permisos, están