All Downloads are FREE. Search and download functionalities are using the official Maven repository.

de.svws_nrw.davapi.data.repos.bycategory.SchuelerWithCategoriesRepository Maven / Gradle / Ivy

Go to download

Diese Bibliothek enthält die Java-Server-Definition der CalDAV und CardDAV-Schnittstelle für die Schulverwaltungssoftware in NRW

The newest version!
package de.svws_nrw.davapi.data.repos.bycategory;

import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.ArrayList;
import java.util.function.Function;
import java.util.stream.Collectors;

import de.svws_nrw.core.data.adressbuch.AdressbuchEintrag;
import de.svws_nrw.core.data.adressbuch.Telefonnummer;
import de.svws_nrw.core.types.SchuelerStatus;
import de.svws_nrw.core.types.benutzer.BenutzerKompetenz;
import de.svws_nrw.davapi.data.CollectionRessourceQueryParameters;
import de.svws_nrw.davapi.data.IAdressbuchKontaktRepository;
import de.svws_nrw.db.Benutzer;
import de.svws_nrw.db.DBEntityManager;
import de.svws_nrw.db.dto.current.schild.erzieher.DTOSchuelerTelefon;
import de.svws_nrw.db.dto.current.schild.erzieher.DTOTelefonArt;
import de.svws_nrw.db.dto.current.schild.katalog.DTOOrt;
import de.svws_nrw.db.dto.current.schild.klassen.DTOKlassen;
import de.svws_nrw.db.dto.current.schild.kurse.DTOKurs;
import de.svws_nrw.db.dto.current.schild.kurse.DTOKursSchueler;
import de.svws_nrw.db.dto.current.schild.schueler.DTOSchueler;
import de.svws_nrw.db.dto.current.schild.schueler.DTOSchuelerLernabschnittsdaten;
import de.svws_nrw.db.dto.current.schild.schule.DTOJahrgang;
import de.svws_nrw.db.dto.current.schild.schule.DTOSchuljahresabschnitte;

/**
 * Diese Implementierung des {@link IAdressbuchKontaktRepository} dient der
 * Suche nach Schuelern aus der Datenbank als {@link AdressbuchEintrag}, welche
 * mit Kategorien ergänzt werden.
 *
 */
public final class SchuelerWithCategoriesRepository implements IAdressbuchKontaktRepository {

	/**
	 * die Datenbankverbindung
	 */
	private final DBEntityManager conn;
	/**
	 * der aktuelle Schuljahresabschnitt
	 */
	private final DTOSchuljahresabschnitte aktuellerSchuljahresabschnitt;
	/**
	 * Utility zum Erzeugen von Adressbuchkategorien
	 */
	private final AdressbuchKategorienUtil kategorienUtil;
	/**
	 * Name der eigenen Schule
	 */
	private final String schulName;
	/** der Benutzer, dessen Adressbuecher gesucht werden */
	private final Benutzer user;

	/**
	 * Konstruktor zum Erstellen des Repositories mit einer Datenbankverbindung
	 *
	 * @param conn                          die Datenbankverbindung
	 * @param user                          der Benutzer, dessen Adressbuecher
	 *                                      gesucht werden
	 * @param kategorienUtil                Instanz der Utility zum Erzeugen von
	 *                                      Adressbuchkategorien
	 * @param aktuellerSchuljahresabschnitt der aktuelle Schuljahresabschnitt für
	 *                                      weitere Queries
	 */
	public SchuelerWithCategoriesRepository(final DBEntityManager conn, final Benutzer user,
			final DTOSchuljahresabschnitte aktuellerSchuljahresabschnitt, final AdressbuchKategorienUtil kategorienUtil) {
		this.conn = conn;
		this.user = user;
		this.aktuellerSchuljahresabschnitt = aktuellerSchuljahresabschnitt;
		this.kategorienUtil = kategorienUtil;
		this.schulName = IAdressbuchKontaktRepository.getSchulname(conn);
	}

