ifms-conselho/connect.py
2023-03-27 14:55:54 -04:00

515 lines
64 KiB
Python

# coding: utf8
# ---introdução----------------------
# 1-justificativa
# 2-apresentação
# 3-ação
# ---justificativa----------------------
# 1 - tornar a experiência do conselho agradável?
# 2 - coletar informações precisas do desempenho dos estudantes e turmas ?
# 3 - fazer a informação colhida não se perca, mas cumpra seu objetivo?
# 4 - fornecer um canal aberto em tempo real entre: professores, cordenação, e equipe pedagógica?
# ---fim----------------------
# 1. email de boas vindas
# 2. tela inicial e escolha da turmas / disciplina
# 3. ok / alerta / participa
# 4. relatorio / report do estudante
# 5. relatorio / report da turma
# -------------------------
# 1. ideia da estrutura para o conselho
# 2. sugestões
# 3.
# -------------------------
from libx import *
# ----------------------------------------------------------------------------------------------
app = Flask(__name__)
app.secret_key = APP_SECRET_KEY
app.permanent_session_lifetime = timedelta(days=APP_SESSION_DAYS)
# ----------------------------------------------------------------------------------------------
print("-"*40)
print("HOST_ADDRESS = ", HOST_ADDRESS)
print("EXTERN_ADDRESS = ", EXTERN_ADDRESS)
print("APP_PORT = ", APP_PORT)
print("MONGO_ADDRESS = ", MONGO_ADDRESS)
print("MONGO_PORT = ", MONGO_PORT)
print("GMAIL_USER = ", GMAIL_USER)
print("-"*40)
# ----------------------------------------------------------------------------------------------
CONNECT_DB = MongoClient(MONGO_LINK).get_database(DATABASE_NAME)
CONTATOS = CONNECT_DB.get_collection('contatos')
PROFESSORES = CONNECT_DB.get_collection('professores')
MATRICULAS = CONNECT_DB.get_collection('matriculas')
DISCIPLINAS = CONNECT_DB.get_collection('disciplinas')
FOTOS = CONNECT_DB.get_collection('fotos')
MARCAS = CONNECT_DB.get_collection('marcas')
COMENTARIOS = CONNECT_DB.get_collection('comentarios')
# ----------------------------------------------------------------------------------------------
like_nao = ""
like_sim = ""
atencao_nao = ""
atencao_sim = ""
# participa_nao = ""
# participa_sim = ""
# participa_nao = "";
# participa_sim = "";
# participa_nao = "";
# participa_sim = "";
participa_sim = ""
participa_nao = ""
comentario_nao = ""
comentario_sim = ""
imagens_marcar = {"ok": {True: like_sim, False: like_nao}, "alerta": {
True: atencao_sim, False: atencao_nao}, "participa": {True: participa_sim, False: participa_nao}}
opacity = {True: "1.0", False: "0.2"}
# ----------------------------------------------------------------------------------------------
def MSG(s):
return ''.join([
s,
])
def IMG_64(img_type, img_name):
encoded_string = ""
image_file = open(img_type+"/"+img_name+".png", "rb")
encoded_string = base64.b64encode(image_file.read())
return encoded_string
def NOME_SIMPLIFICADO(s):
nn = s.split(" ")
n = ""
for k in nn:
if n == "":
n += (k + " ")
else:
if len(k) > 0:
n += k[0]
return n.title()
return n.title()
def NOME_INICIAIS(s):
nn = s.split(" ")
n = ""
for k in nn:
if n == "":
n += k[0].title()
else:
if len(k) > 0:
n += k[0].title()
return n
return n.title()
def TELA_PRINCIAL():
r = []
arquivo = open("static/arquivo.html", "r", encoding='UTF8').read()
arquivo = arquivo.replace(
"[NOME_PROFESSOR]", NOME_INICIAIS(session["nome"]))
r.append(arquivo)
return ''.join(r)
def LISTA_DISCIPLINAS(cmd):
disciplinas = []
for disciplina in DISCIPLINAS.find({"curso": cmd[1], "semestre": cmd[2]}).sort("disciplina", 1):
disciplinas.append(disciplina["disciplina"])
return {"cmd": cmd, "disciplinas": disciplinas}
def CARREGAR(cmd):
r = []
lista_estudantes = []
professor = session["nome"]
filtro_busca1 = {}
if cmd[4] == "Equipe Pedagógica" or cmd[4] == "NAPNE":
filtro_busca1 = {"curso": cmd[1], "semestre": cmd[2],
"turno": cmd[3], "turma": cmd[5], "ano_sem": cmd[6]}
else:
filtro_busca1 = {"curso": cmd[1], "semestre": cmd[2], "turno": cmd[3],
"disciplina": cmd[4], "turma": cmd[5], "ano_sem": cmd[6]}
for aluno in MATRICULAS.find(filtro_busca1).sort("nome", 1):
matricula = aluno["matricula"]
f = ""
for foto in FOTOS.find({"matricula": aluno["matricula"]}):
f = foto["foto"]
foto = "<div style='position:relative;left:0;overflow:hidden; width:20vh; height:20vh'><img class='hover1' style = 'position:absolute;left:0;width:19.5vh;' src='" + \
f+f"' onclick=\"show_student('{matricula}');\" ></div>"
nome = NOME_SIMPLIFICADO(aluno["nome"])
if not matricula in lista_estudantes:
lista_estudantes.append(matricula)
# style_next_invitations=" display: inline-block; width:20vh; height:37vh;margin-left:1vw; margin-right:1vw;margin-top:1vw; box-shadow: 0px 0px 1vw rgba(0, 0, 0, 0.12), 2px 2vw 2vw rgba(0, 0, 0, 0.24);background: #eAeAeA;border-radius:1vw;"
num_alerta = MARCAS.count_documents(
{"matricula": matricula, "tipo": "alerta", "valor": True, "ano_sem": cmd[6]})
if num_alerta >= NUM_MAX_ALERTA:
background = "#ff9999"
else:
background = "#ffffff"
style_next_invitations = f" display: inline-block; width:20vh; height:38vh;margin-left:1vw; margin-right:1vw;margin-top:1vw; box-shadow: 0px 0px 1vw rgba(0, 0, 0, 0.12), 2px 2vw 2vw rgba(0, 0, 0, 0.24);background: {background};border-radius:1vw;"
# style_next_invitations="width:30vw; height:25vh;margin-left:5vw; margin-right:5vw; margin-top:2.5vw;"
# r.append("<div style=' display: inline-block;height:25vh; '>")
r.append(f"<div style='{style_next_invitations}'>")
r.append("<table width='100%'>")
r.append("<tr>")
r.append(
f"<td colspan='3' style='font-family: Lato;font-style: bold;text-align: center;font-size: 2.5vh; color: #444444;'><br>{nome}<br><br></td>")
r.append("</tr>")
r.append("<tr>")
r.append(f"<td colspan='3'>{foto}</td>")
r.append("</tr>")
r.append("<tr>")
for item in ["ok", "alerta", "participa"]:
r.append("<td style='text-align: center;font-size: 1.0vh;'>")
filtro = {"matricula": matricula, "tipo": item,
"disciplina": cmd[4], "professor": professor, "ano_sem": cmd[6]}
marcado = VERIFICA_MARCACAO(filtro)
img = imagens_marcar[item][marcado]
o = opacity[marcado]
r.append(
f"<br><img id='{matricula}_{item}' class='hover2' style ='opacity:{o}; height:3.5vh;' onclick='marcar(\"{matricula}\",\"{item}\")' src='{img}'>")
r.append("</td>")
r.append("</tr>")
r.append("</table>")
r.append("</div>")
return {"cmd": cmd, "ret": ''.join(r)}
def VERIFICA_MARCACAO(filtro):
for marca in MARCAS.find(filtro):
if marca["valor"]:
return True
return False
def MUDA_MARCACAO(cmd):
professor = session["nome"]
# 1 2 3 4 5 6 7
# api("marcar"+"|"+matricula+"|"+tipo+"|"+disciplina+"|"+curso+"|"+semestre+"|"+turno+"|"+turma);
filtro = {"matricula": cmd[1], "tipo": cmd[2], "disciplina": cmd[3], "curso": cmd[4],
"semestre": cmd[5], "turno": cmd[6], "turma": cmd[7], "ano_sem": cmd[8], "professor": professor}
if VERIFICA_MARCACAO(filtro):
MARCAS.update_one(filtro, {"$set": {"valor": False}}, upsert=True)
return {"cmd": cmd, "valor": False}
else:
MARCAS.update_one(filtro, {"$set": {"valor": True}}, upsert=True)
return {"cmd": cmd, "valor": True}
def SHOW_STUDENT(cmd):
matricula = cmd[1]
if matricula != "":
professor = session["nome"]
foto = ""
for m in MATRICULAS.find({"matricula": cmd[1]}).limit(1):
nome = m["nome"].title()
foto
for f in FOTOS.find({"matricula": cmd[1]}):
foto = f["foto"]
telefones = ""
emails = ""
for c in CONTATOS.find({"matricula": cmd[1]}):
telefones = c["telefones"]
emails = c["emails"]
comentarios = LISTA_COMENTARIOS(
{"matricula": matricula, "ano_sem": cmd[6]})
num_ok = MARCAS.count_documents(
{"matricula": matricula, "tipo": "ok", "valor": True, "ano_sem": cmd[6]})
num_alerta = MARCAS.count_documents(
{"matricula": matricula, "tipo": "alerta", "valor": True, "ano_sem": cmd[6]})
num_participa = MARCAS.count_documents(
{"matricula": matricula, "tipo": "participa", "valor": True, "ano_sem": cmd[6]})
num_comentarios = COMENTARIOS.count_documents(
{"matricula": matricula, "ano_sem": cmd[6]})
num_matriculas = MATRICULAS.count_documents(
{"matricula": matricula, "ano_sem": cmd[6]})
if num_matriculas == 0:
num_matriculas = 1
num_ok_m = str(num_ok) + "<small><i> (" + \
str(int(100*num_ok / num_matriculas))+"%)</i></small>"
num_alerta_m = str(num_alerta) + "<small><i> (" + \
str(int(100*num_alerta / num_matriculas))+"%)</i></small>"
num_participa_m = str(num_participa) + "<small><i> (" + \
str(int(100*num_participa / num_matriculas))+"%)</i></small>"
num_comentarios_m = str(num_comentarios) + "<small><i> (" + \
str(int(100*num_comentarios / num_matriculas))+"%)</i></small>"
j = {"cmd": cmd, "nome": nome, "foto": foto, "telefones": telefones, "emails": emails, "comentarios": comentarios,
"num_ok": num_ok_m, "num_alerta": num_alerta_m, "num_participa": num_participa_m, "num_comentarios": num_comentarios_m}
else:
professor = session["nome"]
foto = ""
# 0 1 2 3 4
# api("show_student"+"|"+matricula+"|"+curso+"|"+semestre+"|"+turno);
nome = cmd[2] + " - " + cmd[3] + " - "+cmd[4] + " - "+cmd[5]
curso, semestre, turno, turma = cmd[2], cmd[3], cmd[4], cmd[5]
foto = "static/class.png"
telefones = ""
emails = ""
comentarios = LISTA_COMENTARIOS(
{"curso": curso, "semestre": semestre, "turno": turno, "turma": turma, "ano_sem": cmd[6]})
num_ok = MARCAS.count_documents({"curso": curso, "semestre": semestre, "turno": turno,
"turma": turma, "tipo": "ok", "valor": True, "ano_sem": cmd[6]})
num_alerta = MARCAS.count_documents(
{"curso": curso, "semestre": semestre, "turno": turno, "turma": turma, "tipo": "alerta", "valor": True, "ano_sem": cmd[6]})
num_participa = MARCAS.count_documents(
{"curso": curso, "semestre": semestre, "turno": turno, "turma": turma, "tipo": "participa", "valor": True, "ano_sem": cmd[6]})
num_comentarios = COMENTARIOS.count_documents(
{"curso": curso, "semestre": semestre, "turno": turno, "turma": turma, "ano_sem": cmd[6]})
num_matriculas = MATRICULAS.count_documents(
{"curso": curso, "semestre": semestre, "turno": turno, "turma": turma, "ano_sem": cmd[6]})
if num_matriculas == 0:
num_matriculas = 1
num_ok_m = str(num_ok) + "<small><i> (" + \
str(int(100*num_ok / num_matriculas))+"%)</i></small>"
num_alerta_m = str(num_alerta) + "<small><i> (" + \
str(int(100*num_alerta / num_matriculas))+"%)</i></small>"
num_participa_m = str(num_participa) + "<small><i> (" + \
str(int(100*num_participa / num_matriculas))+"%)</i></small>"
num_comentarios_m = str(num_comentarios) + "<small><i> (" + \
str(int(100*num_comentarios / num_matriculas))+"%)</i></small>"
j = {"cmd": cmd, "nome": nome, "foto": foto, "telefones": telefones, "emails": emails, "comentarios": comentarios,
"num_ok": num_ok_m, "num_alerta": num_alerta_m, "num_participa": num_participa_m, "num_comentarios": num_comentarios_m}
return j
def LISTA_COMENTARIOS(filtro):
r = []
r.append("<table class='table_coment' width='100%' border='0' cellpadding ='' cellspacing ='0' style=' border-color:#ffffff; border-spacing:0'><tr><td></td><td width='90%'><B></B></td></tr>")
sprof = session["nome"]
for c in COMENTARIOS.find(filtro).sort("data", -1):
professor = NOME_SIMPLIFICADO(c["professor"])
if len(c["disciplina"]) > 15:
disciplina = c["disciplina"][0:15] + "..." + c["disciplina"][-1]
else:
disciplina = c["disciplina"]
dis = NOME_SIMPLIFICADO(c["professor"])
data = c["data"]
comando = ""
if "matricula" in filtro:
comentario = "<br><b>" + \
c["comentario"] + \
"</b><br>" f"<small><i>{professor}, {disciplina}<br>{data} </i> </small>"
else:
nome = "GERAL"
if c["matricula"] != "":
for m in MATRICULAS.find({"matricula": c["matricula"]}).limit(1):
nome = NOME_SIMPLIFICADO(m["nome"])
comentario = "<br><b>" + c["comentario"] + "</b><br>" + \
f"<small><i> {professor} para {nome}<br>{data} </i> </small>"
if c["professor"] == sprof:
apagar = "<img src = 'static/delete1.png' style='height:2vh' onclick=\"apagar_comentario('" + str(
c.get('_id')) + "')\">"
else:
apagar = ""
r.append(
f"<tr><td style='text-align:center;'>{apagar}</td><td width='80%'>{comentario}</td></tr>")
r.append("</table><table>")
filtro["valor"] = True
if "matricula" in filtro:
for c in MARCAS.find(filtro).sort("tipo", 1):
professor = NOME_SIMPLIFICADO(c["professor"])
if len(c["disciplina"]) > 15:
disciplina = c["disciplina"][0:15] + \
"..." + c["disciplina"][-1]
else:
disciplina = c["disciplina"]
dis = NOME_SIMPLIFICADO(c["professor"])
data = c["ano_sem"]
comando = ""
imagem = "<img src = 'static/" + \
c["tipo"]+".png' style='height:2vh'>"
comentario = "<br><b>" + imagem + \
"</b><br>" f"<small><i>{professor}, {disciplina}<br>{data} </i> </small>"
r.append(
f"<tr><td>&nbsp&nbsp&nbsp&nbsp&nbsp&nbsp&nbsp&nbsp&nbsp&nbsp&nbsp</td><td></td><td width='80%' style='text-align:right;'>{comentario}</td></tr>")
r.append("</table>")
return ''.join(r)
def SEND_NEW_COMENT(cmd):
professor = session["nome"]
matricula = cmd[1]
if matricula != "":
comentario = cmd[2]
data = datetime.now().strftime("%m/%d/%Y, %H:%M:%S")
# api("send_new_comment"+"|"+matricula+"|"+comentario+"|"+disciplina+"|"+curso+"|"+semestre+"|"+turno+"|"+turma);
filtro = {"data": data, "matricula": cmd[1], "disciplina": cmd[3], "curso": cmd[4], "semestre": cmd[5],
"turno": cmd[6], "turma": cmd[7], "ano_sem": cmd[8], "professor": professor, "comentario": comentario}
filtro2 = {"matricula": cmd[1], "ano_sem": cmd[8]}
COMENTARIOS.insert_one(filtro)
num_comentarios = COMENTARIOS.count_documents(filtro2)
else:
comentario = cmd[2]
data = datetime.now().strftime("%m/%d/%Y, %H:%M:%S")
filtro = {"data": data, "matricula": "", "disciplina": cmd[3], "curso": cmd[4], "semestre": cmd[5],
"turno": cmd[6], "turma": cmd[7], "professor": professor, "comentario": comentario, "ano_sem": cmd[8]}
filtro2 = {"curso": cmd[4], "semestre": cmd[5],
"turno": cmd[6], "ano_sem": cmd[8]}
COMENTARIOS.insert_one(filtro)
num_comentarios = COMENTARIOS.count_documents(filtro2)
return {"cmd": cmd, "comentarios": LISTA_COMENTARIOS(filtro2), "num": num_comentarios}
def APAGAR_COMENTARIO(cmd):
COMENTARIOS.delete_one({"_id": ObjectId(cmd[2])})
matricula = cmd[1]
if matricula != "":
filtro2 = {"matricula": cmd[1], "ano_sem": cmd[7]}
else:
filtro2 = {"curso": cmd[3], "semestre": cmd[4],
"turno": cmd[5], "turma": cmd[6], "ano_sem": cmd[7]}
num_comentarios = COMENTARIOS.count_documents(filtro2)
return {"cmd": cmd, "comentarios": LISTA_COMENTARIOS(filtro2), "num": num_comentarios}
def RELATORIO(curso, semestre, turno, turma, ano_sem):
r = []
# ['2014', [OK], [ALERTA], [PARTICIPA]]
num_ok = MARCAS.count_documents({"curso": curso, "semestre": semestre, "turno": turno,
"turma": turma, "tipo": "ok", "valor": True, "ano_sem": ano_sem})
num_alerta = MARCAS.count_documents({"curso": curso, "semestre": semestre, "turno": turno,
"turma": turma, "tipo": "alerta", "valor": True, "ano_sem": ano_sem})
num_participa = MARCAS.count_documents(
{"curso": curso, "semestre": semestre, "turno": turno, "turma": turma, "tipo": "participa", "valor": True, "ano_sem": ano_sem})
num_comentarios = COMENTARIOS.count_documents(
{"curso": curso, "semestre": semestre, "turno": turno, "turma": turma, "ano_sem": ano_sem})
num_matriculas = MATRICULAS.count_documents(
{"curso": curso, "semestre": semestre, "turno": turno, "turma": turma, "ano_sem": ano_sem})
if num_matriculas == 0:
num_matriculas = 1
num_ok_m = str(int(100*num_ok / num_matriculas))
num_alerta_m = str(int(100*num_alerta / num_matriculas))
num_participa_m = str(int(100*num_participa / num_matriculas))
num_comentarios_m = str(int(100*num_comentarios / num_matriculas))
TITULO = f"{curso} - {semestre} - {turno} - {turma}"
CONTAGENS = f"OK: {num_ok} ({num_ok_m}%) , ALERTA: {num_alerta} ({num_alerta_m}%), PARTICIPA: {num_participa} ({num_participa_m}%), COMENTARIOS: {num_comentarios} ({num_comentarios_m}%)"
arquivo = open("static/relatorio.html", "r", encoding='UTF8').read()
arquivo = arquivo.replace("[CONTAGENS]", CONTAGENS)
arquivo = arquivo.replace("[TITULO]", TITULO)
arquivo = arquivo.replace("[OK]", str(num_ok_m))
arquivo = arquivo.replace("[ALERTA]", str(num_alerta_m))
arquivo = arquivo.replace("[PARTICIPA]", str(num_participa_m))
r.append(arquivo)
return ''.join(r)
# ----------------------------------------------------------------------------------------------
@app.route('/api', methods=["POST", "GET"])
def api():
if "token" in session:
if "cmd" in request.json:
cmd = request.json["cmd"].split("|")
print(session["email"], ">", cmd)
if cmd[0] == "carregar":
return CARREGAR(cmd)
elif cmd[0] == "lista_disciplinas":
return LISTA_DISCIPLINAS(cmd)
elif cmd[0] == "marcar":
return MUDA_MARCACAO(cmd)
elif cmd[0] == "show_student":
return SHOW_STUDENT(cmd)
elif cmd[0] == "send_new_comment":
return SEND_NEW_COMENT(cmd)
elif cmd[0] == "apagar_comentario":
return APAGAR_COMENTARIO(cmd)
return {"cmd": ""}
# ----------------------------------------------------------------------------------------------
@app.route('/relatorio', methods=["POST", "GET"])
def relatorio():
if "ano_sem" in request.args:
if "curso" in request.args:
if "semestre" in request.args:
if "turno" in request.args:
if "turma" in request.args:
return RELATORIO(request.args["curso"], request.args["semestre"], request.args["turno"], request.args["turma"], request.args["ano_sem"])
return "faltam argumentos"
@app.route('/', methods=["POST", "GET"])
def home():
if "token" in request.args:
professor_encontrado = False
token = request.args["token"]
for professor in PROFESSORES.find({"token": token}):
professor_encontrado = True
session.permanent = True
session["email"] = professor["email"]
session["nome"] = professor["nome"]
session["token"] = professor["token"]
session.modified = True
return redirect(url_for("home"))
return MSG("usuário não encontrado")
if "token" in session:
return TELA_PRINCIAL()
else:
return MSG("Desconectado...\n\nFaça um novo acesso pelo link enviado para o seu e-mail.")
# PROFESSORES.update_one({"email":"robson.verly@ifms.edu.br"},{"$set":{"email":"robson.verly@ifms.edu.br","nome":"Robson Jaques Verly","token":"123"}},upsert = True)
# CURSOS.update_one({"matriz":"21"},{"$set":{"matriz":"21","nome":"Curso Técnico em Informática"}},upsert = True)
# ----------------------------------------------------------------------------------------------
if __name__ == "__main__":
if APP_RUN_DEBUG_MODE:
app.run(debug=True, host=HOST_ADDRESS, port=APP_PORT)
else:
serve(app, host=HOST_ADDRESS, port=APP_PORT, threads=SERVER_THREADS,
backlog=MAXIMUM_QUEUE_SIZE, channel_timeout=SERVER_TIMEOUT)