forked from ScoDoc/ScoDoc
205 lines
6.6 KiB
Python
Executable File
205 lines
6.6 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
"""Simple fake HTTP serveur
|
|
emulating "Apogee" Web service
|
|
|
|
Usage:
|
|
/opt/scodoc/tools/fakeportal/fakeportal.py
|
|
|
|
et régler "URL du portail" sur la page de *Paramétrage* du département testé,
|
|
typiquement: http://localhost:8678
|
|
et "Version de l'API" à 2
|
|
|
|
"""
|
|
from pathlib import Path
|
|
from urllib.parse import parse_qs
|
|
from urllib.parse import urlparse
|
|
import http.server
|
|
import os
|
|
import random
|
|
import signal
|
|
import socketserver
|
|
import sys
|
|
import time
|
|
|
|
from gen_nomprenoms import nomprenom
|
|
|
|
script_dir = Path(os.path.abspath(__file__)).parent
|
|
os.chdir(script_dir)
|
|
|
|
# Les "photos" des étudiants
|
|
if os.path.exists("/opt/ExtraFaces"):
|
|
FAKE_FACES_PATHS = list((Path("extra_faces").glob("*/*.jpg")))
|
|
FAKE_FACES_PATHS_BY_CIVILITE = {
|
|
"M": list((Path("extra_faces").glob("M/*.jpg"))),
|
|
"F": list((Path("extra_faces").glob("F/*.jpg"))),
|
|
"X": list((Path("extra_faces").glob("X/*.jpg"))),
|
|
}
|
|
else:
|
|
FAKE_FACES_PATHS = list((Path("faces").glob("*.jpg")))
|
|
FAKE_FACES_PATHS_BY_CIVILITE = {
|
|
"M": FAKE_FACES_PATHS,
|
|
"F": FAKE_FACES_PATHS,
|
|
"X": FAKE_FACES_PATHS,
|
|
}
|
|
|
|
# Etudiant avec tous les champs (USPN)
|
|
ETUD_TEMPLATE_FULL = open(script_dir / "etud_template.xml", encoding="UTF-8").read()
|
|
# Etudiant avec seulement les champs requis
|
|
ETUD_TEMPLATE_MINI = open(
|
|
script_dir / "etud_minimal_template.xml", encoding="UTF-8"
|
|
).read()
|
|
|
|
ETUD_HEAD = """<?xml version="1.0" encoding="UTF-8"?>
|
|
<etudiants>"""
|
|
ETUD_TAIL = """</etudiants>
|
|
"""
|
|
|
|
CODES_ETAPES = ("V1RT", "V2RT", "V2RT2", "")
|
|
|
|
|
|
def make_random_etud(nip, etape=None, annee=None, template=ETUD_TEMPLATE_FULL):
|
|
"""return XML for a student"""
|
|
random.seed(nip) # deterministic choice based on nip
|
|
gender = random.choice(("M", "F"))
|
|
nom, prenom = nomprenom(gender)
|
|
if not etape:
|
|
etape = random.choice(CODES_ETAPES)
|
|
if not annee:
|
|
annee = time.strftime("%Y") # current year
|
|
diplome = "VDRT"
|
|
data = template.format(
|
|
nip=nip,
|
|
ine=str(random.randint(1000000000, 9999999999)) + "Z",
|
|
gender=gender,
|
|
nom=nom,
|
|
prenom=prenom,
|
|
etape=etape,
|
|
diplome=diplome,
|
|
annee=annee,
|
|
ville_naissance=random.choice(("Paris", "Berlin", "Londres", "")),
|
|
code_dep_naissance=random.choice(("75", "99", "89")),
|
|
libelle_dep_naissance="nom département",
|
|
paiementinscription=random.choice(("true", "false")),
|
|
datefinalisationinscription=random.choice(("30/10/2024", "1/10/66", "")),
|
|
)
|
|
return data
|
|
|
|
|
|
def make_random_etape_etuds(etape, annee):
|
|
"""Liste d'etudiants d'une etape"""
|
|
random.seed(etape + annee)
|
|
nb = random.randint(0, 50)
|
|
print(f"generating {nb} students")
|
|
L = []
|
|
for i in range(nb):
|
|
if i % 2:
|
|
template = ETUD_TEMPLATE_MINI
|
|
else:
|
|
template = ETUD_TEMPLATE_FULL
|
|
nip = str(random.randint(10000000, 99999999)) # 8 digits
|
|
L.append(make_random_etud(nip, etape=etape, annee=annee, template=template))
|
|
return "\n".join(L)
|
|
|
|
|
|
def get_photo_filename(nip: str, civilite: str | None = None) -> str:
|
|
"""get an existing filename for a fake photo, found in faces/
|
|
Returns a path relative to the current working dir
|
|
If civilite is not None, use it to select a subdir
|
|
"""
|
|
print("get_photo_filename")
|
|
if civilite:
|
|
faces = FAKE_FACES_PATHS_BY_CIVILITE[civilite]
|
|
else:
|
|
faces = FAKE_FACES_PATHS
|
|
nb_faces = len(faces)
|
|
if nb_faces == 0:
|
|
print("WARNING: aucun fichier image disponible !")
|
|
return ""
|
|
print(faces[hash(nip) % nb_faces])
|
|
return faces[hash(nip) % nb_faces]
|
|
|
|
|
|
class MyHttpRequestHandler(http.server.SimpleHTTPRequestHandler):
|
|
def send_xml(self, data):
|
|
self.send_response(200)
|
|
self.send_header("Content-type", "text/xml;charset=UTF-8")
|
|
self.end_headers()
|
|
self.wfile.write(bytes(data, "utf8"))
|
|
|
|
def do_GET(self):
|
|
query_components = parse_qs(urlparse(self.path).query)
|
|
print(f"path={self.path}", file=sys.stderr)
|
|
print(query_components, file=sys.stderr)
|
|
|
|
if "etapes" in self.path.lower():
|
|
self.path = str(Path(script_dir / "etapes.xml").relative_to(Path.cwd()))
|
|
elif "scodocEtudiant" in self.path: # API v2
|
|
# 2 forms: nip=xxx or etape=eee&annee=aaa
|
|
if "nip" in query_components:
|
|
nip = query_components["nip"][0]
|
|
print(f"requesting nip={nip}")
|
|
data = ETUD_HEAD + make_random_etud(nip) + ETUD_TAIL
|
|
return self.send_xml(data)
|
|
elif "etape" in query_components:
|
|
etape = query_components["etape"][0]
|
|
print(f"requesting etape={etape}", file=sys.stderr)
|
|
if "annee" in query_components:
|
|
annee = query_components["annee"][0]
|
|
data = ETUD_HEAD + make_random_etape_etuds(etape, annee) + ETUD_TAIL
|
|
return self.send_xml(data)
|
|
else:
|
|
print(
|
|
f"Error 404: (missing annee) path={self.path}", file=sys.stderr
|
|
)
|
|
self.send_response(404)
|
|
return
|
|
else:
|
|
print(
|
|
f"Error 404: (missing nip or etape) path={self.path}",
|
|
file=sys.stderr,
|
|
)
|
|
self.send_response(404)
|
|
return
|
|
elif ("getPhoto" in self.path) or ("scodocPhoto" in self.path):
|
|
nip = query_components["nip"][0]
|
|
civilite = query_components.get("civilite")
|
|
civilite = civilite[0] if civilite else None
|
|
self.path = str(get_photo_filename(nip, civilite=civilite))
|
|
print(f"photo for nip={nip}: {self.path}")
|
|
else:
|
|
print(f"Error 404: path={self.path}")
|
|
self.send_response(404)
|
|
return
|
|
|
|
# Sending an '200 OK' response
|
|
self.send_response(200)
|
|
http.server.SimpleHTTPRequestHandler.do_GET(self)
|
|
|
|
return
|
|
|
|
|
|
PORT = 8678
|
|
|
|
|
|
def signal_handler(sig, frame):
|
|
print("You pressed Ctrl+C!")
|
|
raise SystemExit()
|
|
|
|
|
|
signal.signal(signal.SIGINT, signal_handler)
|
|
|
|
if __name__ == "__main__":
|
|
# Help message
|
|
print(f"Les étapes (codes Apogée) sont: {CODES_ETAPES}")
|
|
print(f"Définir l'URL du portail comme: http://localhost:{PORT}")
|
|
print("""et "Version de l'API" à 2""")
|
|
# Start the server
|
|
print(f"Server listening on port {PORT}...")
|
|
my_server = socketserver.TCPServer(("", PORT), MyHttpRequestHandler)
|
|
try:
|
|
my_server.serve_forever()
|
|
finally:
|
|
print("shutting down...")
|
|
my_server.shutdown()
|