	@Override
	public List getKontakteByAdressbuch(final String adressbuchId,
			final CollectionRessourceQueryParameters params) {
		if (!params.includeRessources || !user.pruefeKompetenz(BenutzerKompetenz.SCHUELER_INDIVIDUALDATEN_ANSEHEN)) {
			return new ArrayList<>();
		}
		final List filteredDtoSchuelers = conn.queryAll(DTOSchueler.class).stream().filter(SCHUELER_FILTER).toList();

		if (params.includeEintragIDs && !params.includeEintragPayload) {
			return filteredDtoSchuelers.stream().map(s -> {
				final AdressbuchEintrag a = new AdressbuchEintrag();
				a.id = IAdressbuchKontaktRepository.createSchuelerId(s.ID);
				return a;
			}).toList();
		}

		final Set ortIds = filteredDtoSchuelers.stream().map(s -> s.Ort_ID).collect(Collectors.toSet());
		final List schuelerIds = filteredDtoSchuelers.stream().map(s -> s.ID).toList();
		final Map ortByOrtID = IAdressbuchKontaktRepository.queryOrteByOrtIds(ortIds, conn);
		final Map> telefonnummerBySchuelerId = queryTelefonNummernBySchuelerIds(schuelerIds, conn);

		final Map schuelerStatusById = filteredDtoSchuelers.stream()
				.collect(Collectors.toMap(s -> s.ID, s -> s.Status));
		final Map> categoriesBySchuelerId = getCategoriesBySchuelerId(schuelerStatusById);
		return filteredDtoSchuelers.stream()
				.map(s -> IAdressbuchKontaktRepository.mapDTOSchuelerToKontakt(s, telefonnummerBySchuelerId.get(s.ID),
						ortByOrtID.get(s.Ort_ID), categoriesBySchuelerId.get(s.ID), schulName))
				.toList();
	}

	/**
	 * Sucht und erzeugt eine Map, in der jeder gegebenen {@link DTOSchueler#ID}
	 * über {@link DTOSchuelerTelefon} eine Liste von Telefonnummern zugeordnet ist.
	 * Schueler, denen über {@link DTOSchuelerTelefon} keine Telefonnummern
	 * zugeordnet sind, tauchen in der Map nicht auf.
	 *
	 * @param schuelerIds ids der Schueler, für die nach Telefonnummern gesucht
	 *                    werden soll
	 * @param conn        die Datenbankverbindung
	 * @return eine Map, in der jeder Schueler ID eine Liste von Telefonnummern
	 *         zugeordnet ist
	 */
	static Map> queryTelefonNummernBySchuelerIds(final Collection schuelerIds,
			final DBEntityManager conn) {
		final List dtoSchuelerTelefonQueryResult = conn.queryList(DTOSchuelerTelefon.QUERY_LIST_BY_SCHUELER_ID,
				DTOSchuelerTelefon.class, schuelerIds);
		final List dtoTelefonArtQueryResults = conn.queryAll(DTOTelefonArt.class);
		final Map telefonArtById = dtoTelefonArtQueryResults.stream()
				.collect(Collectors.toMap(ta -> ta.ID, Function.identity()));
		final Map> telefonnummerBySchuelerId = new HashMap<>();

		for (final DTOSchuelerTelefon dto : dtoSchuelerTelefonQueryResult) {
			final DTOTelefonArt dtoTelefonArt = telefonArtById.get(dto.TelefonArt_ID);
			if (dto.Gesperrt.booleanValue() || (dto.Telefonnummer == null) || !dtoTelefonArt.Sichtbar.booleanValue()) {
				continue;
			}
			List telefonnummern = telefonnummerBySchuelerId.get(dto.Schueler_ID);

			if (telefonnummern == null) {
				telefonnummern = new ArrayList<>();
				telefonnummerBySchuelerId.put(dto.Schueler_ID, telefonnummern);
			}
			final Telefonnummer tel = new Telefonnummer();
			tel.number = dto.Telefonnummer;
			tel.type = dtoTelefonArt.Bezeichnung;
			telefonnummern.add(tel);
		}
		return telefonnummerBySchuelerId;
	}

	/**
	 * Hilfsmethode für die Suche aller Kategorien einer SchülerId
	 *
	 * @param schuelerStatusByID
	 *
	 * @return eine Map ,in der jeder SchülerId die entsprechenden Kategorien
	 *         zugeordnet sind
	 */
	private Map> getCategoriesBySchuelerId(final Map schuelerStatusByID) {
		final List dtoKlassenQueryResult = conn.queryList(DTOKlassen.QUERY_BY_SCHULJAHRESABSCHNITTS_ID, DTOKlassen.class,
				aktuellerSchuljahresabschnitt.ID);
		final Map klassenById = dtoKlassenQueryResult.stream()
				.collect(Collectors.toMap(s -> s.ID, Function.identity()));
		final List dtoSchuelerLernabschnittsdatenQueryResult = conn.queryList(
				DTOSchuelerLernabschnittsdaten.QUERY_LIST_BY_KLASSEN_ID, DTOSchuelerLernabschnittsdaten.class, klassenById.keySet());
		final Map jahrgangKrzByJahrgangId = conn.queryList(DTOJahrgang.QUERY_BY_SICHTBAR, DTOJahrgang.class, true)
				.stream().collect(Collectors.toMap(j -> j.ID, j -> j.InternKrz));

		final Map> result = new HashMap<>();
		for (final Entry entry : schuelerStatusByID.entrySet()) {
			if (entry.getValue() == SchuelerStatus.NEUAUFNAHME) {
				final Set listForSchuelerId = result.computeIfAbsent(entry.getKey(), s -> new HashSet<>());
				listForSchuelerId.add(kategorienUtil.formatSchuelerNeuaufnahmenAlle());
			}
		}
		for (final DTOSchuelerLernabschnittsdaten dtoSLA : dtoSchuelerLernabschnittsdatenQueryResult) {
			final Set listForSchuelerId = result.computeIfAbsent(dtoSLA.Schueler_ID, s -> new HashSet<>());
			if (klassenById.containsKey(dtoSLA.Klassen_ID)) {
				final DTOKlassen dtoKlassen = klassenById.get(dtoSLA.Klassen_ID);
				if (schuelerStatusByID.get(dtoSLA.Schueler_ID) == SchuelerStatus.NEUAUFNAHME) {
					listForSchuelerId.add(kategorienUtil.formatSchuelerNeuaufnahmeKlasse(dtoKlassen.Klasse));
					listForSchuelerId.add(kategorienUtil
							.formatSchuelerNeuaufnahmeJahrgang(jahrgangKrzByJahrgangId.get(dtoKlassen.Jahrgang_ID)));
				}
				listForSchuelerId.add(kategorienUtil.formatSchuelerKlasse(dtoKlassen.Klasse));
				listForSchuelerId.add(
						kategorienUtil.formatSchuelerJahrgang(jahrgangKrzByJahrgangId.get(dtoKlassen.Jahrgang_ID)));
			}
		}
		final List dtoKursResult = conn.queryList(DTOKurs.QUERY_BY_SCHULJAHRESABSCHNITTS_ID, DTOKurs.class, aktuellerSchuljahresabschnitt.ID);
		final Map kursNameById = dtoKursResult.stream().collect(Collectors.toMap(k -> k.ID, k -> k));
		// TODO kurse sind nur in einem Jahrgang eindeutig, Kursbezeichnung um
		// Jahrgangsbezeichnung anreichnern
		// 'Kurs' + [Jahrgang bei eindeutigkeit] + 'Kursbezeichnung' + [(Jahrgänge bei
		// nicht-Eindeutigkeit)]
		final List dtoKursSchuelerQueryResult =
				conn.queryList("SELECT e FROM DTOKursSchueler e WHERE e.Kurs_ID IN ?1 AND e.LernabschnittWechselNr = 0", DTOKursSchueler.class,
						kursNameById.keySet());
		for (final DTOKursSchueler dtoKursSchueler : dtoKursSchuelerQueryResult) {
			final Set listForSchuelerId = result.computeIfAbsent(dtoKursSchueler.Schueler_ID, s -> new HashSet<>());
			if (kursNameById.containsKey(dtoKursSchueler.Kurs_ID)) {
				final DTOKurs dtoKurs = kursNameById.get(dtoKursSchueler.Kurs_ID);
				listForSchuelerId.add(kategorienUtil.formatSchuelerKurs(dtoKurs.KurzBez,
						jahrgangKrzByJahrgangId.get(dtoKurs.Jahrgang_ID)));
			}
		}
		return result;
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy