{"config":{"lang":["fr"],"separator":"[\\s\\-]+","pipeline":["stopWordFilter"]},"docs":[{"location":"","title":"Documentation du site de l'association des \u00e9tudiants de l'UTBM","text":"

Bonjour, camarade.

Si tu es ici, c'est sans doute pour mieux connaitre le fonctionnement interne du site AE. Peut-\u00eatre m\u00eame as-tu envie de contribuer \u00e0 son d\u00e9veloppement. Si tel est le cas, nous ne pouvons faire autrement que te souhaiter la bienvenue sur cette documentation et esp\u00e9rer que ce que nous avons \u00e9crit est assez clair et assez complet pour satisfaire ton esprit en qu\u00eate de connaissances.

Et si tu viens pour d'autres motifs, \u00e7a ne change rien, soit le bienvenu (ou la bienvenue) quand m\u00eame.

Pour que tu saches o\u00f9 chercher quelles informations, voici comment nous avons d\u00e9coup\u00e9 la documentation :

"},{"location":"explanation/","title":"Accueil","text":""},{"location":"explanation/#objectifs","title":"Objectifs","text":"

Le but de ce projet est de fournir \u00e0 l'Association des \u00c9tudiants de l'UTBM une plate-forme pratique et centralis\u00e9e de ses services. Le Sith de l'AE tient \u00e0 jour le registre des cotisations \u00e0 l'association, prend en charge la tr\u00e9sorerie, les ventes de produits et services, la diffusion d\u2019\u00e9v\u00e9nements, la gestion de la laverie et bien plus encore.

C'est un projet b\u00e9n\u00e9vole qui tire ses origines des ann\u00e9es 2000. Il s'agit de la troisi\u00e8me version du site de l'AE. Son d\u00e9veloppement a commenc\u00e9 en 2015. C'est une r\u00e9\u00e9criture compl\u00e8te en rupture totale des deux versions qui l'ont pr\u00e9c\u00e9d\u00e9e.

"},{"location":"explanation/#pourquoi-reecrire-le-site","title":"Pourquoi r\u00e9\u00e9crire le site","text":"

L'ancienne version du site, sobrement baptis\u00e9e ae2, pr\u00e9sentait un nombre impressionnant de fonctionnalit\u00e9s. Il avait \u00e9t\u00e9 \u00e9crit en PHP et se basait sur son propre framework maison.

Malheureusement, son entretien \u00e9tait plus ou moins hasardeux et son framework reposait sur des principes assez diff\u00e9rents de ce qui se fait aujourd'hui, rendant la maintenance difficile. De plus, la version de PHP qu'il utilisait \u00e9tait plus que d\u00e9pr\u00e9ci\u00e9e et \u00e0 l'heure de l'arriv\u00e9e de PHP 7 et de sa non-r\u00e9trocompatibilit\u00e9 il \u00e9tait vital de faire quelque chose. Il a donc \u00e9t\u00e9 d\u00e9cid\u00e9 de le r\u00e9\u00e9crire.

"},{"location":"explanation/#la-philosophie-initiale","title":"La philosophie initiale","text":"

Pour \u00e9viter les erreurs du pass\u00e9, ce projet met l'accent sur la maintenabilit\u00e9. Le choix des technologies ne s'est donc pas fait uniquement sur le fait qu'elle soit r\u00e9centes, mais \u00e9galement sur leur robustesse, leur fiabilit\u00e9 et leur potentiel \u00e0 \u00eatre maintenu loin dans le futur.

La maintenabilit\u00e9 passe \u00e9galement par le choix minutieux des d\u00e9pendances qui doivent, elles aussi, passer l'\u00e9preuve du temps pour \u00e9viter qu'elles ne mettent le projet en danger.

Cela passe \u00e9galement par la minimisation des frameworks employ\u00e9s de mani\u00e8re \u00e0 r\u00e9duire un maximum les connaissances n\u00e9cessaires pour contribuer au projet et donc simplifier la prise en main. La simplicit\u00e9 est \u00e0 privil\u00e9gier si elle est possible.

Le projet doit \u00eatre simple \u00e0 installer et \u00e0 d\u00e9ployer.

Le projet \u00e9tant \u00e0 destination d'\u00e9tudiants, il est pr\u00e9f\u00e9rable de minimiser les ressources utilis\u00e9es par l'utilisateur final. Il faut qu'il soit au maximum \u00e9conome en bande passante et calcul c\u00f4t\u00e9 client.

Le projet est un logiciel libre et est sous licence GPL. Aucune d\u00e9pendance propri\u00e9taire n'est accept\u00e9e.

"},{"location":"explanation/#la-philosophie-10-ans-plus-tard","title":"La philosophie, 10 ans plus tard","text":"

Malgr\u00e9 la bonne volont\u00e9 et le travail colossal fourni par les developpeurs de la version actuelle du projet, force est de constater que nombre d'erreurs ont malheureusement \u00e9t\u00e9 commises : usage complexe et excessif de certains m\u00e9canismes OO, r\u00e9\u00e9criture maison de fonctionnalit\u00e9s de Django, syst\u00e8me de gestion des permissions rigide et co\u00fbteux en requ\u00eate \u00e0 la base de donn\u00e9es...

Mais malgr\u00e9 tout \u00e7a, le site tourne. En effet, force est de constater que le pari initial de choisir un framework stable et durable a pay\u00e9. Aujourd'hui encore, Django est activement maintenu, ses mises \u00e0 jour sont r\u00e9guli\u00e8res sans pour autant n\u00e9cessiter beaucoup de changements lors des changements de version majeure.

Quant aux erreurs qui ont \u00e9t\u00e9 commises, que celui qui n'a jamais reconsid\u00e9r\u00e9 a posteriori que ce qui lui semblait une bonne architecture \u00e9tait en fait un ensemble brinquebalant, leur jette la premi\u00e8re pierre.

La solidit\u00e9 des fondations ayant \u00e9t\u00e9 prouv\u00e9e par l'\u00e9preuve du temps, le travail restant \u00e0 accomplir n'est pas de r\u00e9\u00e9crire encore une fois le site en utilisant encore d'autres technologies, mais plut\u00f4t de raboter les surcouches du site, pour refixer le plus solidement possiblement le projet sur ces fondations.

"},{"location":"explanation/archives/","title":"Archives","text":"

Page g\u00e9n\u00e9r\u00e9e

Cette page est g\u00e9n\u00e9r\u00e9e \u00e0 partir du wiki GitHub du projet puisqu'elle h\u00e9berge du contenu binaire lourd qui encombrerait le repo git. La page originale ici peut \u00eatre plus \u00e0 jour puisque le contenu de cette page date de la derni\u00e8re g\u00e9n\u00e9ration de la documentation.

Cette page contient certains des pdf qui ont pu \u00eatre produits dans le cadre du travail pass\u00e9 sur le site. Vous ne trouverez pas des informations compl\u00e8tes et \u00e0 jour. En revanche, vous trouverez des morceaux d'histoire qui vous permettront de mieux comprendre ceux \u00e0 quoi pensaient les anciens d\u00e9veloppeurs et, par extension, vous comprendrez peut-\u00eatre un peu mieux le site.

"},{"location":"explanation/archives/#comptes-rendus","title":"Comptes-rendus","text":""},{"location":"explanation/archives/#rapports-de-twto","title":"Rapports de TW/TO","text":""},{"location":"explanation/archives/#skia","title":"Skia","text":"

Rapport Skia

"},{"location":"explanation/archives/#skia-et-loj","title":"Skia et LoJ","text":"

Rapport Skia+LoJ

"},{"location":"explanation/archives/#sli","title":"Sli","text":""},{"location":"explanation/conventions/","title":"Conventions","text":"

Cette page traite des conventions utilis\u00e9es dans le d\u00e9veloppement du site.

"},{"location":"explanation/conventions/#langue","title":"Langue","text":"

Les noms, de fonctions, de classe, de fichiers et de dossiers sont en anglais. De m\u00eame, les commentaires et les docstrings sont r\u00e9dig\u00e9s en anglais.

En revanche la documentation est r\u00e9dig\u00e9e en fran\u00e7ais. En effet, les d\u00e9veloppeurs et les d\u00e9veloppeuses qui ont \u00e9t\u00e9, sont et seront amen\u00e9s \u00e0 travailler sur le site sont presque tous des francophones. Or, la bonne compr\u00e9hension prime. Une documentation, qui se doit d'utiliser au mieux les mots justes, compris de mani\u00e8re juste, gagne \u00e0 \u00eatre \u00e9crite en langue vernaculaire, lorsqu'on est assur\u00e9 qu'une coop\u00e9ration internationale est peu probable.

De la sorte, on s'assure au mieux que les r\u00e9dacteurs et r\u00e9dactrices s'expriment bien et que, r\u00e9ciproquement, les lecteurs et lectrices comprennent au mieux.

A ce titre, on ne vous en voudra pas si vous r\u00e9digez des commentaires ou des docstrings en fran\u00e7ais.

En revanche, le code en lui-m\u00eame doit rester imp\u00e9rativement en anglais ; les instructions \u00e9tant en langue anglaise, introduire des mots fran\u00e7ais au milieu cr\u00e9e un contraste qui nuit \u00e0 la compr\u00e9hension.

De mani\u00e8re g\u00e9n\u00e9rale, demandez-vous juste \u00e0 qui vous \u00eates en train d'\u00e9crire :

"},{"location":"explanation/conventions/#gestion-de-version","title":"Gestion de version","text":"

Le projet utilise Git pour g\u00e9rer les versions et GitHub pour h\u00e9berger le d\u00e9p\u00f4t distant.

L'arbre poss\u00e8de deux branches prot\u00e9g\u00e9es : master et taiste.

master est la branche contenant le code tel qu'il tourne effectivement sur le vrai site de l'AE. Celle-ci doit, autant que faire se peut, rester impeccable.

taiste est la branche de r\u00e9f\u00e9rence pour le d\u00e9veloppement. Cette derni\u00e8re est r\u00e9guli\u00e8rement d\u00e9ploy\u00e9e sur le site de test. Elle permet de s'assurer que les diverses modifications fonctionnent bien entre elles et fonctionnent bien sur le serveur, avant d'\u00eatre envoy\u00e9es sur master.

"},{"location":"explanation/conventions/#gestion-des-branches","title":"Gestion des branches","text":"

Toutes les modifications appliqu\u00e9es sur taiste doivent se faire via des Pull Requests depuis les diff\u00e9rentes branches de d\u00e9veloppement. Toutes les modifications appliqu\u00e9es sur master doivent se faire via des Pull Requests depuis taiste, ou bien depuis une branche de hotfix, dans le cas o\u00f9 il faut r\u00e9parer un bug urgent apparu de mani\u00e8re impromptue.

Aucun push direct n'est admis, ni sur l'une, ni sur l'autre branche.

En obligeant \u00e0 passer par des PR, on s'assure qu'au moins une autre personne aura lu votre code et que les outils de test et de v\u00e9rification de code auront valid\u00e9 vos modifications.

Par extension du mode de travail par PR, les branches master et taiste ne peuvent recevoir du code que sous la forme de merge commits.

De plus, ces branches doivent recevoir, mais jamais donner (\u00e0 part entre elles). Lorsqu'une modification a \u00e9t\u00e9 effectu\u00e9e sur taiste et que vous souhaitez la r\u00e9cup\u00e9rer dans une de vos branches, vous devez proc\u00e9der par rebase, et non par merge.

En d'autres termes, vous devez respecter les deux r\u00e8gles suivantes :

  1. Les branches master et taiste ne doivent contenir que des merge commits
  2. Seules les branches master et taiste peuvent contenir des merge commits
Bien \u2714\ufe0fPas bien \u274c
gitGraph:\n    commit id: \"initial commit\"\n    branch bar\n    checkout main\n    checkout bar\n    commit id: \"baz\"\n    checkout main\n    merge bar id: \"Merge branch bar\"\n    branch foo\n    commit id: \"foo a\"\n    commit id: \"foo b\"\n    commit id: \"foo c\"\n    checkout main\n    merge foo id: \"Merge branch foo\"
gitGraph:\n    commit\n    branch bar\n    branch foo\n    commit id: \"foo a\"\n    commit id: \"foo b\"\n    checkout main\n    checkout bar\n    commit id: \"baz\"\n    checkout main\n    merge bar id: \"Merge branch bar\"\n    checkout foo\n    merge main id: \"Merge branch main\"\n    commit id: \"foo c\"\n    checkout main\n    merge foo id: \"Merge branch foo\"
"},{"location":"explanation/conventions/#style-de-code","title":"Style de code","text":""},{"location":"explanation/conventions/#conventions-de-nommage","title":"Conventions de nommage","text":"

Les conventions de nommage sont celles de la PEP8 :

En parall\u00e8le de la casse, les r\u00e8gles de formatage du code sont celles du formateur Ruff. Ces r\u00e8gles sont automatiquement appliqu\u00e9es quand vous faites tourner Ruff, donc vous n'avez pas \u00e0 trop vous poser de questions de ce c\u00f4t\u00e9-l\u00e0.

En ce qui concerne les templates Jinja et les fichiers SCSS, la norme de formatage est celle par d\u00e9faut de djHTML.

Pour Javascript, nous utilisons biome. C'est \u00e0 la fois un formateur et un linter avec tr\u00e8s peu de configuration, un peu comme ruff.

Le javascript dans les templates jinja

Biome n'est pas capable de lire dans les fichiers jinja, c'est sa principale limitation.

Il est donc recommand\u00e9 d'\u00e9viter de mettre trop de Javascript directement dans jinja mais de pr\u00e9f\u00e9rer des fichiers d\u00e9di\u00e9s.

"},{"location":"explanation/conventions/#qualite-du-code","title":"Qualit\u00e9 du code","text":"

Pour s'assurer de la qualit\u00e9 du code, Ruff et Biome sont \u00e9galement utilis\u00e9s.

Tout comme pour le format, Ruff et Biome doivent tourner avant chaque commit.

to edit or not to edit

Vous constaterez sans doute que ruff format modifie votre code, mais que ruff check vous signale juste une liste d'erreurs sans rien modifier.

En effet, ruff format ne s'occupe que de la forme du code, alors que ruff check regarde la logique du code. Si Ruff modifiait automatiquement la logique du code, \u00e7a serait un coup \u00e0 introduire plus de bugs que \u00e7a n'en r\u00e9soud.

Il existe cependant certaines cat\u00e9gories d'erreurs que Ruff peut r\u00e9parer de mani\u00e8re s\u00fbre. Pour appliquer ces r\u00e9parations, faites :

ruff check --fix\n

Biome se comporte d'une mani\u00e8re tr\u00e8s similaire

npx @biomejs/biome check # Liste toutes les erreurs et leurs cat\u00e9gories\nnpx @biomejs/biome check --write # Applique tous les fix consid\u00e9r\u00e9s safe et formate le code\n
"},{"location":"explanation/conventions/#documentation","title":"Documentation","text":"

La documentation est \u00e9crite en markdown, avec les fonctionnalit\u00e9s offertes par MkDocs, MkDocs-material et leurs extensions.

La documentation est int\u00e9gralement en fran\u00e7ais, \u00e0 l'exception des exemples, qui suivent les conventions donn\u00e9es plus haut.

"},{"location":"explanation/conventions/#decoupage","title":"D\u00e9coupage","text":"

La s\u00e9paration entre les diff\u00e9rentes parties de la documentation se fait en suivant la m\u00e9thodologie Diataxis. On compte quatre sections :

  1. Explications : parlez dans cette section de ce qui est bon \u00e0 savoir sans que \u00e7a touche aux d\u00e9tails pr\u00e9cis de l'impl\u00e9mentation. Si vous parlez de pourquoi un choix a \u00e9t\u00e9 fait ou que vous montrez grossi\u00e8rement les contours d'une partie du projet, c'est une explication.
  2. Tutoriels : parlez dans cette section d'\u00e9tapes pr\u00e9cises ou de d\u00e9tails d'impl\u00e9mentation qu'un nouveau d\u00e9veloppeur doit suivre pour commencer \u00e0 travailler sur le projet.
  3. Utilisation : parlez dans cette section de m\u00e9thodes utiles pour un d\u00e9veloppeur qui a d\u00e9j\u00e0 pris en main le projet. Voyez cette partie comme un livre de recettes de cuisine.
  4. R\u00e9f\u00e9rence : parlez dans cette section des d\u00e9tails d'impl\u00e9mentation du projet. En r\u00e9alit\u00e9, vous n'aurez pas besoin de beaucoup vous pencher dessus, puisque cette partie est compos\u00e9e presque uniquement des docstrings pr\u00e9sents dans le code.

Pour plus de d\u00e9tails, lisez directement la documentation de Diataxis, qui expose ces concepts de mani\u00e8re beaucoup plus compl\u00e8te.

"},{"location":"explanation/conventions/#style","title":"Style","text":"

Votre markdown doit \u00eatre compos\u00e9 de lignes courtes ; \u00e0 partir de 88 caract\u00e8res, c'est trop long. Si une phrase est trop longue pour tenir sur une ligne, vous pouvez l'\u00e9crire sur plusieurs.

Une ligne ne peut pas contenir plus d'une seule phrase. Dit autrement, quand vous finissez une phrase, faites syst\u00e9matiquement un saut de ligne.

Bien \u2714\ufe0fPas bien \u274c
First shalt thou take out the Holy Pin,\nthen shalt thou count to three, no more, no less.\nThree shalt be the number thou shalt count,\nand the number of the counting shalt be three.\nFour shalt thou not count, neither count thou two,\nexcepting that thou then proceed to three.\nFive is right out.\nOnce the number three, being the third number, be reached,\nthen lobbest thou thy Holy Hand Grenade of Antioch towards thou foe,\nwho being naughty in My sight, shall snuff it.\n
First shalt thou take out the Holy Pin, then shalt thou count to three, no more, no less. Three shalt be the number thou shalt count, and the number of the counting shalt be three. Four shalt thou not count, neither count thou two, excepting that thou then proceed to three. Five is right out. Once the number three, being the third number, be reached, then lobbest thou thy Holy Hand Grenade of Antioch towards thou foe, who being naughty in My sight, shall snuff it.\n

\u00c0 noter que ces deux exemples donnent le m\u00eame r\u00e9sultat dans la documentation g\u00e9n\u00e9r\u00e9e. Mais la version avec de courtes lignes est beaucoup plus facile \u00e0 modifier et \u00e0 versioner.

Grammaire et orthographe

Ca peut paraitre \u00e9vident dit comme \u00e7a, mais c'est toujours bon \u00e0 rappeler : \u00e9vitez de faire des fautes de fran\u00e7ais. Relisez vous quand vous avez fini d'\u00e9crire.

"},{"location":"explanation/conventions/#docstrings","title":"Docstrings","text":"

Les docstrings sont \u00e9crits en suivant la norme Google et les fonctionnalit\u00e9s de Griffe.

Ils doivent \u00eatre explicites sur ce que la fonction accomplit, mais ne pas parler de comment elle le fait. Un bon docstring est celui qui dit exactement ce qu'il faut pour qu'on puisse savoir comment utiliser la fonction ou la classe document\u00e9e sans avoir \u00e0 lire son code.

Tout comme les p\u00e9dales d'une voiture : pour pouvoir conduire, vous avez juste besoin de savoir ce qui se passe quand vous appuyez dessus. La connaissance de la m\u00e9canique interne est inutile dans ce cadre.

N'h\u00e9sitez pas \u00e0 mettre des examples dans vos docstrings.

"},{"location":"explanation/conventions/#pourquoi-une-partie-du-projet-ne-respecte-pas-ces-conventions","title":"Pourquoi une partie du projet ne respecte pas ces conventions ?","text":"

Parce que le projet est vieux. Le commit initial date du 18 novembre 2015. C'\u00e9tait il y a presque dix ans au moment o\u00f9 ces lignes sont \u00e9crites. Au d\u00e9but, on ne se posait pas forc\u00e9ment ce genre de questions. Puis le projet a grandi, de mani\u00e8re s\u00e9dimentaire, fonctionnalit\u00e9 apr\u00e8s fonctionnalit\u00e9, d\u00e9velopp\u00e9 par des personnes n'ayant pas toutes la m\u00eame esth\u00e9tique.

On retrouve dans le code ces inspirations diverses de personnes vari\u00e9es \u00e0 travers une d\u00e9cennie. Au bout d'un moment, il est bon de se poser et de normaliser les choses.

De ce c\u00f4t\u00e9-l\u00e0, une premi\u00e8re pierre a \u00e9t\u00e9 pos\u00e9e en novembre 2018, avec l'utilisation d'un formateur. Il convient de poursuivre ce travail d'unification.

Cependant, l\u00e0 o\u00f9 on peut reformater automatiquement du code, il faut y aller \u00e0 la main pour retravailler un style de code. C'est un travail de fourmi qui prendra du temps.

"},{"location":"explanation/technos/","title":"Technologies utilis\u00e9es","text":"

Bien choisir ses technologies est crucial puisqu'une fois que le projet est suffisamment avanc\u00e9, il est tr\u00e8s difficile voir impossible de revenir en arri\u00e8re.

En novembre 2015, plusieurs choix se pr\u00e9sentaient :

Le PHP 5, bient\u00f4t 7, de l'\u00e9poque \u00e9tant assez discutable comme cet article le montre, et l'ancien site ayant laiss\u00e9 un go\u00fbt amer \u00e0 certains d\u00e9veloppeurs, celui-ci a \u00e9t\u00e9 mis de c\u00f4t\u00e9.

L'\u00e9cosyst\u00e8me Javascript \u00e9tant \u00e0 peine naissant et les frameworks allant et venant en seulement quelques mois, il \u00e9tait impossible de pr\u00e9dire avec certitude si ceux-ci passeraient l'\u00e9preuve du temps, il \u00e9tait inconcevable de tout parier l\u00e0-dessus.

Ne restait plus que le Python et le Ruby avec les frameworks Django et Ruby On Rails. Ruby ayant une r\u00e9putation d'\u00eatre tr\u00e8s \"cutting edge\", c'est Python, un langage bien implant\u00e9 et ayant fait ses preuves, qui a \u00e9t\u00e9 retenu.

Il est \u00e0 noter que r\u00e9\u00e9crire le site avec un framework PHP comme Laravel ou Symphony eut aussi \u00e9t\u00e9 possible, ces deux technologies \u00e9tant assez matures et robustes au moment o\u00f9 le d\u00e9veloppement a commenc\u00e9. Cependant, il aurait \u00e9t\u00e9 potentiellement fastidieux de maintenir en parall\u00e8le deux versions de PHP sur le serveur durant toute la dur\u00e9e du d\u00e9veloppement. Il faut aussi prendre en compte que nous \u00e9tions \u00e0 ce moment d\u00e9go\u00fbt\u00e9s du PHP.

"},{"location":"explanation/technos/#backend","title":"Backend","text":""},{"location":"explanation/technos/#python-3","title":"Python 3","text":"

Site officiel

Le python est un langage de programmation interpr\u00e9t\u00e9 multi paradigme sorti en 1991. Il est tr\u00e8s populaire dans de nombreux domaines pour sa simplicit\u00e9 d'utilisation, sa polyvalence, sa s\u00e9curit\u00e9 ainsi que sa grande communaut\u00e9 de d\u00e9veloppeur. Sa version 3, non r\u00e9tro compatible avec sa version 2, a \u00e9t\u00e9 publi\u00e9e en 2008.

Note

Puisque toutes les d\u00e9pendances du backend sont des packages Python, elles sont toutes ajout\u00e9es directement dans le fichier pyproject.toml, \u00e0 la racine du projet.

"},{"location":"explanation/technos/#django","title":"Django","text":"

Site officiel

Documentation

Django est un framework web pour Python apparu en 2005. Il fournit un grand nombre de fonctionnalit\u00e9s pour d\u00e9velopper un site rapidement et simplement. Cela inclut entre autre un serveur Web de d\u00e9veloppement, un parseur d'URLs pour le routage, un ORM (Object-Relational Mapper) pour la gestion de la base de donn\u00e9e, un cadre pour l'\u00e9criture et l'ex\u00e9cution des tests, de nombreux utilitaires pour l'internationalisation, les zones horaires et autres, une interface web d'administration ais\u00e9ment configurable, un moteur de templates pour le rendu HTML...

Django propose une version LTS (Long Term Support) qui reste stable et est maintenu sur des cycles plus longs. Ce sont ces versions qui sont utilis\u00e9es.

"},{"location":"explanation/technos/#postgresql-sqlite3","title":"PostgreSQL / SQLite3","text":"

Site officiel PostgreSQL

Site officiel SQLite

Comme la majorit\u00e9 des sites internet, le Sith de l'AE enregistre ses donn\u00e9es dans une base de donn\u00e9es. Nous utilisons une base de donn\u00e9e relationnelle puisque c'est la mani\u00e8re typique d'utiliser Django et c'est ce qu'utilise son ORM.

Le principal \u00e0 retenir ici est :

PostgreSQL est-ce avec quoi le site doit fonctionner. Cependant, pour permettre aux d\u00e9veloppeurs de travailler en installant le moins de d\u00e9pendances possible sur leur ordinateur, il est \u00e9galement d\u00e9sirable de chercher la compatibilit\u00e9 avec SQLite. Aussi, dans la mesure du possible, nous cherchons \u00e0 ce que les interactions avec la base de donn\u00e9es fonctionnent avec l'un comme avec l'autre.

Heureusement, et gr\u00e2ce \u00e0 l'ORM de Django, cette double compatibilit\u00e9 est presque toujours possible.

"},{"location":"explanation/technos/#frontend","title":"Frontend","text":""},{"location":"explanation/technos/#jinja2","title":"Jinja2","text":"

Site officiel

Jinja2 est un moteur de template \u00e9crit en Python qui s'inspire fortement de la syntaxe des templates de Django. Ce moteur apporte toutefois son lot d'am\u00e9liorations non n\u00e9gligeables. Il permet par exemple l'ajout de macros, sortes de fonctions \u00e9crivant du HTML.

Un moteur de templates permet de g\u00e9n\u00e9rer du contenu textuel de mani\u00e8re proc\u00e9dural en fonction des donn\u00e9es \u00e0 afficher, cela permet de pouvoir inclure du code proche du Python dans la syntaxe au milieu d'un document contenant principalement du HTML. On peut facilement faire des boucles ou des conditions ainsi m\u00eame que de l'h\u00e9ritage de templates.

Note

le rendu est fait c\u00f4t\u00e9 serveur, si on souhaite faire des modifications c\u00f4t\u00e9 client, il faut utiliser du Javascript, rien ne change \u00e0 ce niveau-l\u00e0.

"},{"location":"explanation/technos/#jquery","title":"jQuery","text":"

Site officiel

jQuery est une biblioth\u00e8que JavaScript libre et multiplateforme cr\u00e9\u00e9e pour faciliter l'\u00e9criture de scripts c\u00f4t\u00e9 client dans le code HTML des pages web. La premi\u00e8re version est lanc\u00e9e en janvier 2006 par John Resig.

C'est une vieille technologie et certains feront remarquer \u00e0 juste titre que le Javascript moderne permet d'utiliser assez simplement la majorit\u00e9 de ce que fournit jQuery sans rien avoir \u00e0 installer. Cependant, de nombreuses d\u00e9pendances du projet utilisent encore jQuery qui est toujours tr\u00e8s implant\u00e9 aujourd'hui. Le sucre syntaxique qu'offre cette librairie reste tr\u00e8s agr\u00e9able \u00e0 utiliser et \u00e9conomise parfois beaucoup de temps. \u00c7a fonctionne et \u00e7a fonctionne tr\u00e8s bien. C'est maintenu et pratique.

"},{"location":"explanation/technos/#alpinejs","title":"AlpineJS","text":"

Site officiel

AlpineJS est une librairie l\u00e9g\u00e8re et minimaliste permettant le rendu dynamique d'\u00e9l\u00e9ments sur une page web, code de mani\u00e8re d\u00e9clarative. La librairie est d\u00e9crite par ses cr\u00e9ateurs comme : \"un outil robuste et minimal pour composer un comportement directement dans vos balises\".

Alpine permet d'accomplir la plupart du temps le m\u00eame r\u00e9sultat qu'un usage des fonctionnalit\u00e9s de base des plus gros frameworks Javascript, mais est beaucoup plus l\u00e9ger, un peu plus facile \u00e0 prendre en main et ne s'embarrasse pas d'un DOM virtuel. Gr\u00e2ce \u00e0 son architecture, il est extr\u00eamement bien adapt\u00e9 pour un usage dans un site multipage. C'est une technologie simple et puissante qui se veut comme le jQuery du web moderne.

"},{"location":"explanation/technos/#htmx","title":"Htmx","text":"

Site officiel

En plus de AlpineJS, l\u2019interactivit\u00e9 sur le site est augment\u00e9e via Htmx. C'est une librairie js qui s'utilise \u00e9galement au moyen d'attributs HTML \u00e0 ajouter directement dans les templates.

Son principe est de remplacer certains \u00e9l\u00e9ments du html par un fragment de HTML renvoy\u00e9 par le serveur backend. Cela se marie tr\u00e8s bien avec le fonctionnement de django et en particulier de ses formulaires afin d'\u00e9viter de doubler le travail pour la v\u00e9rification des donn\u00e9es.

"},{"location":"explanation/technos/#sass","title":"Sass","text":"

Site officiel

Sass (Syntactically Awesome Stylesheets) est un langage dynamique de g\u00e9n\u00e9ration de feuilles CSS apparu en 2006. C'est un langage de CSS \"am\u00e9lior\u00e9\" qui permet l'ajout de variables (\u00e0 une \u00e9poque o\u00f9 le CSS ne les supportait pas), de fonctions, mixins ainsi qu'une syntaxe pour imbriquer plus facilement et proprement les r\u00e8gles sur certains \u00e9l\u00e9ments. Le Sass est traduit en CSS directement c\u00f4t\u00e9 serveur et le client ne re\u00e7oit que du CSS.

C'est une technologie stable, mature et pratique qui ne n\u00e9cessite pas \u00e9norm\u00e9ment d'apprentissage.

"},{"location":"explanation/technos/#fontawesome","title":"Fontawesome","text":"

Site officiel

Fontawesome regroupe tout un ensemble d'ic\u00f4nes libres de droits utilisables facilement sur n'importe quelle page web. Ils sont simples \u00e0 modifier puisque modifiables via le CSS et pr\u00e9sentent l'avantage de fonctionner sur tous les navigateurs contrairement \u00e0 un simple ic\u00f4ne unicode qui s'affiche lui diff\u00e9remment selon la plate-forme.

Note

C'est une d\u00e9pendance capricieuse qui \u00e9volue tr\u00e8s vite et qu'il faut tr\u00e8s souvent mettre \u00e0 jour.

Warning

Il a \u00e9t\u00e9 d\u00e9cid\u00e9 de ne pas utiliser de CDN puisque le site ralentissait r\u00e9guli\u00e8rement. Il est pr\u00e9f\u00e9rable de fournir cette d\u00e9pendance avec le site.

"},{"location":"explanation/technos/#workflow","title":"Workflow","text":""},{"location":"explanation/technos/#git","title":"Git","text":"

Site officiel

Git est un logiciel de gestion de versions \u00e9crit par Linus Torvalds pour les besoins du noyau linux en 2005. C'est ce logiciel qui remplace svn anciennement utilis\u00e9 pour g\u00e9rer les sources du projet (rappelez vous, l'ancien site date d'avant 2005). Git est plus complexe \u00e0 utiliser, mais est bien plus puissant, permet de g\u00e9rer plusieurs versions en parall\u00e8le et g\u00e9n\u00e8re des codebases vraiment plus l\u00e9g\u00e8res puisque seules les modifications sont enregistr\u00e9es (contrairement \u00e0 svn qui garde une copie de la codebase par version).

Git s'\u00e9tant impos\u00e9 comme le principal outil de gestion de versions, sa communaut\u00e9 est tr\u00e8s grande et sa documentation tr\u00e8s fournie. Il est \u00e9galement ais\u00e9 de trouver des outils avec une interface graphique, qui simplifient grandement son usage.

"},{"location":"explanation/technos/#github","title":"GitHub","text":"

Site officiel

Page github du P\u00f4le Informatique de l'AE

Github est un service web d'h\u00e9bergement et de gestion de d\u00e9veloppement de logiciel. C'est une plate-forme avec interface web permettant de d\u00e9poser du code g\u00e9r\u00e9 avec Git offrant \u00e9galement de l'int\u00e9gration continue et du d\u00e9ploiement automatique. C'est au travers de cette plate-forme que le Sith de l'AE est g\u00e9r\u00e9.

"},{"location":"explanation/technos/#sentry","title":"Sentry","text":"

Site officiel

Instance de l'AE

Sentry est une plate-forme libre qui permet de se tenir inform\u00e9 des bugs qui ont lieu sur le site. \u00c0 chaque crash du logiciel (erreur 500), une erreur est envoy\u00e9e sur la plate-forme et il est indiqu\u00e9 pr\u00e9cis\u00e9ment \u00e0 quelle ligne de code celle-ci a eu lieu, \u00e0 quelle heure, combien de fois, avec quel navigateur la page a \u00e9t\u00e9 visit\u00e9e et m\u00eame \u00e9ventuellement un commentaire de l'utilisateur qui a rencontr\u00e9 le bug.

C'est un outil incroyablement pratique pour savoir tout ce qui ne fonctionne pas, et surtout pour r\u00e9colter toutes les informations n\u00e9cessaires \u00e0 la r\u00e9paration des bugs.

"},{"location":"explanation/technos/#uv","title":"UV","text":"

UV

UV est un utilitaire qui permet de cr\u00e9er et g\u00e9rer des environnements Python de mani\u00e8re simple et intuitive. Il permet \u00e9galement de g\u00e9rer et mettre \u00e0 jour le fichier de d\u00e9pendances.

L'avantage d'utiliser uv (et les environnements virtuels en g\u00e9n\u00e9ral) est de pouvoir g\u00e9rer plusieurs projets diff\u00e9rents en parall\u00e8le puisqu'il permet d'avoir sur sa machine plusieurs environnements diff\u00e9rents et donc plusieurs versions d'une m\u00eame d\u00e9pendance dans plusieurs projets diff\u00e9rents sans impacter le syst\u00e8me sur lequel le tout est install\u00e9.

UV poss\u00e8de \u00e9galement l'avantage par rapport \u00e0 un simple venv que les versions exactes de toutes les d\u00e9pendances, y compris celles utilis\u00e9es par d'autres d\u00e9pendances, sont consign\u00e9es dans un fichier .lock. On est donc s\u00fbr et certain que deux environnements virtuels configur\u00e9s avec le m\u00eame fichier lock utiliseront exactement les m\u00eames versions des m\u00eames d\u00e9pendances, y compris si celles-ci ne sont pas indiqu\u00e9es explicitement.

UV se charge m\u00eame de t\u00e9l\u00e9charger la bonne version de Python automatiquement !

Les d\u00e9pendances utilis\u00e9es par uv sont d\u00e9clar\u00e9es dans le fichier pyproject.toml, situ\u00e9 \u00e0 la racine du projet.

Aussi, uv est rapide, genre TR\u00c8S TR\u00c8S rapide \u26a1\ufe0f

"},{"location":"explanation/technos/#ruff","title":"Ruff","text":"

Site officiel

Pour faciliter la lecture du code, il est toujours appr\u00e9ciable d'avoir une norme d'\u00e9criture coh\u00e9rente. C'est g\u00e9n\u00e9ralement \u00e0 l'\u00e9tape de relecture des modifications par les autres contributeurs que sont rep\u00e9r\u00e9es ces fautes de normes qui se doivent d'\u00eatre corrig\u00e9es pour le bien commun.

Imposer une norme est tr\u00e8s fastidieux, que ce soit pour ceux qui relisent ou pour ceux qui \u00e9crivent. C'est pour cela que nous utilisons Ruff, qui est un formateur et linter automatique de code. Une fois l'outil lanc\u00e9, il parcourt la codebase pour y rep\u00e9rer les fautes de norme et les erreurs de logique courantes et les corrige automatiquement (quand c'est possible) sans que l'utilisateur ait \u00e0 s'en soucier. Bien install\u00e9, il peut effectuer ce travail \u00e0 chaque sauvegarde d'un fichier dans son \u00e9diteur, ce qui est tr\u00e8s agr\u00e9able pour travailler.

"},{"location":"explanation/technos/#biome","title":"Biome","text":"

Site officiel

Puisque Ruff ne fonctionne malheureusement que pour le Python, nous utilisons Biome pour le javascript.

Biome est \u00e9galement capable d'analyser et formater les fichiers json et css.

Tout comme Ruff, Biome fait office de formateur et de linter.

"},{"location":"explanation/technos/#djhtml","title":"DjHTML","text":"

Site officiel

Ruff permet de formater les fichiers Python et Biome les fichiers js, mais ils ne formattent pas les templates et les feuilles de style. Pour \u00e7a, il faut un autre outil, ais\u00e9ment int\u00e9grable dans la CI : djHTML.

En utilisant conjointement Ruff, Biome et djHTML, on arrive donc \u00e0 la fois \u00e0 formater les fichiers Python et les fichiers relatifs au frontend.

"},{"location":"explanation/technos/#npm","title":"Npm","text":"

Utiliser npm

Npm est un gestionnaire de paquets pour Node.js. C'est l'un des gestionnaires les plus r\u00e9pandus sur le march\u00e9 et est tr\u00e8s complet et utilis\u00e9.

Npm poss\u00e8de, tout comme Poetry, la capacit\u00e9 de locker les d\u00e9pendances au moyen d'un fichier .lock. Il a \u00e9galement l'avantage de presque toujours \u00eatre facilement disponible \u00e0 l'installation.

Nous l'utilisons ici pour g\u00e9rer les d\u00e9pendances JavaScript. Celle-ci sont d\u00e9clar\u00e9es dans le fichier package.json situ\u00e9 \u00e0 la racine du projet.

"},{"location":"explanation/technos/#vite","title":"Vite","text":"

Utiliser vite

Vite est un bundler de fichiers static. Il nous sert ici \u00e0 mettre \u00e0 disposition les d\u00e9pendances frontend g\u00e9r\u00e9es par npm.

Il sert \u00e9galement \u00e0 int\u00e9grer les autres outils JavaScript au workflow du Sith de mani\u00e8re transparente.

Vite a \u00e9t\u00e9 choisi pour sa versatilit\u00e9 et sa popularit\u00e9. Il est moderne et tr\u00e8s rapide avec un fort soutien de la communaut\u00e9.

Il int\u00e8gre aussi tout le n\u00e9cessaire pour la r\u00e9tro-compatibilit\u00e9 et le Typescript.

Le logiciel se configure au moyen du fichier vite.config.mts \u00e0 la racine du projet.

"},{"location":"howto/direnv/","title":"Direnv","text":"

Pour \u00e9viter d'avoir \u00e0 sourcer l'environnement \u00e0 chaque fois qu'on rentre dans le projet, il est possible d'utiliser l'utilitaire direnv.

Comme pour beaucoup de choses, il faut commencer par l'installer :

LinuxmacOS Debian/UbuntuArch Linux
sudo apt install direnv\n
sudo pacman -S direnv\n
brew install direnv\n

Puis on configure :

bashzshnu
echo 'eval \"$(direnv hook bash)\"' >> ~/.bashrc\nexit # On red\u00e9marre le terminal\n
echo 'eval \"$(direnv hook zsh)\"' >> ~/.zshrc\nexit # On red\u00e9marre le terminal\n

D\u00e9sol\u00e9, par direnv hook pour nu

Une fois le terminal red\u00e9marr\u00e9, dans le r\u00e9pertoire du projet :

direnv allow .\n

Une fois que cette configuration a \u00e9t\u00e9 appliqu\u00e9e, aller dans le dossier du site applique automatiquement l'environnement virtuel. \u00c7a peut faire gagner pas mal de temps.

Direnv est un utilitaire tr\u00e8s puissant et qui peut s'av\u00e9rer pratique dans bien des situations, n'h\u00e9sitez pas \u00e0 aller vous renseigner plus en d\u00e9tail sur celui-ci.

"},{"location":"howto/js-import-paths/","title":"Ajouter un chemin d'import javascript","text":"

Vous avez ajout\u00e9 une application et vous voulez y mettre du javascript ?

Vous voulez importer depuis cette nouvelle application dans votre script g\u00e9r\u00e9 par le bundler ?

Eh bien il faut manuellement enregistrer dans node o\u00f9 les trouver et c'est tr\u00e8s simple.

D'abord, il faut ajouter dans node via package.json:

{\n    // ...\n    \"imports\": {\n        // ...\n        \"#mon_app:*\": \"./mon_app/static/bundled/*\"\n    }\n    // ...\n}\n

Ensuite, pour faire fonctionne l'auto-compl\u00e9tion, il faut configurer tsconfig.json:

{\n    \"compilerOptions\": {\n        // ...\n        \"paths\": {\n            // ...\n            \"#mon_app:*\": [\"./mon_app/static/bundled/*\"]\n        }\n    }\n}\n

Et c'est tout !

Note

Il se peut qu'il soit n\u00e9cessaire de red\u00e9marrer ./manage.py runserver pour que les changements prennent effet.

"},{"location":"howto/logo/","title":"Ajouter un logo de promo","text":""},{"location":"howto/logo/#les-logos-de-promo","title":"Les logos de promo","text":"

Une fois par an, il est g\u00e9n\u00e9ralement n\u00e9cessaire d'ajouter le nouveau logo d'une promo. C'est un processus manuel.

Automatisation

Cr\u00e9er une interface automatique sur le site serait compliqu\u00e9 et long \u00e0 faire. Les erreurs de format des utilisateurs sont g\u00e9n\u00e9ralement nombreuses et on se retrouverais souvent avec des logos ratatin\u00e9s. Il est donc plus simple et plus fiable de faire cette op\u00e9ration manuellement, \u00e7a prend quelques minutes et on est certain de la qualit\u00e9 \u00e0 la fin.

Les logos de promo sont \u00e0 manuellement ajouter dans le projet. Ils se situent dans le dossier core/static/core/img/.

Leur format est le suivant:

"},{"location":"howto/migrations/","title":"G\u00e9rer les migrations","text":""},{"location":"howto/migrations/#quest-ce-quune-migration","title":"Qu'est-ce qu'une migration ?","text":"

Une migration est un fichier Python qui contient des instructions pour modifier la base de donn\u00e9es. Une base de donn\u00e9es \u00e9volue au cours du temps, et les migrations permettent de garder une trace de ces modifications.

Gr\u00e2ce \u00e0 elles, on peut \u00e9galement apporter des modifications \u00e0 la base de donn\u00e9es sans \u00eatre oblig\u00e9es de la recr\u00e9er. On applique seulement les modifications n\u00e9cessaires.

"},{"location":"howto/migrations/#appliquer-les-migrations","title":"Appliquer les migrations","text":"

Pour appliquer les migrations, ex\u00e9cutez la commande suivante :

python ./manage.py migrate\n

Vous remarquerez peut-\u00eatre que cette commande a \u00e9t\u00e9 utilis\u00e9e dans la section Installation. En effet, en partant d'une base de donn\u00e9es vierge et en appliquant toutes les migrations, on arrive \u00e0 l'\u00e9tat actuel de la base de donn\u00e9es. Logique.

Si vous utilisez cette commande sur une base de donn\u00e9es sur laquelle toutes les migrations ont \u00e9t\u00e9 appliqu\u00e9es, elle ne fera rien.

Si vous utilisez cette commande sur une base de donn\u00e9es sur laquelle seule une partie des migrations ont \u00e9t\u00e9 appliqu\u00e9es, seules les migrations manquantes seront appliqu\u00e9es.

"},{"location":"howto/migrations/#creer-une-migration","title":"Cr\u00e9er une migration","text":"

Pour cr\u00e9er une migration, ex\u00e9cutez la commande suivante :

python ./manage.py makemigrations\n

Cette commande comparera automatiquement le contenu des classes de mod\u00e8les et le comparera avec les migrations d\u00e9j\u00e0 appliqu\u00e9es. A partir de cette comparaison, elle g\u00e9n\u00e9rera automatiquement une nouvelle migration.

Note

La commande makemigrations ne fait que g\u00e9n\u00e9rer les fichiers de migration. Elle ne modifie pas la base de donn\u00e9es. Pour appliquer la migration, n'oubliez pas la commande migrate.

Un fichier de migration ressemble \u00e0 \u00e7a :

from django.db import migrations\n\n\nclass Migration(migrations.Migration):\n\n    dependencies = [\n        # liste des autres migrations \u00e0 appliquer avant celle-ci\n    ]\n\n    operations = [\n        # liste des op\u00e9rations \u00e0 appliquer sur la db\n    ]\n

Gr\u00e2ce \u00e0 la liste des d\u00e9pendances, Django sait dans quel ordre les migrations doivent \u00eatre appliqu\u00e9es. Gr\u00e2ce \u00e0 la liste des op\u00e9rations, Django sait quelles sont les op\u00e9rations \u00e0 appliquer durant cette migration.

"},{"location":"howto/migrations/#revenir-a-une-migration-anterieure","title":"Revenir \u00e0 une migration ant\u00e9rieure","text":"

Lorsque vous d\u00e9veloppez, il peut arriver que vous vouliez revenir \u00e0 une migration ant\u00e9rieure. Pour cela, il suffit d'appliquer la commande migrate en sp\u00e9cifiant le nom de la migration \u00e0 laquelle vous voulez revenir :

python ./manage.py migrate <application> <num\u00e9ro de la migration>\n

Par exemple, si vous voulez revenir \u00e0 la migration 0001_initial de l'application customer, vous pouvez ex\u00e9cuter la commande suivante :

python ./manage.py migrate customer 0001\n
"},{"location":"howto/migrations/#customiser-une-migration","title":"Customiser une migration","text":"

Il peut arriver que vous ayez besoin de modifier le fichier de migration g\u00e9n\u00e9r\u00e9 par Django. Par exemple, si vous voulez ex\u00e9cuter un script Python lors de l'application de la migration.

Dans ce cas, vous pouvez trouver les fichiers de migration dans le dossier migrations de chaque application. Vous pouvez modifier le fichier Python correspondant \u00e0 la migration que vous voulez modifier.

Ajoutez l'op\u00e9ration que vous voulez effectuer dans l'attribut operations de la classe Migration.

Par exemple :

from django.db import migrations\n\ndef forwards_func(apps, schema_editor):\n    print(\"Appplication de la migration\")\n\ndef reverse_func(apps, schema_editor):\n    print(\"Annulation de la migration\")\n\nclass Migration(migrations.Migration):\n\n    dependencies = []\n\n    operations = [\n        migrations.RunPython(forwards_func, reverse_func),\n    ]\n

Script d'annulation de la migration

Lorsque vous incluez un script Python dans une migration, incluez toujours aussi un script d'annulation, sinon Django ne pourra pas annuler la migration apr\u00e8s son application.

Vous ne pourrez donc pas revenir \u00e0 un \u00e9tat ant\u00e9rieur de la db, \u00e0 moins de la recr\u00e9er de z\u00e9ro.

"},{"location":"howto/migrations/#fusionner-des-migrations","title":"Fusionner des migrations","text":"

Quand on travaille sur une fonctionnalit\u00e9 qui n\u00e9cessite une modification de la base de donn\u00e9es, les fichiers de migration sont comme toute chose : on peut se rendre compte que les changements apport\u00e9s pourraient \u00eatre meilleurs.

Par exemple, supposons que nous voulons cr\u00e9er un mod\u00e8le repr\u00e9sentant une UE suivie par un \u00e9tudiant (ne demandez pas pourquoi on voudrait faire \u00e7a, c'est juste pour l'exemple). Un tel mod\u00e8le aurait besoin des informations suivantes :

On \u00e9crirait donc, dans l'application pedagogy :

from django.db import models\nfrom core.models import User\n\nclass UserUe(models.Model):\n    user = models.ForeignKey(User, on_delete=models.CASCADE)\n    ue = models.CharField(max_length=10)\n

Et nous aurions le fichier de migration suivant :

from django.db import migrations, models\nfrom django.conf import settings\n\nclass Migration(migrations.Migration):\n    dependencies = [\n        migrations.swappable_dependency(settings.AUTH_USER_MODEL),\n        (\"pedagogy\", \"0003_alter_uv_language\"),\n    ]\n\n    operations = [\n        migrations.CreateModel(\n            name=\"UserUe\",\n            fields=[\n                (\n                    \"id\",\n                    models.AutoField(\n                        auto_created=True,\n                        primary_key=True,\n                        serialize=False,\n                        verbose_name=\"ID\",\n                    ),\n                ),\n                (\"ue\", models.CharField(max_length=10)),\n                (\n                    \"user\",\n                    models.ForeignKey(\n                        on_delete=models.deletion.CASCADE,\n                        to=settings.AUTH_USER_MODEL,\n                    ),\n                ),\n            ],\n        ),\n    ]\n

On finit son travail, on soumet la PR. Mais l\u00e0, quelqu'un fait remarquer qu'il existe d\u00e9j\u00e0 un mod\u00e8le pour repr\u00e9senter une UE. On modifie donc le mod\u00e8le :

from django.db import models\nfrom core.models import User\nfrom pedagogy.models import UV\n\nclass UserUe(models.Model):\n    user = models.ForeignKey(User, on_delete=models.CASCADE)\n    ue = models.ForeignKey(UV, on_delete=models.CASCADE)\n

On refait la commande makemigrations et on obtient :

from django.db import migrations, models\n\n\nclass Migration(migrations.Migration):\n    dependencies = [\n        (\"pedagogy\", \"0004_userue\"),\n    ]\n\n    operations = [\n        migrations.AlterField(\n            model_name=\"userue\",\n            name=\"ue\",\n            field=models.ForeignKey(\n                on_delete=models.deletion.CASCADE, to=\"pedagogy.uv\"\n            ),\n        ),\n    ]\n

Sauf que maintenant, nous avons deux fichiers de migration, alors qu'en r\u00e9alit\u00e9, on ne souhaite faire qu'une seule migration, une fois qu'on aura exp\u00e9di\u00e9 le code en prod. Certes, \u00e7a fonctionnerait d'appliquer les deux, mais \u00e7a pose un probl\u00e8me d'encombrement.

Plus il y a de fichiers de migrations, plus il y a de migrations \u00e0 r\u00e9soudre au moment de l'installation du projet chez quelqu'un, plus c'est emb\u00eatant \u00e0 g\u00e9rer et plus Django prendra du temps \u00e0 r\u00e9soudre les migrations.

C'est pourquoi il est bon de respecter le principe : une PR = un fichier de migration maximum par application.

Nous voulons donc fusionner les deux, pour n'en garder qu'une. Pour \u00e7a, deux mani\u00e8res de proc\u00e9der :

Pour la m\u00e9thode manuelle, on ne pourrait pas vous dire exhaustivement comment faire. Mais ne vous inqui\u00e9tez pas, ce n'est pas tr\u00e8s dur. Regardez bien quelles sont les instructions utilis\u00e9es par django pour les op\u00e9rations de migrations, et avec un peu d'astuce et quelques copier-coller, vous vous en sortirez comme des chefs.

Pour la m\u00e9thode squashmigrations, ex\u00e9cutez la commande

python ./manage.py squasmigrations <app> <migration de d\u00e9but (incluse)> <migration de fin (incluse)> \n

Par exemple, dans notre cas, \u00e7a donnera :

python ./manage.py squasmigrations pedagogy 0004 0005 \n

La commande vous donnera ceci :

from django.conf import settings\nfrom django.db import migrations, models\n\nclass Migration(migrations.Migration):\n    replaces = [(\"pedagogy\", \"0004_userue\"), (\"pedagogy\", \"0005_alter_userue_ue\")]\n\n    dependencies = [\n        (\"pedagogy\", \"0003_alter_uv_language\"),\n        migrations.swappable_dependency(settings.AUTH_USER_MODEL),\n    ]\n\n    operations = [\n        migrations.CreateModel(\n            name=\"UserUe\",\n            fields=[\n                (\n                    \"id\",\n                    models.AutoField(\n                        auto_created=True,\n                        primary_key=True,\n                        serialize=False,\n                        verbose_name=\"ID\",\n                    ),\n                ),\n                (\n                    \"ue\",\n                    models.ForeignKey(\n                        on_delete=models.deletion.CASCADE, to=\"pedagogy.uv\"\n                    ),\n                ),\n                (\n                    \"user\",\n                    models.ForeignKey(\n                        on_delete=models.deletion.CASCADE,\n                        to=settings.AUTH_USER_MODEL,\n                    ),\n                ),\n            ],\n        ),\n    ]\n

Vous pouvez alors supprimer les deux autres fichiers.

Vous remarquerez peut-\u00eatre la pr\u00e9sence de la ligne suivante :

replaces = [(\"pedagogy\", \"0004_userue\"), (\"pedagogy\", \"0005_alter_userue_ue\")]\n

Cela sert \u00e0 dire que cette migration doit \u00eatre appliqu\u00e9e \u00e0 la place des deux autres. Une fois que vous aurez supprim\u00e9 les deux fichiers, supprimez \u00e9galement cette ligne.

Warning

Django sait quelles migrations ont \u00e9t\u00e9 appliqu\u00e9es, en les stockant dans une table de la db. Si une migration est enregistr\u00e9e en db, sans que le fichier de migration correspondant existe, la commande migrate \u00e9choue.

Quand vous faites un squashmigrations, pensez donc \u00e0 appliquer la commande migrate juste apr\u00e8s (mais avant la suppression des anciens fichiers), pour que Django supprime de la base de donn\u00e9es les migrations devenues inutiles.

"},{"location":"howto/prod/","title":"Configurer pour la production","text":""},{"location":"howto/prod/#configurer-sentry","title":"Configurer Sentry","text":"

Pour connecter l'application \u00e0 une instance de sentry (ex: https://sentry.io), il est n\u00e9cessaire de configurer la variable SENTRY_DSN dans le fichier settings_custom.py. Cette variable est compos\u00e9e d'un lien complet vers votre projet sentry.

"},{"location":"howto/prod/#recuperer-les-statiques","title":"R\u00e9cup\u00e9rer les statiques","text":"

Nous utilisons du SCSS dans le projet. En environnement de d\u00e9veloppement (DEBUG=True), le SCSS est compil\u00e9 \u00e0 chaque fois que le fichier est demand\u00e9. Pour la production, le projet consid\u00e8re que chacun des fichiers est d\u00e9j\u00e0 compil\u00e9. C'est pourquoi le SCSS est automatiquement compil\u00e9 lors de la r\u00e9cup\u00e9ration des fichiers statiques. Les fichiers JS sont \u00e9galement automatiquement minifi\u00e9s.

Il peut \u00eatre judicieux de supprimer les anciens fichiers statiques avant de collecter les nouveaux. Pour \u00e7a, ajoutez le flag --clear \u00e0 la commande collectstatic :

python ./manage.py collectstatic --clear\n

Tip

Le dossier o\u00f9 seront enregistr\u00e9s ces fichiers statiques peut \u00eatre chang\u00e9 en modifiant la variable STATIC_ROOT dans les param\u00e8tres.

"},{"location":"howto/querysets/","title":"L'ORM de Django","text":"

L'ORM de Django est puissant, tr\u00e8s puissant, non par parce qu'il est performant (apr\u00e8s tout, ce n'est qu'une interface, le gros du boulot, c'est la db qui le fait), mais parce qu'il permet d'\u00e9crire de mani\u00e8re relativement simple un grand panel de requ\u00eates.

De mani\u00e8re g\u00e9n\u00e9rale, puisqu'un ORM est un syst\u00e8me consistant \u00e0 manipuler avec un code orient\u00e9-objet une db relationnelle (c'est-\u00e0-dire deux paradigmes qui ne fonctionnent absolument pas pareil), on rencontre un des deux probl\u00e8mes suivants :

Django est dans ce deuxi\u00e8me cas.

C'est pourquoi nous ne parlerons pas ici de son fonctionnement exact ni de toutes les fonctions que l'on peut utiliser (la doc officielle fait d\u00e9j\u00e0 \u00e7a mieux que nous), mais plut\u00f4t des pi\u00e8ges courants et des astuces pour les \u00e9viter.

"},{"location":"howto/querysets/#les-n1-queries","title":"Les N+1 queries","text":""},{"location":"howto/querysets/#le-probleme","title":"Le probl\u00e8me","text":"

Normalement, quand on veut r\u00e9cup\u00e9rer une liste, on fait une requ\u00eate et c'est fini. Mais des fois, \u00e7a n'est pas si simple. Par exemple, supposons que nous voulons r\u00e9cup\u00e9rer les 100 utilisateurs les plus riches, avec leurs informations client :

from core.models import User\n\nfor user in User.objects.order_by(\"-customer__amount\")[:100]:\n    print(user.customer.amount)\n

Combien de requ\u00eates le bout de code suivant effectue-t-il ? 101. En deux pauvres lignes de code, nous avons demand\u00e9 \u00e0 la base de donn\u00e9es d'effectuer 101 requ\u00eates. Une requ\u00eate toute seule n'est d\u00e9j\u00e0 une op\u00e9ration anodine, alors je vous laisse imaginer ce que \u00e7a donne pour 101.

Si vous ne comprenez pourquoi ce nombre, c'est tr\u00e8s simple :

En effet, les informations client sont stock\u00e9es dans une autre table, mais le fait d'\u00e9tablir un lien de clef \u00e9trang\u00e8re permet de manipuler customer comme si c'\u00e9tait un membre \u00e0 part enti\u00e8re de User.

Il est \u00e0 noter cependant, que Django n'effectue une requ\u00eate que pour le premier acc\u00e8s \u00e0 un membre d'une relation de clef \u00e9trang\u00e8re. Toutes les fois suivantes, l'objet est d\u00e9j\u00e0 l\u00e0, et django le r\u00e9cup\u00e8re :

from core.models import User\n\n# l'utilisateur le plus riche\nuser = User.objects.order_by(\"-customer__amount\").first()  # <-- requ\u00eate db\nprint(user.customer.amount)  # <-- requ\u00eate db\nprint(user.customer.account_id)  # on a d\u00e9j\u00e0 r\u00e9cup\u00e9r\u00e9 `customer`, donc pas de requ\u00eate\n

Ce n'est donc pas gravissime si vous faites cette erreur quand vous manipulez un seul objet. En revanche, quand vous en manipulez plusieurs, il faut r\u00e9gler le probl\u00e8me. Pour \u00e7a, il y a plusieurs m\u00e9thodes, en fonction de votre cas.

"},{"location":"howto/querysets/#select_related","title":"select_related","text":"

La m\u00e9thode la plus basique consiste \u00e0 annoter le queryset, avec la m\u00e9thode select_related(). En faisant \u00e7a, Django fera une jointure sur l'autre table et demandera des informations en plus \u00e0 la db lors de la requ\u00eate.

De la sorte, lorsque vous appellerez le membre reli\u00e9, les informations seront d\u00e9j\u00e0 l\u00e0.

from core.models import User\n\nrichest = User.objects.order_by(\"-customer__amount\")\nfor user in richest.select_related(\"customer\")[:100]:\n    print(user.customer)\n

Le code ci-dessus effectue une seule requ\u00eate. Chaque fois qu'on veut acc\u00e9der \u00e0 customer, c'est bon, \u00e7a a d\u00e9j\u00e0 \u00e9t\u00e9 r\u00e9cup\u00e9r\u00e9 \u00e0 travers le select_related.

"},{"location":"howto/querysets/#prefetch_related","title":"prefetch_related","text":"

Maintenant, un cas plus compliqu\u00e9. Supposons que vous ne vouliez pas r\u00e9cup\u00e9rer des informations reli\u00e9es par une relation One-to-One, mais par une relation One-to-Many.

Par exemple, un utilisateur a un seul compte client, mais il peut avoir plusieurs cotisations \u00e0 son actif. Et dans ces cas-l\u00e0, annotate ne marche plus. En effet, s'il peut exister plusieurs cotisations, comment savoir laquelle on veut ?

Il faut alors utiliser un prefetch_related. C'est un m\u00e9canisme un peu diff\u00e9rent : au lieu de faire une jointure et d'ajouter les informations voulues dans la m\u00eame requ\u00eate, Django va effectuer une deuxi\u00e8me requ\u00eate pour r\u00e9cup\u00e9rer les \u00e9l\u00e9ments de l'autre table, puis, \u00e0 partir de ces \u00e9l\u00e9ments, peupler la relation de son c\u00f4t\u00e9.

C'est un m\u00e9canisme qui peut \u00eatre un peu co\u00fbteux en m\u00e9moire et qui demande une deuxi\u00e8me requ\u00eate, mais qui reste quand m\u00eame largement pr\u00e9f\u00e9rable \u00e0 faire N requ\u00eates en plus.

from core.models import User\n\nfor user in User.objects.prefetch_related(\"subscriptions\")[:100]:\n    # c'est bon, la m\u00e9thode prefetch a r\u00e9cup\u00e9r\u00e9 en avance les `subscriptions`\n    print(user.subscriptions.all())\n

Danger

La m\u00e9thode prefetch_related ne marche que si vous utilisez la m\u00e9thode all() pour acc\u00e9der au membre. Si vous utilisez une autre m\u00e9thode (comme filter ou annotate), alors Django effectuera une nouvelle requ\u00eate, et vous retomberez dans le probl\u00e8me initial.

from core.models import User\nfrom django.db.models import Count\n\nfor user in User.objects.prefetch_related(\"subscriptions\")[:100]:\n    # Le prefetch_related ne marche plus !\n    print(user.subscriptions.annotate(count=Count(\"*\")))\n
"},{"location":"howto/querysets/#recuperer-ce-dont-vous-avez-besoin","title":"R\u00e9cup\u00e9rer ce dont vous avez besoin","text":"

Des fois (souvent, m\u00eame), penser explicitement \u00e0 la jointure est le meilleur choix.

En effet, vous remarquerez que dans tous les exemples pr\u00e9c\u00e9dents, nous n'utilisions qu'une partie des informations (par exemple, nous ne r\u00e9cup\u00e9rions que la somme d'argent sur les comptes, et \u00e9ventuellement le num\u00e9ro de compte).

Nous pouvons utiliser la m\u00e9thode annotate pour sp\u00e9cifier explicitement les donn\u00e9es que l'on veut joindre \u00e0 notre requ\u00eate.

Quand nous voulions r\u00e9cup\u00e9rer les informations utilisateur, nous aurions tout aussi bien pu \u00e9crire :

from core.models import User\nfrom django.db.models import F\n\nrichest = User.objects.order_by(\"-customer__amount\")\nfor user in richest.annotate(amount=F(\"customer__amount\"))[:100]:\n    print(user.amount)\n

On aurait m\u00eame pu r\u00e9organiser \u00e7a :

from core.models import User\nfrom django.db.models import F\n\nrichest = User.objects.annotate(amount=F(\"customer__amount\")).order_by(\"-amount\")\nfor user in richest[:100]:\n    print(user.amount)\n

\u00c7a peut sembler moins bien qu'un select_related, comme \u00e7a. Des fois, c'est en effet moins bien, et des fois c'est mieux. La comparaison est plus \u00e9vidente avec le prefetch_related.

En effet, quand nous voulions r\u00e9cup\u00e9rer le nombre de cotisations des utilisateurs, le prefetch_related ne marchait plus. Pourtant, nous voulions r\u00e9cup\u00e9rer une seule information.

Il aurait donc \u00e9t\u00e9 suffisant d'\u00e9crire :

from core.models import User\nfrom django.db.models import Count\n\nfor user in User.objects.annotate(nb_subscriptions=Count(\"subscriptions\"))[:100]:\n    # Et l\u00e0 \u00e7a marche, en une seule requ\u00eate.\n    print(user.nb_subscriptions)\n

Faire une jointure, c'est normal en SQL. Et pourtant avec Django on les oublie trop facilement. Posez-vous toujours la question des donn\u00e9es que vous pourriez avoir besoin d'annoter, et vous \u00e9viterez beaucoup d'ennuis.

"},{"location":"howto/querysets/#les-aggregations-manquees","title":"Les aggr\u00e9gations manqu\u00e9es","text":"

Il arrive souvent que l'on veuille une information qui porte sur un ensemble d'objets de notre db.

Imaginons par exemple que nous voulons connaitre la somme totale des ventes faites \u00e0 un comptoir.

Nous avons tous suivi nos cours de programmation, nous \u00e9crivons donc instinctivement :

from counter.models import Counter\n\nfoyer = Counter.objects.get(name=\"Foyer\")\ntotal_amount = sum(\n    sale.amount * sale.unit_price\n    for sale in foyer.sellings.all()\n)\n

On pourrait penser qu'il n'y a pas de probl\u00e8me. Apr\u00e8s tout, on ne fait qu'une seule requ\u00eate. Eh bien si, il y a un probl\u00e8me : on fait beaucoup de choses en trop.

Concr\u00e8tement, on demande \u00e0 la base de donn\u00e9es de renvoyer toutes les informations, ce qui rallonge inutilement la dur\u00e9e de l'\u00e9change entre le serveur et la db, puis on perd du temps \u00e0 convertir ces informations en objets Python (op\u00e9ration qui a un co\u00fbt \u00e9galement), et enfin on reperd du temps \u00e0 calculer en Python quelque chose que la db aurait pu calculer \u00e0 notre plus bien plus vite.

Nous aurions d\u00fb aggr\u00e9ger la requ\u00eate, avec la m\u00e9thode aggregate :

from counter.models import Counter\nfrom django.db.models import Sum, F\n\nfoyer = Counter.objects.get(name=\"Foyer\")\ntotal_amount = (\n    foyer.sellings.aggregate(amount=Sum(F(\"amount\") * F(\"unit_price\"), default=0))\n)[\"amount__sum\"]\n

En effectuant cette requ\u00eate, la base de donn\u00e9es nous renverra exactement l'information dont nous avons besoin. Et de notre c\u00f4t\u00e9, nous n'aurons pas \u00e0 faire de traitement en plus.

"},{"location":"howto/querysets/#benchmark","title":"Benchmark","text":""},{"location":"howto/querysets/#ce-quil-faut-mesurer","title":"Ce qu'il faut mesurer","text":"

Quand on parle d'interaction avec une base de donn\u00e9es, la question de la performance est cruciale. Et quand on parle de performance, on en vient forc\u00e9ment \u00e0 parler d'optimisation.

Or, pour optimiser, il faut savoir quoi optimiser. C'est-\u00e0-dire qu'il nous faut un benchmark pour \u00e9tudier les performances r\u00e9elles de notre code. En ce qui concerne des requ\u00eates \u00e0 une base de donn\u00e9es, deux aspects sont \u00e9tudiables :

Le premier aspect est celui qui nous int\u00e9resse le plus, puisqu'il est reli\u00e9 au probl\u00e8me le plus fr\u00e9quent et le plus facile \u00e0 mesurer. Le second aspect, au contraire, est bien moins fr\u00e9quent (dans 99% des cas, une requ\u00eate complexe prendra moins de temps que deux requ\u00eates, m\u00eame simples) et bien plus dur \u00e0 mesurer (il faut r\u00e9ussir \u00e0 faire des mesures fiables, dans un environnement proche de celui de la prod, avec les donn\u00e9es de la prod).

Nous consid\u00e9rerons donc que dans la quasi-totalit\u00e9 des cas, le probl\u00e8me vient du nombre de requ\u00eates, pas du temps d'ex\u00e9cution d'une requ\u00eate en particulier. Partez du principe que moins vous faites de requ\u00eates, mieux c'est, sans pr\u00eater attention au temps d'ex\u00e9cution des requ\u00eates.

Pour quantifier de mani\u00e8re fiables les requ\u00eates effectu\u00e9es, il y a quelques outils.

"},{"location":"howto/querysets/#django-debug-toolbar","title":"django-debug-toolbar","text":"

La django-debug-toolbar est une interface disponible sur toutes les pages quand vous \u00eates en mode debug. Elle s'affiche \u00e0 droite et vous permet de voir toutes sortes d'informations, parmi lesquelles le nombre de requ\u00eates effectu\u00e9es.

Cette interface est tr\u00e8s pratique, puisqu'elle va plus loin que simplement compter les requ\u00eates, elle vous donne \u00e9galement le SQL qui a \u00e9t\u00e9 utilis\u00e9, l'endroit du code, avec fichier et num\u00e9ro de ligne, o\u00f9 cette requ\u00eate a \u00e9t\u00e9 faite et, encore mieux, elle vous indique quelles requ\u00eates semblent dupliqu\u00e9es.

Quand django-debug-toolbar vous indique qu'une requ\u00eate a \u00e9t\u00e9 dupliqu\u00e9e quatre fois, cinq fois, ou m\u00eame deux cent fois (le chiffre peut sembler \u00e9norme, mais c'est d\u00e9j\u00e0 arriv\u00e9), vous pouvez \u00eatre s\u00fbr qu'il y a l\u00e0 quelque chose \u00e0 optimiser.

Warning

Le widget de django-debug-toolbar ne s'affiche que sur les pages html. Si vous voulez \u00e9tudier autre chose, comme une simple fonction, ou bien comme une vue retournant du JSON, vous n'aurez donc pas django-debug-toolbar.

"},{"location":"howto/querysets/#connectionqueries","title":"connection.queries","text":"

Quand vous voulez examiner les requ\u00eates d'un bout de code en particulier, Django met \u00e0 disposition un m\u00e9canisme permettant d'examiner toutes les requ\u00eates qui sont faites : connection.queries

C'est un historique de toutes les requ\u00eates effectu\u00e9es, qui est assez simple \u00e0 utiliser :

from django.db import connection\nfrom core.models import User\n\nprint(len(connection.queries))  # 0\n\nnb_users = User.objects.count()\n\nprint(len(connection.queries))  # 1\nprint(connection.queries)  # affiche toutes les requ\u00eates effectu\u00e9es\n
"},{"location":"howto/querysets/#assertnumqueries","title":"assertNumQueries","text":"

Quand on a mis en place une fonctionnalit\u00e9, ou qu'on en a am\u00e9lior\u00e9 les performances, on veut absolument \u00e9viter la r\u00e9gression.

Or, une r\u00e9gression ne se manifeste pas forc\u00e9ment dans l'apparition d'un bug : \u00e7a peut aussi \u00eatre une augmentation du temps d'ex\u00e9cution, possiblement caus\u00e9 par une augmentation du nombre de requ\u00eates.

C'est pour \u00e7a que django met \u00e0 disposition un moyen de tester automatiquement le nombre de requ\u00eates : assertNumQueries.

Il s'agit d'un gestionnaire de contexte accessible dans les tests, qui teste le nombre de requ\u00eates effectu\u00e9es en son sein.

Par exemple :

from django.test import TestCase\nfrom django.shortcuts import reverse\n\n\nclass FooTest(TestCase):\n    def test_nb_queries(self):\n        \"\"\"Test that the number of db queries is stable.\"\"\"\n        with self.assertNumQueries(6):\n            self.client.get(reverse(\"foo:bar\"))\n

Si l'ex\u00e9cution de la route n\u00e9cessite plus ou moins de six requ\u00eates, alors le test \u00e9choue. S'il y a eu moins que le nombre de requ\u00eate attendu, alors tant mieux, modifiez le test pour coller au nouveau nombre (sous r\u00e9serve que tous les autres tests passent, bien s\u00fbr). Si par contre il y a eu plus, alors d\u00e9sol\u00e9, vous avez sans doute introduit une r\u00e9gression.

"},{"location":"howto/statics/","title":"G\u00e9rer les statics","text":""},{"location":"howto/statics/#cest-quoi-les-fichiers-statics","title":"C'est quoi les fichiers statics ?","text":"

Les fichiers statics sont tous les fichiers qui ne sont pas g\u00e9n\u00e9r\u00e9s par le backend Django et qui sont t\u00e9l\u00e9charg\u00e9s par le navigateur. Cela comprend les fichiers css, javascript, images et autre.

La documentation officielle est tr\u00e8s compr\u00e9hensive.

Pour faire court, dans chaque module d'application il existe un dossier static o\u00f9 mettre tous ces fichiers. Django se d\u00e9brouille ensuite pour aller chercher ce qu'il faut \u00e0 l'int\u00e9rieur.

Pour acc\u00e9der \u00e0 un fichier static dans un template Jinja il suffit d'utiliser la fonction static.

    {# Exemple pour ajouter sith/core/static/core/base.css #}\n    <link rel=\"stylesheet\" href=\"{{ static('core/base.css') }}\">\n
"},{"location":"howto/statics/#lintegration-des-scss","title":"L'int\u00e9gration des scss","text":"

Les scss sont \u00e0 mettre dans le dossier static comme le reste. Il n'y a aucune diff\u00e9rence avec le reste pour les inclure, le syst\u00e8me se d\u00e9brouille automatiquement pour les transformer en .css

    {# Exemple pour ajouter sith/core/static/core/base.scss #}\n    <link rel=\"stylesheet\" href=\"{{ static('core/style.scss') }}\">\n
"},{"location":"howto/statics/#lintegration-avec-le-bundler-javascript","title":"L'int\u00e9gration avec le bundler javascript","text":"

Le bundler javascript est int\u00e9gr\u00e9 un peu diff\u00e9rement. Le principe est tr\u00e8s similaire mais les fichiers sont \u00e0 mettre dans un dossier static/bundled de l'application \u00e0 la place.

Pour acc\u00e9der au fichier, il faut utiliser static comme pour le reste mais en ajouter bundled/ comme prefix.

    {# Example pour ajouter sith/core/bundled/alpine-index.js #}\n    <script type=\"module\" src=\"{{ static('bundled/alpine-index.js') }}\"></script>\n    <script type=\"module\" src=\"{{ static('bundled/other-index.ts') }}\"></script>\n

Note

Seuls les fichiers se terminant par index.js sont export\u00e9s par le bundler. Les autres fichiers sont disponibles \u00e0 l'import dans le JavaScript comme si ils \u00e9taient tous au m\u00eame niveau.

Warning

Le bundler ne g\u00e9n\u00e8re que des modules javascript. Ajouter type=\"module\" n'est pas optionnel !

"},{"location":"howto/statics/#les-imports-au-sein-des-fichiers-des-fichiers-javascript-bundles","title":"Les imports au sein des fichiers des fichiers javascript bundl\u00e9s","text":"

Pour importer au sein d'un fichier js bundl\u00e9, il faut pr\u00e9fixer ses imports de #app:.

Exemple:

import { paginated } from \"#core:utils/api\";\n
"},{"location":"howto/statics/#comment-ca-fonctionne-le-post-processing","title":"Comment \u00e7a fonctionne le post processing ?","text":"

Le post processing est g\u00e9r\u00e9 par le module staticfiles. Les fichiers sont compil\u00e9s \u00e0 la vol\u00e9e en mode d\u00e9veloppement.

Pour la production, ils sont compil\u00e9s uniquement lors du ./manage.py collectstatic. Les fichiers g\u00e9n\u00e9r\u00e9s sont ajout\u00e9s dans le dossier staticfiles/generated. Celui-ci est ensuite enregistr\u00e9 comme dossier suppl\u00e9mentaire \u00e0 collecter dans Django.

"},{"location":"howto/subscriptions/","title":"Ajouter une cotisation","text":""},{"location":"howto/subscriptions/#ajouter-une-nouvelle-cotisation","title":"Ajouter une nouvelle cotisation","text":"

Il peut arriver que le type de cotisation propos\u00e9 varie en prix et en dur\u00e9e. Ces param\u00e8tres sont configurables directement dans les param\u00e8tres du projet.

Pour modifier les cotisations disponibles, tout se g\u00e8re dans la configuration avec la variable SITH_SUBSCRIPTIONS.

Par exemple, si nous voulons ajouter une nouvelle cotisation d'un mois, voici ce que nous ajouterons :

settings.py
from django.utils.translation import gettext_lazy as _\n\nSITH_SUBSCRIPTIONS = {\n    # Voici un \u00e9chantillon de la v\u00e9ritable configuration \u00e0 l'heure de l'\u00e9criture.\n    # Celle-ci est donn\u00e9e \u00e0 titre d'exemple pour mieux comprendre comment cela fonctionne.\n    \"un-semestre\": {\"name\": _(\"One semester\"), \"price\": 15, \"duration\": 1},\n    \"deux-semestres\": {\"name\": _(\"Two semesters\"), \"price\": 28, \"duration\": 2},\n    \"cursus-tronc-commun\": {\n        \"name\": _(\"Common core cursus\"),\n        \"price\": 45,\n        \"duration\": 4,\n    },\n    \"cursus-branche\": {\"name\": _(\"Branch cursus\"), \"price\": 45, \"duration\": 6},\n    \"cursus-alternant\": {\"name\": _(\"Alternating cursus\"), \"price\": 30, \"duration\": 6},\n    \"membre-honoraire\": {\"name\": _(\"Honorary member\"), \"price\": 0, \"duration\": 666},\n    \"un-jour\": {\"name\": _(\"One day\"), \"price\": 0, \"duration\": 0.00555333},\n\n    # On rajoute ici notre cotisation\n    # Elle se nomme \"Un mois\"\n    # Co\u00fbte 6\u20ac\n    # Dure 1 mois (on raisonne en semestre, ici, c'est 1/6 de semestre)\n    \"un-mois\": {\"name\": _(\"One month\"), \"price\": 6, \"duration\": 0.166}\n}\n

Une fois ceci fait, il faut cr\u00e9er une nouvelle migration :

python ./manage.py makemigrations subscription\npython ./manage.py migrate\n

N'oubliez pas non plus les traductions (cf. ici)

"},{"location":"howto/terminal/","title":"Terminal","text":""},{"location":"howto/terminal/#quel-terminal-utiliser","title":"Quel terminal utiliser ?","text":"

Quel que soit votre configuration, si vous avez r\u00e9ussi \u00e0 installer le projet, il y a de fortes chances que bash existe sur votre ordinateur. Certains d'entre vous utilisent peut-\u00eatre un autre shell, comme zsh.

En effet, bash est bien, il fait le taff ; mais son ergonomie finit par montrer ses limites. C'est pourquoi il existe des shells plus avanc\u00e9s, qui peuvent am\u00e9liorer l'ergonomie, la compl\u00e9tion des commandes, et l'apparence. C'est le cas de zsh. Certains vont m\u00eame plus loin et refont carr\u00e9ment la syntaxe. C'est le cas de nu.

Pour choisir un terminal, demandez-vous juste quel est votre usage du terminal :

Note

Ce ne sont que des suggestions. Le meilleur choix restera toujours celui avec lequel vous \u00eates le plus confortable.

"},{"location":"howto/terminal/#commandes-utiles","title":"Commandes utiles","text":""},{"location":"howto/terminal/#compter-le-nombre-de-lignes-du-projet","title":"Compter le nombre de lignes du projet","text":"bash/zshnu

sudo apt install cloc\ncloc --exclude-dir=doc,env .\n
Ok, c'est de la triche, on installe un package externe. Mais bon, \u00e7a marche, et l'\u00e9quivalent pur bash serait carr\u00e9ment plus moche.

Nombre de lignes, group\u00e9 par fichier :

ls **/*.py | insert linecount { get name | open | lines | length }\n

Nombre de lignes total :

ls **/*.py | insert linecount { get name | open | lines | length } | math sum\n

Vous pouvez aussi exlure les lignes vides et les les lignes de commentaire :

ls **/*.py |\ninsert linecount {\n    get name |\n    open |\n    lines |\n    each { str trim } |\n    filter { |l| not ($l | str starts-with \"#\") } |  # commentaires\n    filter { |l| ($l | str length) > 0 } |  # lignes vides\n    length\n} |\nmath sum\n

"},{"location":"howto/translation/","title":"G\u00e9rer les traductions","text":"

Le code du site est enti\u00e8rement \u00e9crit en anglais, le texte affich\u00e9 aux utilisateurs l'est \u00e9galement. La traduction en fran\u00e7ais se fait ult\u00e9rieurement avec un fichier de traduction. Voici un petit guide rapide pour apprendre \u00e0 s'en servir.

"},{"location":"howto/translation/#dans-le-code-du-logiciel","title":"Dans le code du logiciel","text":"

Imaginons que nous souhaitons afficher \"Hello\" et le traduire en fran\u00e7ais. Voici comment signaler que ce mot doit \u00eatre traduit.

Si le mot est dans le code Python :

from django.utils.translation import gettext as _\n\nhelp_text=_(\"Hello\")\n

Si le mot appara\u00eet dans le template Jinja :

{% trans %}Hello{% endtrans %}\n

Si on est dans un fichier javascript ou typescript :

gettext(\"Hello\");\n
"},{"location":"howto/translation/#generer-le-fichier-djangopo","title":"G\u00e9n\u00e9rer le fichier django.po","text":"

La traduction se fait en trois \u00e9tapes. Il faut d'abord g\u00e9n\u00e9rer un fichier de traductions, l'\u00e9diter et enfin le compiler au format binaire pour qu'il soit lu par le serveur.

# Pour le backend\n./manage.py makemessages \\\n  --locale=fr \\\n  -e py,jinja \\\n  --ignore=.venv \\\n  --ignore=node_modules \\\n  --add-location=file \n\n# Pour le frontend\n./manage.py makemessages \\\n  --locale=fr \\\n  -d djangojs \\\n  -e js,ts \\\n  --ignore=.venv \\\n  --ignore=node_modules \\\n  --ignore=staticfiles/generated \\\n  --add-location=file\n
"},{"location":"howto/translation/#editer-le-fichier-djangopo","title":"\u00c9diter le fichier django.po","text":"
# locale/fr/LC_MESSAGES/django.po\n\n# ...\nmsgid \"Hello\"\nmsgstr \"\" # Ligne \u00e0 modifier\n\n# ...\n

Note

Si les commentaires suivants apparaissent, pensez \u00e0 les supprimer. Ils peuvent g\u00eaner votre traduction.

#, fuzzy\n#| msgid \"Bonjour\"\n
"},{"location":"howto/translation/#generer-le-fichier-djangomo","title":"G\u00e9n\u00e9rer le fichier django.mo","text":"

Il s'agit de la derni\u00e8re \u00e9tape. Un fichier binaire est g\u00e9n\u00e9r\u00e9 \u00e0 partir du fichier django.mo.

./manage.py compilemessages\n

Tip

Pensez \u00e0 red\u00e9marrer le serveur si les traductions ne s'affichent pas

"},{"location":"howto/weekmail/","title":"Modifier le weekmail","text":"

Le site est capable de g\u00e9n\u00e9rer des mails automatiques contenant l\u2019agr\u00e9gation d'articles \u00e9crits par les administrateurs de clubs. Le contenu est ins\u00e9r\u00e9 dans un template standardis\u00e9 et contr\u00f4l\u00e9 directement dans le code. Il arrive r\u00e9guli\u00e8rement que l'\u00e9quipe communication souhaite modifier ce template. Que ce soient les couleurs, l'agencement ou encore la banni\u00e8re ou le footer, voici tout ce qu'il y a \u00e0 savoir sur le fonctionnement du weekmail en commen\u00e7ant par la classe qui le contr\u00f4le.

"},{"location":"howto/weekmail/#modifier-la-banniere-et-le-footer","title":"Modifier la banni\u00e8re et le footer","text":"

Ces \u00e9l\u00e9ments sont contr\u00f4l\u00e9s par les m\u00e9thodes get_banner et get_footer de la classe Weekmail. Les modifier est donc tr\u00e8s simple, il suffit de modifier le contenu de la fonction et de rajouter les nouvelles images dans les statics.

Les images sont \u00e0 ajouter dans core/static/com/img et sont \u00e0 nommer selon le type (banner ou footer), le semestre (Automne ou Printemps) et l'ann\u00e9e. Exemple : weekmail_bannerA18.jpg pour la banni\u00e8re de l'automne 2018.

from django.conf import settings\nfrom django.templatetags.static import static\n\n# S\u00e9lectionnez le fichier de banni\u00e8re pour le weekmail de l'automne 2018\n\ndef get_banner(self):\n    return \"http://\" + settings.SITH_URL + static(\"com/img/weekmail_bannerA18.jpg\")\n

Note

Penser \u00e0 prendre les images au format jpg et \u00e0 les compresser un peu, pour qu'elles soient le plus l\u00e9ger possible, c'est bien mieux pour l'utilisateur final.

Warning

Pensez \u00e0 laisser les anciennes images dans le dossier pour que les anciens weekmails ne soient pas affect\u00e9s par les changements.

"},{"location":"howto/weekmail/#modifier-le-template","title":"Modifier le template","text":"

Il existe deux templates diff\u00e9rents :

Ces deux templates sont respectivement accessibles aux emplacements suivants :

Note

Pour le rendu HTML, pensez \u00e0 utiliser le CSS et le javascript le plus simple possible pour que le rendu se fasse correctement dans les clients mails qui sont souvent capricieux.

Note

Le CSS est inclus statiquement pour que toute modification ult\u00e9rieure de celui-ci n'affecte pas les versions pr\u00e9c\u00e9demment envoy\u00e9es.

Warning

Si vous souhaitez ajouter du contenu, n'oubliez pas de bien inclure ce contenu dans les deux templates.

"},{"location":"reference/accounting/models/","title":"Models","text":""},{"location":"reference/accounting/models/#accounting.models.CurrencyField","title":"CurrencyField(*args, **kwargs)","text":"

Bases: DecimalField

Custom database field used for currency.

Source code in accounting/models.py
def __init__(self, *args, **kwargs):\n    kwargs[\"max_digits\"] = 12\n    kwargs[\"decimal_places\"] = 2\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/accounting/models/#accounting.models.Company","title":"Company","text":"

Bases: Model

"},{"location":"reference/accounting/models/#accounting.models.Company.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID)\n
"},{"location":"reference/accounting/models/#accounting.models.Company.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def can_be_edited_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    return user.memberships.filter(\n        end_date=None, club__role=settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n    ).exists()\n
"},{"location":"reference/accounting/models/#accounting.models.Company.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Check if that object can be viewed by the given user.

Source code in accounting/models.py
def can_be_viewed_by(self, user):\n    \"\"\"Check if that object can be viewed by the given user.\"\"\"\n    return user.memberships.filter(\n        end_date=None, club__role_gte=settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n    ).exists()\n
"},{"location":"reference/accounting/models/#accounting.models.BankAccount","title":"BankAccount","text":"

Bases: Model

"},{"location":"reference/accounting/models/#accounting.models.BankAccount.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    m = self.club.get_membership_for(user)\n    return m is not None and m.role >= settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/models/#accounting.models.ClubAccount","title":"ClubAccount","text":"

Bases: Model

"},{"location":"reference/accounting/models/#accounting.models.ClubAccount.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID)\n
"},{"location":"reference/accounting/models/#accounting.models.ClubAccount.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def can_be_edited_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    m = self.club.get_membership_for(user)\n    return m and m.role == settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/models/#accounting.models.ClubAccount.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Check if that object can be viewed by the given user.

Source code in accounting/models.py
def can_be_viewed_by(self, user):\n    \"\"\"Check if that object can be viewed by the given user.\"\"\"\n    m = self.club.get_membership_for(user)\n    return m and m.role >= settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/models/#accounting.models.GeneralJournal","title":"GeneralJournal","text":"

Bases: Model

Class storing all the operations for a period of time.

"},{"location":"reference/accounting/models/#accounting.models.GeneralJournal.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    return self.club_account.can_be_edited_by(user)\n
"},{"location":"reference/accounting/models/#accounting.models.GeneralJournal.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def can_be_edited_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    return self.club_account.can_be_edited_by(user)\n
"},{"location":"reference/accounting/models/#accounting.models.Operation","title":"Operation","text":"

Bases: Model

An operation is a line in the journal, a debit or a credit.

"},{"location":"reference/accounting/models/#accounting.models.Operation.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    if self.journal.closed:\n        return False\n    m = self.journal.club_account.club.get_membership_for(user)\n    return m is not None and m.role >= settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/models/#accounting.models.Operation.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def can_be_edited_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    if self.journal.closed:\n        return False\n    m = self.journal.club_account.club.get_membership_for(user)\n    return m is not None and m.role == settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/models/#accounting.models.AccountingType","title":"AccountingType","text":"

Bases: Model

Accounting types.

Those are numbers used in accounting to classify operations

"},{"location":"reference/accounting/models/#accounting.models.AccountingType.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID)\n
"},{"location":"reference/accounting/models/#accounting.models.SimplifiedAccountingType","title":"SimplifiedAccountingType","text":"

Bases: Model

Simplified version of AccountingType.

"},{"location":"reference/accounting/models/#accounting.models.Label","title":"Label","text":"

Bases: Model

Label allow a club to sort its operations.

"},{"location":"reference/accounting/views/","title":"Views","text":""},{"location":"reference/accounting/views/#accounting.views.AccountingType","title":"AccountingType","text":"

Bases: Model

Accounting types.

Those are numbers used in accounting to classify operations

"},{"location":"reference/accounting/views/#accounting.views.AccountingType.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID)\n
"},{"location":"reference/accounting/views/#accounting.views.BankAccount","title":"BankAccount","text":"

Bases: Model

"},{"location":"reference/accounting/views/#accounting.views.BankAccount.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    m = self.club.get_membership_for(user)\n    return m is not None and m.role >= settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/views/#accounting.views.ClubAccount","title":"ClubAccount","text":"

Bases: Model

"},{"location":"reference/accounting/views/#accounting.views.ClubAccount.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID)\n
"},{"location":"reference/accounting/views/#accounting.views.ClubAccount.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def can_be_edited_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    m = self.club.get_membership_for(user)\n    return m and m.role == settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/views/#accounting.views.ClubAccount.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Check if that object can be viewed by the given user.

Source code in accounting/models.py
def can_be_viewed_by(self, user):\n    \"\"\"Check if that object can be viewed by the given user.\"\"\"\n    m = self.club.get_membership_for(user)\n    return m and m.role >= settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/views/#accounting.views.Company","title":"Company","text":"

Bases: Model

"},{"location":"reference/accounting/views/#accounting.views.Company.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID)\n
"},{"location":"reference/accounting/views/#accounting.views.Company.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def can_be_edited_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    return user.memberships.filter(\n        end_date=None, club__role=settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n    ).exists()\n
"},{"location":"reference/accounting/views/#accounting.views.Company.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Check if that object can be viewed by the given user.

Source code in accounting/models.py
def can_be_viewed_by(self, user):\n    \"\"\"Check if that object can be viewed by the given user.\"\"\"\n    return user.memberships.filter(\n        end_date=None, club__role_gte=settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n    ).exists()\n
"},{"location":"reference/accounting/views/#accounting.views.GeneralJournal","title":"GeneralJournal","text":"

Bases: Model

Class storing all the operations for a period of time.

"},{"location":"reference/accounting/views/#accounting.views.GeneralJournal.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    return self.club_account.can_be_edited_by(user)\n
"},{"location":"reference/accounting/views/#accounting.views.GeneralJournal.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def can_be_edited_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    return self.club_account.can_be_edited_by(user)\n
"},{"location":"reference/accounting/views/#accounting.views.Label","title":"Label","text":"

Bases: Model

Label allow a club to sort its operations.

"},{"location":"reference/accounting/views/#accounting.views.Operation","title":"Operation","text":"

Bases: Model

An operation is a line in the journal, a debit or a credit.

"},{"location":"reference/accounting/views/#accounting.views.Operation.is_owned_by","title":"is_owned_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def is_owned_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    if self.journal.closed:\n        return False\n    m = self.journal.club_account.club.get_membership_for(user)\n    return m is not None and m.role >= settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/views/#accounting.views.Operation.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in accounting/models.py
def can_be_edited_by(self, user):\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID):\n        return True\n    if self.journal.closed:\n        return False\n    m = self.journal.club_account.club.get_membership_for(user)\n    return m is not None and m.role == settings.SITH_CLUB_ROLES_ID[\"Treasurer\"]\n
"},{"location":"reference/accounting/views/#accounting.views.SimplifiedAccountingType","title":"SimplifiedAccountingType","text":"

Bases: Model

Simplified version of AccountingType.

"},{"location":"reference/accounting/views/#accounting.views.BankAccountListView","title":"BankAccountListView","text":"

Bases: CanViewMixin, ListView

A list view for the admins.

"},{"location":"reference/accounting/views/#accounting.views.SimplifiedAccountingTypeListView","title":"SimplifiedAccountingTypeListView","text":"

Bases: CanViewMixin, ListView

A list view for the admins.

"},{"location":"reference/accounting/views/#accounting.views.SimplifiedAccountingTypeEditView","title":"SimplifiedAccountingTypeEditView","text":"

Bases: CanViewMixin, UpdateView

An edit view for the admins.

"},{"location":"reference/accounting/views/#accounting.views.SimplifiedAccountingTypeCreateView","title":"SimplifiedAccountingTypeCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

Create an accounting type (for the admins).

"},{"location":"reference/accounting/views/#accounting.views.AccountingTypeListView","title":"AccountingTypeListView","text":"

Bases: CanViewMixin, ListView

A list view for the admins.

"},{"location":"reference/accounting/views/#accounting.views.AccountingTypeEditView","title":"AccountingTypeEditView","text":"

Bases: CanViewMixin, UpdateView

An edit view for the admins.

"},{"location":"reference/accounting/views/#accounting.views.AccountingTypeCreateView","title":"AccountingTypeCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

Create an accounting type (for the admins).

"},{"location":"reference/accounting/views/#accounting.views.BankAccountEditView","title":"BankAccountEditView","text":"

Bases: CanViewMixin, UpdateView

An edit view for the admins.

"},{"location":"reference/accounting/views/#accounting.views.BankAccountDetailView","title":"BankAccountDetailView","text":"

Bases: CanViewMixin, DetailView

A detail view, listing every club account.

"},{"location":"reference/accounting/views/#accounting.views.BankAccountCreateView","title":"BankAccountCreateView","text":"

Bases: CanCreateMixin, CreateView

Create a bank account (for the admins).

"},{"location":"reference/accounting/views/#accounting.views.BankAccountDeleteView","title":"BankAccountDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

Delete a bank account (for the admins).

"},{"location":"reference/accounting/views/#accounting.views.ClubAccountEditView","title":"ClubAccountEditView","text":"

Bases: CanViewMixin, UpdateView

An edit view for the admins.

"},{"location":"reference/accounting/views/#accounting.views.ClubAccountDetailView","title":"ClubAccountDetailView","text":"

Bases: CanViewMixin, DetailView

A detail view, listing every journal.

"},{"location":"reference/accounting/views/#accounting.views.ClubAccountCreateView","title":"ClubAccountCreateView","text":"

Bases: CanCreateMixin, CreateView

Create a club account (for the admins).

"},{"location":"reference/accounting/views/#accounting.views.ClubAccountDeleteView","title":"ClubAccountDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

Delete a club account (for the admins).

"},{"location":"reference/accounting/views/#accounting.views.JournalTabsMixin","title":"JournalTabsMixin","text":"

Bases: TabedViewMixin

"},{"location":"reference/accounting/views/#accounting.views.JournalCreateView","title":"JournalCreateView","text":"

Bases: CanCreateMixin, CreateView

Create a general journal.

"},{"location":"reference/accounting/views/#accounting.views.JournalDetailView","title":"JournalDetailView","text":"

Bases: JournalTabsMixin, CanViewMixin, DetailView

A detail view, listing every operation.

"},{"location":"reference/accounting/views/#accounting.views.JournalEditView","title":"JournalEditView","text":"

Bases: CanEditMixin, UpdateView

Update a general journal.

"},{"location":"reference/accounting/views/#accounting.views.JournalDeleteView","title":"JournalDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

Delete a club account (for the admins).

"},{"location":"reference/accounting/views/#accounting.views.OperationForm","title":"OperationForm(*args, **kwargs)","text":"

Bases: ModelForm

Source code in accounting/views.py
def __init__(self, *args, **kwargs):\n    club_account = kwargs.pop(\"club_account\", None)\n    super().__init__(*args, **kwargs)\n    if club_account:\n        self.fields[\"label\"].queryset = club_account.labels.order_by(\"name\").all()\n    if self.instance.target_type == \"USER\":\n        self.fields[\"user\"].initial = self.instance.target_id\n    elif self.instance.target_type == \"ACCOUNT\":\n        self.fields[\"club_account\"].initial = self.instance.target_id\n    elif self.instance.target_type == \"CLUB\":\n        self.fields[\"club\"].initial = self.instance.target_id\n    elif self.instance.target_type == \"COMPANY\":\n        self.fields[\"company\"].initial = self.instance.target_id\n
"},{"location":"reference/accounting/views/#accounting.views.OperationCreateView","title":"OperationCreateView","text":"

Bases: CanCreateMixin, CreateView

Create an operation.

"},{"location":"reference/accounting/views/#accounting.views.OperationCreateView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add journal to the context.

Source code in accounting/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add journal to the context.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    if self.journal:\n        kwargs[\"object\"] = self.journal\n    return kwargs\n
"},{"location":"reference/accounting/views/#accounting.views.OperationEditView","title":"OperationEditView","text":"

Bases: CanEditMixin, UpdateView

An edit view, working as detail for the moment.

"},{"location":"reference/accounting/views/#accounting.views.OperationEditView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add journal to the context.

Source code in accounting/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add journal to the context.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"object\"] = self.object.journal\n    return kwargs\n
"},{"location":"reference/accounting/views/#accounting.views.OperationPDFView","title":"OperationPDFView","text":"

Bases: CanViewMixin, DetailView

Display the PDF of a given operation.

"},{"location":"reference/accounting/views/#accounting.views.JournalNatureStatementView","title":"JournalNatureStatementView","text":"

Bases: JournalTabsMixin, CanViewMixin, DetailView

Display a statement sorted by labels.

"},{"location":"reference/accounting/views/#accounting.views.JournalNatureStatementView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add infos to the context.

Source code in accounting/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add infos to the context.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"statement\"] = self.big_statement()\n    return kwargs\n
"},{"location":"reference/accounting/views/#accounting.views.JournalPersonStatementView","title":"JournalPersonStatementView","text":"

Bases: JournalTabsMixin, CanViewMixin, DetailView

Calculate a dictionary with operation target and sum of operations.

"},{"location":"reference/accounting/views/#accounting.views.JournalPersonStatementView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add journal to the context.

Source code in accounting/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add journal to the context.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"credit_statement\"] = self.statement(\"CREDIT\")\n    kwargs[\"debit_statement\"] = self.statement(\"DEBIT\")\n    kwargs[\"total_credit\"] = self.total(\"CREDIT\")\n    kwargs[\"total_debit\"] = self.total(\"DEBIT\")\n    return kwargs\n
"},{"location":"reference/accounting/views/#accounting.views.JournalAccountingStatementView","title":"JournalAccountingStatementView","text":"

Bases: JournalTabsMixin, CanViewMixin, DetailView

Calculate a dictionary with operation type and sum of operations.

"},{"location":"reference/accounting/views/#accounting.views.JournalAccountingStatementView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add journal to the context.

Source code in accounting/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add journal to the context.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"statement\"] = self.statement()\n    return kwargs\n
"},{"location":"reference/accounting/views/#accounting.views.CompanyListView","title":"CompanyListView","text":"

Bases: CanViewMixin, ListView

"},{"location":"reference/accounting/views/#accounting.views.CompanyCreateView","title":"CompanyCreateView","text":"

Bases: CanCreateMixin, CreateView

Create a company.

"},{"location":"reference/accounting/views/#accounting.views.CompanyEditView","title":"CompanyEditView","text":"

Bases: CanCreateMixin, UpdateView

Edit a company.

"},{"location":"reference/accounting/views/#accounting.views.LabelListView","title":"LabelListView","text":"

Bases: CanViewMixin, DetailView

"},{"location":"reference/accounting/views/#accounting.views.LabelCreateView","title":"LabelCreateView","text":"

Bases: CanCreateMixin, CreateView

"},{"location":"reference/accounting/views/#accounting.views.LabelEditView","title":"LabelEditView","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/accounting/views/#accounting.views.LabelDeleteView","title":"LabelDeleteView","text":"

Bases: CanEditMixin, DeleteView

"},{"location":"reference/accounting/views/#accounting.views.CloseCustomerAccountForm","title":"CloseCustomerAccountForm","text":"

Bases: Form

"},{"location":"reference/accounting/views/#accounting.views.RefoundAccountView","title":"RefoundAccountView","text":"

Bases: FormView

Create a selling with the same amount than the current user money.

"},{"location":"reference/antispam/forms/","title":"Forms","text":""},{"location":"reference/antispam/forms/#antispam.forms.ToxicDomain","title":"ToxicDomain","text":"

Bases: Model

Domain marked as spam in public databases

"},{"location":"reference/antispam/forms/#antispam.forms.AntiSpamEmailField","title":"AntiSpamEmailField","text":"

Bases: EmailField

An email field that email addresses with a known toxic domain.

"},{"location":"reference/antispam/models/","title":"Models","text":""},{"location":"reference/antispam/models/#antispam.models.ToxicDomain","title":"ToxicDomain","text":"

Bases: Model

Domain marked as spam in public databases

"},{"location":"reference/club/models/","title":"Models","text":""},{"location":"reference/club/models/#club.models.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/club/models/#club.models.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/club/models/#club.models.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/club/models/#club.models.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/club/models/#club.models.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/club/models/#club.models.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/club/models/#club.models.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/club/models/#club.models.MembershipQuerySet","title":"MembershipQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/club/models/#club.models.MembershipQuerySet.ongoing","title":"ongoing()","text":"

Filter all memberships which are not finished yet.

Source code in club/models.py
def ongoing(self) -> Self:\n    \"\"\"Filter all memberships which are not finished yet.\"\"\"\n    return self.filter(Q(end_date=None) | Q(end_date__gt=localdate()))\n
"},{"location":"reference/club/models/#club.models.MembershipQuerySet.board","title":"board()","text":"

Filter all memberships where the user is/was in the board.

Be aware that users who were in the board in the past are included, even if there are no more members.

If you want to get the users who are currently in the board, mind combining this with the :meth:ongoing queryset method

Source code in club/models.py
def board(self) -> Self:\n    \"\"\"Filter all memberships where the user is/was in the board.\n\n    Be aware that users who were in the board in the past\n    are included, even if there are no more members.\n\n    If you want to get the users who are currently in the board,\n    mind combining this with the :meth:`ongoing` queryset method\n    \"\"\"\n    return self.filter(role__gt=settings.SITH_MAXIMUM_FREE_ROLE)\n
"},{"location":"reference/club/models/#club.models.MembershipQuerySet.update","title":"update(**kwargs)","text":"

Refresh the cache and edit group ownership.

Update the cache, when necessary, remove users from club groups they are no more in and add them in the club groups they should be in.

Be aware that this adds three db queries : one to retrieve the updated memberships, one to perform group removal and one to perform group attribution.

Source code in club/models.py
def update(self, **kwargs) -> int:\n    \"\"\"Refresh the cache and edit group ownership.\n\n    Update the cache, when necessary, remove\n    users from club groups they are no more in\n    and add them in the club groups they should be in.\n\n    Be aware that this adds three db queries :\n    one to retrieve the updated memberships,\n    one to perform group removal and one to perform\n    group attribution.\n    \"\"\"\n    nb_rows = super().update(**kwargs)\n    if nb_rows == 0:\n        # if no row was affected, no need to refresh the cache\n        return 0\n\n    cache_memberships = {}\n    memberships = set(self.select_related(\"club\"))\n    # delete all User-Group relations and recreate the necessary ones\n    # It's more concise to write and more reliable\n    Membership._remove_club_groups(memberships)\n    Membership._add_club_groups(memberships)\n    for member in memberships:\n        cache_key = f\"membership_{member.club_id}_{member.user_id}\"\n        if member.end_date is None:\n            cache_memberships[cache_key] = member\n        else:\n            cache_memberships[cache_key] = \"not_member\"\n    cache.set_many(cache_memberships)\n    return nb_rows\n
"},{"location":"reference/club/models/#club.models.MembershipQuerySet.delete","title":"delete()","text":"

Work just like the default Django's delete() method, but add a cache invalidation for the elements of the queryset before the deletion, and a removal of the user from the club groups.

Be aware that this adds some db queries :

Source code in club/models.py
def delete(self) -> tuple[int, dict[str, int]]:\n    \"\"\"Work just like the default Django's delete() method,\n    but add a cache invalidation for the elements of the queryset\n    before the deletion,\n    and a removal of the user from the club groups.\n\n    Be aware that this adds some db queries :\n\n    - 1 to retrieve the deleted elements in order to perform\n      post-delete operations.\n      As we can't know if a delete will affect rows or not,\n      this query will always happen\n    - 1 query to remove the users from the club groups.\n      If the delete operation affected no row,\n      this query won't happen.\n    \"\"\"\n    memberships = set(self.all())\n    nb_rows, rows_counts = super().delete()\n    if nb_rows > 0:\n        Membership._remove_club_groups(memberships)\n        cache.set_many(\n            {\n                f\"membership_{m.club_id}_{m.user_id}\": \"not_member\"\n                for m in memberships\n            }\n        )\n    return nb_rows, rows_counts\n
"},{"location":"reference/club/models/#club.models.Membership","title":"Membership","text":"

Bases: Model

The Membership class makes the connection between User and Clubs.

Both Users and Clubs can have many Membership objects

A User is currently member of all the Clubs where its Membership has an end_date set to null/None. Otherwise, it's a past membership kept because it can be very useful to see who was in which Club in the past.

"},{"location":"reference/club/models/#club.models.Membership.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/club/models/#club.models.Membership.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_root or user.is_board_member:\n        return True\n    membership = self.club.get_membership_for(user)\n    return membership is not None and membership.role >= self.role\n
"},{"location":"reference/club/models/#club.models.Mailing","title":"Mailing","text":"

Bases: Model

A Mailing list for a club.

Warning

Remember that mailing lists should be validated by UTBM.

"},{"location":"reference/club/models/#club.models.MailingSubscription","title":"MailingSubscription","text":"

Bases: Model

Link between user and mailing list.

"},{"location":"reference/club/models/#club.models.get_default_owner_group","title":"get_default_owner_group()","text":"Source code in club/models.py
def get_default_owner_group():\n    return settings.SITH_GROUP_ROOT_ID\n
"},{"location":"reference/club/views/","title":"Views","text":""},{"location":"reference/club/views/#club.views.ClubEditForm","title":"ClubEditForm(*args, **kwargs)","text":"

Bases: ModelForm

Source code in club/forms.py
def __init__(self, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n    self.fields[\"short_description\"].widget = forms.Textarea()\n
"},{"location":"reference/club/views/#club.views.ClubMemberForm","title":"ClubMemberForm(*args, **kwargs)","text":"

Bases: Form

Form handling the members of a club.

Source code in club/forms.py
def __init__(self, *args, **kwargs):\n    self.club = kwargs.pop(\"club\")\n    self.request_user = kwargs.pop(\"request_user\")\n    self.club_members = kwargs.pop(\"club_members\", None)\n    if not self.club_members:\n        self.club_members = (\n            self.club.members.filter(end_date=None).order_by(\"-role\").all()\n        )\n    self.request_user_membership = self.club.get_membership_for(self.request_user)\n    super().__init__(*args, **kwargs)\n\n    # Using a ModelForm binds too much the form with the model and we don't want that\n    # We want the view to process the model creation since they are multiple users\n    # We also want the form to handle bulk deletion\n    self.fields.update(\n        forms.fields_for_model(\n            Membership,\n            fields=(\"role\", \"start_date\", \"description\"),\n            widgets={\"start_date\": SelectDate},\n        )\n    )\n\n    # Role is required only if users is specified\n    self.fields[\"role\"].required = False\n\n    # Start date and description are never really required\n    self.fields[\"start_date\"].required = False\n    self.fields[\"description\"].required = False\n\n    self.fields[\"users_old\"] = forms.ModelMultipleChoiceField(\n        User.objects.filter(\n            id__in=[\n                ms.user.id\n                for ms in self.club_members\n                if ms.can_be_edited_by(self.request_user)\n            ]\n        ).all(),\n        label=_(\"Mark as old\"),\n        required=False,\n        widget=forms.CheckboxSelectMultiple,\n    )\n    if not self.request_user.is_root:\n        self.fields.pop(\"start_date\")\n
"},{"location":"reference/club/views/#club.views.ClubMemberForm.clean_users","title":"clean_users()","text":"

Check that the user is not trying to add an user already in the club.

Also check that the user is valid and has a valid subscription.

Source code in club/forms.py
def clean_users(self):\n    \"\"\"Check that the user is not trying to add an user already in the club.\n\n    Also check that the user is valid and has a valid subscription.\n    \"\"\"\n    cleaned_data = super().clean()\n    users = []\n    for user in cleaned_data[\"users\"]:\n        if not user.is_subscribed:\n            raise forms.ValidationError(\n                _(\"User must be subscriber to take part to a club\"), code=\"invalid\"\n            )\n        if self.club.get_membership_for(user):\n            raise forms.ValidationError(\n                _(\"You can not add the same user twice\"), code=\"invalid\"\n            )\n        users.append(user)\n    return users\n
"},{"location":"reference/club/views/#club.views.ClubMemberForm.clean","title":"clean()","text":"

Check user rights for adding an user.

Source code in club/forms.py
def clean(self):\n    \"\"\"Check user rights for adding an user.\"\"\"\n    cleaned_data = super().clean()\n\n    if \"start_date\" in cleaned_data and not cleaned_data[\"start_date\"]:\n        # Drop start_date if allowed to edition but not specified\n        cleaned_data.pop(\"start_date\")\n\n    if not cleaned_data.get(\"users\"):\n        # No user to add equals no check needed\n        return cleaned_data\n\n    if cleaned_data.get(\"role\", \"\") == \"\":\n        # Role is required if users exists\n        self.add_error(\"role\", _(\"You should specify a role\"))\n        return cleaned_data\n\n    request_user = self.request_user\n    membership = self.request_user_membership\n    if not (\n        cleaned_data[\"role\"] <= settings.SITH_MAXIMUM_FREE_ROLE\n        or (membership is not None and membership.role >= cleaned_data[\"role\"])\n        or request_user.is_board_member\n        or request_user.is_root\n    ):\n        raise forms.ValidationError(_(\"You do not have the permission to do that\"))\n    return cleaned_data\n
"},{"location":"reference/club/views/#club.views.MailingForm","title":"MailingForm(club_id, user_id, mailings, *args, **kwargs)","text":"

Bases: Form

Form handling mailing lists right.

Source code in club/forms.py
def __init__(self, club_id, user_id, mailings, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n\n    self.fields[\"action\"] = forms.TypedChoiceField(\n        choices=(\n            (self.ACTION_NEW_MAILING, _(\"New Mailing\")),\n            (self.ACTION_NEW_SUBSCRIPTION, _(\"Subscribe\")),\n            (self.ACTION_REMOVE_SUBSCRIPTION, _(\"Remove\")),\n        ),\n        coerce=int,\n        label=_(\"Action\"),\n        initial=1,\n        required=True,\n        widget=forms.HiddenInput(),\n    )\n\n    # Generate bulk removal forms, they are never required\n    for mailing in mailings:\n        self.fields[\"removal_\" + str(mailing.id)] = forms.ModelMultipleChoiceField(\n            mailing.subscriptions.all(),\n            label=_(\"Remove\"),\n            required=False,\n            widget=forms.CheckboxSelectMultiple,\n        )\n\n    # Include fields for handling mailing creation\n    mailing_fields = (\"email\",)\n    self.fields.update(forms.fields_for_model(Mailing, fields=mailing_fields))\n    for field in mailing_fields:\n        self.fields[\"mailing_\" + field] = self.fields.pop(field)\n        self.fields[\"mailing_\" + field].required = False\n\n    # Include fields for handling subscription creation\n    subscription_fields = (\"mailing\", \"email\")\n    self.fields.update(\n        forms.fields_for_model(MailingSubscription, fields=subscription_fields)\n    )\n    for field in subscription_fields:\n        self.fields[\"subscription_\" + field] = self.fields.pop(field)\n        self.fields[\"subscription_\" + field].required = False\n\n    self.fields[\"subscription_mailing\"].queryset = Mailing.objects.filter(\n        club__id=club_id, is_moderated=True\n    )\n
"},{"location":"reference/club/views/#club.views.MailingForm.check_required","title":"check_required(cleaned_data, field)","text":"

If the given field doesn't exist or has no value, add a required error on it.

Source code in club/forms.py
def check_required(self, cleaned_data, field):\n    \"\"\"If the given field doesn't exist or has no value, add a required error on it.\"\"\"\n    if not cleaned_data.get(field, None):\n        self.add_error(field, _(\"This field is required\"))\n
"},{"location":"reference/club/views/#club.views.MailingForm.clean_subscription_users","title":"clean_subscription_users()","text":"

Convert given users into real users and check their validity.

Source code in club/forms.py
def clean_subscription_users(self):\n    \"\"\"Convert given users into real users and check their validity.\"\"\"\n    cleaned_data = super().clean()\n    users = []\n    for user in cleaned_data[\"subscription_users\"]:\n        if not user.email:\n            raise forms.ValidationError(\n                _(\"One of the selected users doesn't have an email address\"),\n                code=\"invalid\",\n            )\n        users.append(user)\n    return users\n
"},{"location":"reference/club/views/#club.views.SellingsForm","title":"SellingsForm(club, *args, **kwargs)","text":"

Bases: Form

Source code in club/forms.py
def __init__(self, club, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n    self.fields[\"products\"] = forms.ModelMultipleChoiceField(\n        club.products.order_by(\"name\").filter(archived=False).all(),\n        label=_(\"Products\"),\n        required=False,\n    )\n    self.fields[\"archived_products\"] = forms.ModelMultipleChoiceField(\n        club.products.order_by(\"name\").filter(archived=True).all(),\n        label=_(\"Archived products\"),\n        required=False,\n    )\n
"},{"location":"reference/club/views/#club.views.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/club/views/#club.views.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/club/views/#club.views.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/club/views/#club.views.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/club/views/#club.views.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/club/views/#club.views.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/club/views/#club.views.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/club/views/#club.views.Mailing","title":"Mailing","text":"

Bases: Model

A Mailing list for a club.

Warning

Remember that mailing lists should be validated by UTBM.

"},{"location":"reference/club/views/#club.views.MailingSubscription","title":"MailingSubscription","text":"

Bases: Model

Link between user and mailing list.

"},{"location":"reference/club/views/#club.views.Membership","title":"Membership","text":"

Bases: Model

The Membership class makes the connection between User and Clubs.

Both Users and Clubs can have many Membership objects

A User is currently member of all the Clubs where its Membership has an end_date set to null/None. Otherwise, it's a past membership kept because it can be very useful to see who was in which Club in the past.

"},{"location":"reference/club/views/#club.views.Membership.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/club/views/#club.views.Membership.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Check if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Check if that object can be edited by the given user.\"\"\"\n    if user.is_root or user.is_board_member:\n        return True\n    membership = self.club.get_membership_for(user)\n    return membership is not None and membership.role >= self.role\n
"},{"location":"reference/club/views/#club.views.ClubTabsMixin","title":"ClubTabsMixin","text":"

Bases: TabedViewMixin

"},{"location":"reference/club/views/#club.views.ClubListView","title":"ClubListView","text":"

Bases: ListView

List the Clubs.

"},{"location":"reference/club/views/#club.views.ClubView","title":"ClubView","text":"

Bases: ClubTabsMixin, DetailView

Front page of a Club.

"},{"location":"reference/club/views/#club.views.ClubRevView","title":"ClubRevView","text":"

Bases: ClubView

Display a specific page revision.

"},{"location":"reference/club/views/#club.views.ClubPageEditView","title":"ClubPageEditView","text":"

Bases: ClubTabsMixin, PageEditViewBase

"},{"location":"reference/club/views/#club.views.ClubPageHistView","title":"ClubPageHistView","text":"

Bases: ClubTabsMixin, CanViewMixin, DetailView

Modification hostory of the page.

"},{"location":"reference/club/views/#club.views.ClubToolsView","title":"ClubToolsView","text":"

Bases: ClubTabsMixin, CanEditMixin, DetailView

Tools page of a Club.

"},{"location":"reference/club/views/#club.views.ClubMembersView","title":"ClubMembersView","text":"

Bases: ClubTabsMixin, CanViewMixin, DetailFormView

View of a club's members.

"},{"location":"reference/club/views/#club.views.ClubMembersView.form_valid","title":"form_valid(form)","text":"

Check user rights.

Source code in club/views.py
def form_valid(self, form):\n    \"\"\"Check user rights.\"\"\"\n    resp = super().form_valid(form)\n\n    data = form.clean()\n    users = data.pop(\"users\", [])\n    users_old = data.pop(\"users_old\", [])\n    for user in users:\n        Membership(club=self.get_object(), user=user, **data).save()\n    for user in users_old:\n        membership = self.get_object().get_membership_for(user)\n        membership.end_date = timezone.now()\n        membership.save()\n    return resp\n
"},{"location":"reference/club/views/#club.views.ClubOldMembersView","title":"ClubOldMembersView","text":"

Bases: ClubTabsMixin, CanViewMixin, DetailView

Old members of a club.

"},{"location":"reference/club/views/#club.views.ClubSellingView","title":"ClubSellingView","text":"

Bases: ClubTabsMixin, CanEditMixin, DetailFormView

Sellings of a club.

"},{"location":"reference/club/views/#club.views.ClubSellingCSVView","title":"ClubSellingCSVView","text":"

Bases: ClubSellingView

Generate sellings in csv for a given period.

"},{"location":"reference/club/views/#club.views.ClubSellingCSVView.StreamWriter","title":"StreamWriter","text":"

Implements a file-like interface for streaming the CSV.

"},{"location":"reference/club/views/#club.views.ClubSellingCSVView.StreamWriter.write","title":"write(value)","text":"

Write the value by returning it, instead of storing in a buffer.

Source code in club/views.py
def write(self, value):\n    \"\"\"Write the value by returning it, instead of storing in a buffer.\"\"\"\n    return value\n
"},{"location":"reference/club/views/#club.views.ClubEditView","title":"ClubEditView","text":"

Bases: ClubTabsMixin, CanEditMixin, UpdateView

Edit a Club's main informations (for the club's members).

"},{"location":"reference/club/views/#club.views.ClubEditPropView","title":"ClubEditPropView","text":"

Bases: ClubTabsMixin, CanEditPropMixin, UpdateView

Edit the properties of a Club object (for the Sith admins).

"},{"location":"reference/club/views/#club.views.ClubCreateView","title":"ClubCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

Create a club (for the Sith admin).

"},{"location":"reference/club/views/#club.views.MembershipSetOldView","title":"MembershipSetOldView","text":"

Bases: CanEditMixin, DetailView

Set a membership as beeing old.

"},{"location":"reference/club/views/#club.views.MembershipDeleteView","title":"MembershipDeleteView","text":"

Bases: PermissionRequiredMixin, DeleteView

Delete a membership (for admins only).

"},{"location":"reference/club/views/#club.views.ClubStatView","title":"ClubStatView","text":"

Bases: TemplateView

"},{"location":"reference/club/views/#club.views.ClubMailingView","title":"ClubMailingView","text":"

Bases: ClubTabsMixin, CanEditMixin, DetailFormView

A list of mailing for a given club.

"},{"location":"reference/club/views/#club.views.ClubMailingView.add_new_mailing","title":"add_new_mailing(cleaned_data)","text":"

Create a new mailing list from the form.

Source code in club/views.py
def add_new_mailing(self, cleaned_data) -> ValidationError | None:\n    \"\"\"Create a new mailing list from the form.\"\"\"\n    mailing = Mailing(\n        club=self.get_object(),\n        email=cleaned_data[\"mailing_email\"],\n        moderator=self.request.user,\n        is_moderated=False,\n    )\n    try:\n        mailing.clean()\n    except ValidationError as validation_error:\n        return validation_error\n    mailing.save()\n    return None\n
"},{"location":"reference/club/views/#club.views.ClubMailingView.add_new_subscription","title":"add_new_subscription(cleaned_data)","text":"

Add mailing subscriptions for each user given and/or for the specified email in form.

Source code in club/views.py
def add_new_subscription(self, cleaned_data) -> ValidationError | None:\n    \"\"\"Add mailing subscriptions for each user given and/or for the specified email in form.\"\"\"\n    users_to_save = []\n\n    for user in cleaned_data[\"subscription_users\"]:\n        sub = MailingSubscription(\n            mailing=cleaned_data[\"subscription_mailing\"], user=user\n        )\n        try:\n            sub.clean()\n        except ValidationError as validation_error:\n            return validation_error\n\n        sub.save()\n        users_to_save.append(sub)\n\n    if cleaned_data[\"subscription_email\"]:\n        sub = MailingSubscription(\n            mailing=cleaned_data[\"subscription_mailing\"],\n            email=cleaned_data[\"subscription_email\"],\n        )\n\n    try:\n        sub.clean()\n    except ValidationError as validation_error:\n        return validation_error\n    sub.save()\n\n    # Save users after we are sure there is no error\n    for user in users_to_save:\n        user.save()\n\n    return None\n
"},{"location":"reference/club/views/#club.views.ClubMailingView.remove_subscription","title":"remove_subscription(cleaned_data)","text":"

Remove specified users from a mailing list.

Source code in club/views.py
def remove_subscription(self, cleaned_data):\n    \"\"\"Remove specified users from a mailing list.\"\"\"\n    fields = [\n        val for key, val in cleaned_data.items() if key.startswith(\"removal_\")\n    ]\n    for field in fields:\n        for sub in field:\n            sub.delete()\n
"},{"location":"reference/club/views/#club.views.MailingDeleteView","title":"MailingDeleteView","text":"

Bases: CanEditMixin, DeleteView

"},{"location":"reference/club/views/#club.views.MailingSubscriptionDeleteView","title":"MailingSubscriptionDeleteView","text":"

Bases: CanEditMixin, DeleteView

"},{"location":"reference/club/views/#club.views.MailingAutoGenerationView","title":"MailingAutoGenerationView","text":"

Bases: View

"},{"location":"reference/club/views/#club.views.PosterListView","title":"PosterListView","text":"

Bases: ClubTabsMixin, PosterListBaseView, CanViewMixin

List communication posters.

"},{"location":"reference/club/views/#club.views.PosterCreateView","title":"PosterCreateView","text":"

Bases: PosterCreateBaseView, CanCreateMixin

Create communication poster.

"},{"location":"reference/club/views/#club.views.PosterEditView","title":"PosterEditView","text":"

Bases: ClubTabsMixin, PosterEditBaseView, CanEditMixin

Edit communication poster.

"},{"location":"reference/club/views/#club.views.PosterDeleteView","title":"PosterDeleteView","text":"

Bases: PosterDeleteBaseView, ClubTabsMixin, CanEditMixin

Delete communication poster.

"},{"location":"reference/com/models/","title":"Models","text":""},{"location":"reference/com/models/#com.models.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/com/models/#com.models.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/com/models/#com.models.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/com/models/#com.models.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/com/models/#com.models.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/com/models/#com.models.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/com/models/#com.models.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/com/models/#com.models.Sith","title":"Sith","text":"

Bases: Model

A one instance class storing all the modifiable infos.

"},{"location":"reference/com/models/#com.models.NewsQuerySet","title":"NewsQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/com/models/#com.models.NewsQuerySet.viewable_by","title":"viewable_by(user)","text":"

Filter news that the given user can view.

If the user has the com.view_unmoderated_news permission, all news are viewable. Else the viewable news are those that are either moderated or authored by the user.

Source code in com/models.py
def viewable_by(self, user: User) -> Self:\n    \"\"\"Filter news that the given user can view.\n\n    If the user has the `com.view_unmoderated_news` permission,\n    all news are viewable.\n    Else the viewable news are those that are either moderated\n    or authored by the user.\n    \"\"\"\n    if user.has_perm(\"com.view_unmoderated_news\"):\n        return self\n    q_filter = Q(is_moderated=True)\n    if user.is_authenticated:\n        q_filter |= Q(author_id=user.id)\n    return self.filter(q_filter)\n
"},{"location":"reference/com/models/#com.models.News","title":"News","text":"

Bases: Model

News about club events.

"},{"location":"reference/com/models/#com.models.NewsDate","title":"NewsDate","text":"

Bases: Model

A date associated with news.

A News can have multiple dates, for example if it is a recurring event.

"},{"location":"reference/com/models/#com.models.Weekmail","title":"Weekmail","text":"

Bases: Model

The weekmail class.

:ivar title: Title of the weekmail :ivar intro: Introduction of the weekmail :ivar joke: Joke of the week :ivar protip: Tip of the week :ivar conclusion: Conclusion of the weekmail :ivar sent: Track if the weekmail has been sent

"},{"location":"reference/com/models/#com.models.Weekmail.send","title":"send()","text":"

Send the weekmail to all users with the receive weekmail option opt-in.

Also send the weekmail to the mailing list in settings.SITH_COM_EMAIL.

Source code in com/models.py
def send(self):\n    \"\"\"Send the weekmail to all users with the receive weekmail option opt-in.\n\n    Also send the weekmail to the mailing list in settings.SITH_COM_EMAIL.\n    \"\"\"\n    dest = [\n        i[0]\n        for i in Preferences.objects.filter(receive_weekmail=True).values_list(\n            \"user__email\"\n        )\n    ]\n    with transaction.atomic():\n        email = EmailMultiAlternatives(\n            subject=self.title,\n            body=self.render_text(),\n            from_email=settings.SITH_COM_EMAIL,\n            to=Sith.objects.first().weekmail_destinations.split(\" \"),\n            bcc=dest,\n        )\n        email.attach_alternative(self.render_html(), \"text/html\")\n        email.send()\n        self.sent = True\n        self.save()\n        Weekmail().save()\n
"},{"location":"reference/com/models/#com.models.Weekmail.render_text","title":"render_text()","text":"

Renders a pure text version of the mail for readers without HTML support.

Source code in com/models.py
def render_text(self):\n    \"\"\"Renders a pure text version of the mail for readers without HTML support.\"\"\"\n    return render(\n        None, \"com/weekmail_renderer_text.jinja\", context={\"weekmail\": self}\n    ).content.decode(\"utf-8\")\n
"},{"location":"reference/com/models/#com.models.Weekmail.render_html","title":"render_html()","text":"

Renders an HTML version of the mail with images and fancy CSS.

Source code in com/models.py
def render_html(self):\n    \"\"\"Renders an HTML version of the mail with images and fancy CSS.\"\"\"\n    return render(\n        None, \"com/weekmail_renderer_html.jinja\", context={\"weekmail\": self}\n    ).content.decode(\"utf-8\")\n
"},{"location":"reference/com/models/#com.models.Weekmail.get_banner","title":"get_banner()","text":"

Return an absolute link to the banner.

Source code in com/models.py
def get_banner(self):\n    \"\"\"Return an absolute link to the banner.\"\"\"\n    return (\n        \"http://\" + settings.SITH_URL + static(\"com/img/weekmail_bannerV2P22.png\")\n    )\n
"},{"location":"reference/com/models/#com.models.Weekmail.get_footer","title":"get_footer()","text":"

Return an absolute link to the footer.

Source code in com/models.py
def get_footer(self):\n    \"\"\"Return an absolute link to the footer.\"\"\"\n    return \"http://\" + settings.SITH_URL + static(\"com/img/weekmail_footerP22.png\")\n
"},{"location":"reference/com/models/#com.models.WeekmailArticle","title":"WeekmailArticle","text":"

Bases: Model

"},{"location":"reference/com/models/#com.models.Screen","title":"Screen","text":"

Bases: Model

"},{"location":"reference/com/models/#com.models.Poster","title":"Poster","text":"

Bases: Model

"},{"location":"reference/com/models/#com.models.news_notification_callback","title":"news_notification_callback(notif)","text":"Source code in com/models.py
def news_notification_callback(notif):\n    count = News.objects.filter(\n        dates__start_date__gt=timezone.now(), is_moderated=False\n    ).count()\n    if count:\n        notif.viewed = False\n        notif.param = str(count)\n        notif.date = timezone.now()\n    else:\n        notif.viewed = True\n
"},{"location":"reference/com/views/","title":"Views","text":""},{"location":"reference/com/views/#com.views.sith","title":"sith = Sith.objects.first module-attribute","text":""},{"location":"reference/com/views/#com.views.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/com/views/#com.views.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/com/views/#com.views.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/com/views/#com.views.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/com/views/#com.views.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/com/views/#com.views.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/com/views/#com.views.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/com/views/#com.views.Mailing","title":"Mailing","text":"

Bases: Model

A Mailing list for a club.

Warning

Remember that mailing lists should be validated by UTBM.

"},{"location":"reference/com/views/#com.views.IcsCalendar","title":"IcsCalendar","text":""},{"location":"reference/com/views/#com.views.NewsDateForm","title":"NewsDateForm(*args, **kwargs)","text":"

Bases: ModelForm

Form to select the dates of an event.

Source code in com/forms.py
def __init__(self, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n    self.label_suffix = \"\"\n
"},{"location":"reference/com/views/#com.views.NewsDateForm.get_occurrences","title":"get_occurrences(number) classmethod","text":"

Find the occurrence choice corresponding to numeric number of occurrences.

Source code in com/forms.py
@classmethod\ndef get_occurrences(cls, number: int) -> tuple[str, str] | None:\n    \"\"\"Find the occurrence choice corresponding to numeric number of occurrences.\"\"\"\n    if number < 2:\n        # If only 0 or 1 date, there cannot be weekly events\n        return None\n    # occurrences have all a numeric value, except \"SEMESTER_END\"\n    str_num = str(number)\n    occurrences = next((c for c in cls.occurrence_choices if c[0] == str_num), None)\n    if occurrences:\n        return occurrences\n    return next((c for c in cls.occurrence_choices if c[0] == \"SEMESTER_END\"), None)\n
"},{"location":"reference/com/views/#com.views.NewsForm","title":"NewsForm(*args, author, date_form, **kwargs)","text":"

Bases: ModelForm

Form to create or edit news.

Source code in com/forms.py
def __init__(self, *args, author: User, date_form: NewsDateForm, **kwargs):\n    super().__init__(*args, **kwargs)\n    self.author = author\n    self.date_form = date_form\n    self.label_suffix = \"\"\n    # if the author is an admin, he/she can choose any club,\n    # otherwise, only clubs for which he/she is a board member can be selected\n    if author.is_root or author.is_com_admin:\n        self.fields[\"club\"] = forms.ModelChoiceField(\n            queryset=Club.objects.all(), widget=AutoCompleteSelectClub\n        )\n    else:\n        active_memberships = author.memberships.board().ongoing()\n        self.fields[\"club\"] = forms.ModelChoiceField(\n            queryset=Club.objects.filter(\n                Exists(active_memberships.filter(club=OuterRef(\"pk\")))\n            )\n        )\n
"},{"location":"reference/com/views/#com.views.PosterForm","title":"PosterForm(*args, **kwargs)","text":"

Bases: ModelForm

Source code in com/forms.py
def __init__(self, *args, **kwargs):\n    self.user = kwargs.pop(\"user\", None)\n    super().__init__(*args, **kwargs)\n    if self.user and not self.user.is_com_admin:\n        self.fields[\"club\"].queryset = Club.objects.filter(\n            id__in=self.user.clubs_with_rights\n        )\n        self.fields.pop(\"display_time\")\n
"},{"location":"reference/com/views/#com.views.News","title":"News","text":"

Bases: Model

News about club events.

"},{"location":"reference/com/views/#com.views.NewsDate","title":"NewsDate","text":"

Bases: Model

A date associated with news.

A News can have multiple dates, for example if it is a recurring event.

"},{"location":"reference/com/views/#com.views.Poster","title":"Poster","text":"

Bases: Model

"},{"location":"reference/com/views/#com.views.Screen","title":"Screen","text":"

Bases: Model

"},{"location":"reference/com/views/#com.views.Sith","title":"Sith","text":"

Bases: Model

A one instance class storing all the modifiable infos.

"},{"location":"reference/com/views/#com.views.Weekmail","title":"Weekmail","text":"

Bases: Model

The weekmail class.

:ivar title: Title of the weekmail :ivar intro: Introduction of the weekmail :ivar joke: Joke of the week :ivar protip: Tip of the week :ivar conclusion: Conclusion of the weekmail :ivar sent: Track if the weekmail has been sent

"},{"location":"reference/com/views/#com.views.Weekmail.send","title":"send()","text":"

Send the weekmail to all users with the receive weekmail option opt-in.

Also send the weekmail to the mailing list in settings.SITH_COM_EMAIL.

Source code in com/models.py
def send(self):\n    \"\"\"Send the weekmail to all users with the receive weekmail option opt-in.\n\n    Also send the weekmail to the mailing list in settings.SITH_COM_EMAIL.\n    \"\"\"\n    dest = [\n        i[0]\n        for i in Preferences.objects.filter(receive_weekmail=True).values_list(\n            \"user__email\"\n        )\n    ]\n    with transaction.atomic():\n        email = EmailMultiAlternatives(\n            subject=self.title,\n            body=self.render_text(),\n            from_email=settings.SITH_COM_EMAIL,\n            to=Sith.objects.first().weekmail_destinations.split(\" \"),\n            bcc=dest,\n        )\n        email.attach_alternative(self.render_html(), \"text/html\")\n        email.send()\n        self.sent = True\n        self.save()\n        Weekmail().save()\n
"},{"location":"reference/com/views/#com.views.Weekmail.render_text","title":"render_text()","text":"

Renders a pure text version of the mail for readers without HTML support.

Source code in com/models.py
def render_text(self):\n    \"\"\"Renders a pure text version of the mail for readers without HTML support.\"\"\"\n    return render(\n        None, \"com/weekmail_renderer_text.jinja\", context={\"weekmail\": self}\n    ).content.decode(\"utf-8\")\n
"},{"location":"reference/com/views/#com.views.Weekmail.render_html","title":"render_html()","text":"

Renders an HTML version of the mail with images and fancy CSS.

Source code in com/models.py
def render_html(self):\n    \"\"\"Renders an HTML version of the mail with images and fancy CSS.\"\"\"\n    return render(\n        None, \"com/weekmail_renderer_html.jinja\", context={\"weekmail\": self}\n    ).content.decode(\"utf-8\")\n
"},{"location":"reference/com/views/#com.views.Weekmail.get_banner","title":"get_banner()","text":"

Return an absolute link to the banner.

Source code in com/models.py
def get_banner(self):\n    \"\"\"Return an absolute link to the banner.\"\"\"\n    return (\n        \"http://\" + settings.SITH_URL + static(\"com/img/weekmail_bannerV2P22.png\")\n    )\n
"},{"location":"reference/com/views/#com.views.Weekmail.get_footer","title":"get_footer()","text":"

Return an absolute link to the footer.

Source code in com/models.py
def get_footer(self):\n    \"\"\"Return an absolute link to the footer.\"\"\"\n    return \"http://\" + settings.SITH_URL + static(\"com/img/weekmail_footerP22.png\")\n
"},{"location":"reference/com/views/#com.views.WeekmailArticle","title":"WeekmailArticle","text":"

Bases: Model

"},{"location":"reference/com/views/#com.views.ComTabsMixin","title":"ComTabsMixin","text":"

Bases: TabedViewMixin

"},{"location":"reference/com/views/#com.views.IsComAdminMixin","title":"IsComAdminMixin","text":"

Bases: AccessMixin

"},{"location":"reference/com/views/#com.views.ComEditView","title":"ComEditView","text":"

Bases: ComTabsMixin, CanEditPropMixin, UpdateView

"},{"location":"reference/com/views/#com.views.AlertMsgEditView","title":"AlertMsgEditView","text":"

Bases: ComEditView

"},{"location":"reference/com/views/#com.views.InfoMsgEditView","title":"InfoMsgEditView","text":"

Bases: ComEditView

"},{"location":"reference/com/views/#com.views.WeekmailDestinationEditView","title":"WeekmailDestinationEditView","text":"

Bases: ComEditView

"},{"location":"reference/com/views/#com.views.NewsCreateView","title":"NewsCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

View to either create or update News.

"},{"location":"reference/com/views/#com.views.NewsCreateView.get_date_form_kwargs","title":"get_date_form_kwargs()","text":"

Get initial data for NewsDateForm

Source code in com/views.py
def get_date_form_kwargs(self) -> dict[str, Any]:\n    \"\"\"Get initial data for NewsDateForm\"\"\"\n    if self.request.method == \"POST\":\n        return {\"data\": self.request.POST}\n    return {}\n
"},{"location":"reference/com/views/#com.views.NewsUpdateView","title":"NewsUpdateView","text":"

Bases: PermissionOrAuthorRequiredMixin, UpdateView

"},{"location":"reference/com/views/#com.views.NewsUpdateView.get_date_form_kwargs","title":"get_date_form_kwargs()","text":"

Get initial data for NewsDateForm

Source code in com/views.py
def get_date_form_kwargs(self) -> dict[str, Any]:\n    \"\"\"Get initial data for NewsDateForm\"\"\"\n    response = {}\n    if self.request.method == \"POST\":\n        response[\"data\"] = self.request.POST\n    dates = list(self.object.dates.order_by(\"id\"))\n    if len(dates) == 0:\n        return {}\n    response[\"instance\"] = dates[0]\n    occurrences = NewsDateForm.get_occurrences(len(dates))\n    if occurrences is not None:\n        response[\"initial\"] = {\"is_weekly\": True, \"occurrences\": occurrences}\n    return response\n
"},{"location":"reference/com/views/#com.views.NewsDeleteView","title":"NewsDeleteView","text":"

Bases: PermissionOrAuthorRequiredMixin, DeleteView

"},{"location":"reference/com/views/#com.views.NewsModerateView","title":"NewsModerateView","text":"

Bases: PermissionRequiredMixin, DetailView

"},{"location":"reference/com/views/#com.views.NewsAdminListView","title":"NewsAdminListView","text":"

Bases: PermissionRequiredMixin, ListView

"},{"location":"reference/com/views/#com.views.NewsListView","title":"NewsListView","text":"

Bases: ListView

"},{"location":"reference/com/views/#com.views.NewsDetailView","title":"NewsDetailView","text":"

Bases: CanViewMixin, DetailView

"},{"location":"reference/com/views/#com.views.WeekmailPreviewView","title":"WeekmailPreviewView","text":"

Bases: ComTabsMixin, QuickNotifMixin, CanEditPropMixin, DetailView

"},{"location":"reference/com/views/#com.views.WeekmailPreviewView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add rendered weekmail.

Source code in com/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add rendered weekmail.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"weekmail_rendered\"] = self.object.render_html()\n    kwargs[\"bad_recipients\"] = self.bad_recipients\n    return kwargs\n
"},{"location":"reference/com/views/#com.views.WeekmailEditView","title":"WeekmailEditView","text":"

Bases: ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateView

"},{"location":"reference/com/views/#com.views.WeekmailEditView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add orphan articles.

Source code in com/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add orphan articles.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"orphans\"] = WeekmailArticle.objects.filter(weekmail=None)\n    return kwargs\n
"},{"location":"reference/com/views/#com.views.WeekmailArticleEditView","title":"WeekmailArticleEditView","text":"

Bases: ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateView

Edit an article.

"},{"location":"reference/com/views/#com.views.WeekmailArticleCreateView","title":"WeekmailArticleCreateView","text":"

Bases: QuickNotifMixin, CreateView

Post an article.

"},{"location":"reference/com/views/#com.views.WeekmailArticleDeleteView","title":"WeekmailArticleDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

Delete an article.

"},{"location":"reference/com/views/#com.views.MailingListAdminView","title":"MailingListAdminView","text":"

Bases: ComTabsMixin, ListView

"},{"location":"reference/com/views/#com.views.MailingModerateView","title":"MailingModerateView","text":"

Bases: View

"},{"location":"reference/com/views/#com.views.PosterListBaseView","title":"PosterListBaseView","text":"

Bases: ListView

List communication posters.

"},{"location":"reference/com/views/#com.views.PosterCreateBaseView","title":"PosterCreateBaseView","text":"

Bases: CreateView

Create communication poster.

"},{"location":"reference/com/views/#com.views.PosterEditBaseView","title":"PosterEditBaseView","text":"

Bases: UpdateView

Edit communication poster.

"},{"location":"reference/com/views/#com.views.PosterDeleteBaseView","title":"PosterDeleteBaseView","text":"

Bases: DeleteView

Edit communication poster.

"},{"location":"reference/com/views/#com.views.PosterListView","title":"PosterListView","text":"

Bases: IsComAdminMixin, ComTabsMixin, PosterListBaseView

List communication posters.

"},{"location":"reference/com/views/#com.views.PosterCreateView","title":"PosterCreateView","text":"

Bases: IsComAdminMixin, ComTabsMixin, PosterCreateBaseView

Create communication poster.

"},{"location":"reference/com/views/#com.views.PosterEditView","title":"PosterEditView","text":"

Bases: IsComAdminMixin, ComTabsMixin, PosterEditBaseView

Edit communication poster.

"},{"location":"reference/com/views/#com.views.PosterDeleteView","title":"PosterDeleteView","text":"

Bases: IsComAdminMixin, ComTabsMixin, PosterDeleteBaseView

Delete communication poster.

"},{"location":"reference/com/views/#com.views.PosterModerateListView","title":"PosterModerateListView","text":"

Bases: IsComAdminMixin, ComTabsMixin, ListView

Moderate list communication poster.

"},{"location":"reference/com/views/#com.views.PosterModerateView","title":"PosterModerateView","text":"

Bases: IsComAdminMixin, ComTabsMixin, View

Moderate communication poster.

"},{"location":"reference/com/views/#com.views.ScreenListView","title":"ScreenListView","text":"

Bases: IsComAdminMixin, ComTabsMixin, ListView

List communication screens.

"},{"location":"reference/com/views/#com.views.ScreenSlideshowView","title":"ScreenSlideshowView","text":"

Bases: DetailView

Slideshow of actives posters.

"},{"location":"reference/com/views/#com.views.ScreenCreateView","title":"ScreenCreateView","text":"

Bases: IsComAdminMixin, ComTabsMixin, CreateView

Create communication screen.

"},{"location":"reference/com/views/#com.views.ScreenEditView","title":"ScreenEditView","text":"

Bases: IsComAdminMixin, ComTabsMixin, UpdateView

Edit communication screen.

"},{"location":"reference/com/views/#com.views.ScreenDeleteView","title":"ScreenDeleteView","text":"

Bases: IsComAdminMixin, ComTabsMixin, DeleteView

Delete communication screen.

"},{"location":"reference/core/auth/","title":"Auth","text":""},{"location":"reference/core/auth/#backend","title":"Backend","text":""},{"location":"reference/core/auth/#core.auth.backends.SithModelBackend","title":"SithModelBackend","text":"

Bases: ModelBackend

Custom auth backend for the Sith.

In fact, it's the exact same backend as django.contrib.auth.backend.ModelBackend, with the exception that group permissions are fetched slightly differently. Indeed, django tries by default to fetch the permissions associated with all the django.contrib.auth.models.Group of a user ; however, our User model overrides that, so the actual linked group model is core.models.Group. Instead of having the relation auth_perm --> auth_group <-- core_user, we have auth_perm --> auth_group <-- core_group <-- core_user.

Thus, this backend make the small tweaks necessary to make our custom models interact with the django auth.

"},{"location":"reference/core/auth/#mixins","title":"Mixins","text":""},{"location":"reference/core/auth/#core.auth.mixins.CanCreateMixin","title":"CanCreateMixin(*args, **kwargs)","text":"

Bases: View

Protect any child view that would create an object.

Raises:

Type Description PermissionDenied

If the user has not the necessary permission to create the object of the view.

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/core/auth/#core.auth.mixins.CanEditMixin","title":"CanEditMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to edit this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/core/auth/#core.auth.mixins.CanViewMixin","title":"CanViewMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to view this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/core/auth/#core.auth.mixins.FormerSubscriberMixin","title":"FormerSubscriberMixin","text":"

Bases: AccessMixin

Check if the user was at least an old subscriber.

Raises:

Type Description PermissionDenied

if the user never subscribed.

"},{"location":"reference/core/auth/#core.auth.mixins.PermissionOrAuthorRequiredMixin","title":"PermissionOrAuthorRequiredMixin","text":"

Bases: PermissionRequiredMixin

Require that the user has the required perm or is the object author.

This mixin can be used in combination with DetailView, or another base class that implements the get_object method.

Example

In the following code, a user will be able to edit news if he has the com.change_news permission or if he tries to edit his own news :

class NewsEditView(PermissionOrAuthorRequiredMixin, DetailView):\n    model = News\n    author_field = \"author\"\n    permission_required = \"com.change_news\"\n

This is more or less equivalent to :

class NewsEditView(PermissionOrAuthorRequiredMixin, DetailView):\n    model = News\n\n    def dispatch(self, request, *args, **kwargs):\n        self.object = self.get_object()\n        if not (\n            user.has_perm(\"com.change_news\")\n            or self.object.author == request.user\n        ):\n            raise PermissionDenied\n        return super().dispatch(request, *args, **kwargs)\n
"},{"location":"reference/core/auth/#core.auth.mixins.can_edit_prop","title":"can_edit_prop(obj, user)","text":"

Can the user edit the properties of the object.

Parameters:

Name Type Description Default obj Any

Object to test for permission

required user User

core.models.User to test permissions against

required

Returns:

Type Description bool

True if user is authorized to edit object properties else False

Example
if not can_edit_prop(self.object ,request.user):\n    raise PermissionDenied\n
Source code in core/auth/mixins.py
def can_edit_prop(obj: Any, user: User) -> bool:\n    \"\"\"Can the user edit the properties of the object.\n\n    Args:\n        obj: Object to test for permission\n        user: core.models.User to test permissions against\n\n    Returns:\n        True if user is authorized to edit object properties else False\n\n    Example:\n        ```python\n        if not can_edit_prop(self.object ,request.user):\n            raise PermissionDenied\n        ```\n    \"\"\"\n    return obj is None or user.is_owner(obj)\n
"},{"location":"reference/core/auth/#core.auth.mixins.can_edit","title":"can_edit(obj, user)","text":"

Can the user edit the object.

Parameters:

Name Type Description Default obj Any

Object to test for permission

required user User

core.models.User to test permissions against

required

Returns:

Type Description bool

True if user is authorized to edit object else False

Example
if not can_edit(self.object, request.user):\n    raise PermissionDenied\n
Source code in core/auth/mixins.py
def can_edit(obj: Any, user: User) -> bool:\n    \"\"\"Can the user edit the object.\n\n    Args:\n        obj: Object to test for permission\n        user: core.models.User to test permissions against\n\n    Returns:\n        True if user is authorized to edit object else False\n\n    Example:\n        ```python\n        if not can_edit(self.object, request.user):\n            raise PermissionDenied\n        ```\n    \"\"\"\n    if obj is None or user.can_edit(obj):\n        return True\n    return can_edit_prop(obj, user)\n
"},{"location":"reference/core/auth/#core.auth.mixins.can_view","title":"can_view(obj, user)","text":"

Can the user see the object.

Parameters:

Name Type Description Default obj Any

Object to test for permission

required user User

core.models.User to test permissions against

required

Returns:

Type Description bool

True if user is authorized to see object else False

Example
if not can_view(self.object ,request.user):\n    raise PermissionDenied\n
Source code in core/auth/mixins.py
def can_view(obj: Any, user: User) -> bool:\n    \"\"\"Can the user see the object.\n\n    Args:\n        obj: Object to test for permission\n        user: core.models.User to test permissions against\n\n    Returns:\n        True if user is authorized to see object else False\n\n    Example:\n        ```python\n        if not can_view(self.object ,request.user):\n            raise PermissionDenied\n        ```\n    \"\"\"\n    if obj is None or user.can_view(obj):\n        return True\n    return can_edit(obj, user)\n
"},{"location":"reference/core/auth/#api-permissions","title":"API Permissions","text":"

Permission classes to be used within ninja-extra controllers.

Some permissions are global (like IsInGroup or IsRoot), and some others are per-object (like CanView or CanEdit).

Example
# restrict all the routes of this controller\n# to subscribed users\n@api_controller(\"/foo\", permissions=[IsSubscriber])\nclass FooController(ControllerBase):\n    @route.get(\"/bar\")\n    def bar_get(self):\n        # This route inherits the permissions of the controller\n        # ...\n\n    @route.bar(\"/bar/{bar_id}\", permissions=[CanView])\n    def bar_get_one(self, bar_id: int):\n        # per-object permission resolution happens\n        # when calling either the `get_object_or_exception`\n        # or `get_object_or_none` method.\n        bar = self.get_object_or_exception(Counter, pk=bar_id)\n\n        # you can also call the `check_object_permission` manually\n        other_bar = Counter.objects.first()\n        self.check_object_permissions(other_bar)\n\n        # ...\n\n    # This route is restricted to counter admins and root users\n    @route.delete(\n        \"/bar/{bar_id}\",\n        permissions=[IsRoot | IsInGroup(settings.SITH_GROUP_COUNTER_ADMIN_ID)\n    ]\n    def bar_delete(self, bar_id: int):\n        # ...\n
"},{"location":"reference/core/auth/#core.auth.api_permissions.CanAccessLookup","title":"CanAccessLookup = IsOldSubscriber | IsRoot | IsLoggedInCounter module-attribute","text":""},{"location":"reference/core/auth/#core.auth.api_permissions.IsInGroup","title":"IsInGroup(group_pk)","text":"

Bases: BasePermission

Check that the user is in the group whose primary key is given.

Source code in core/auth/api_permissions.py
def __init__(self, group_pk: int):\n    self._group_pk = group_pk\n
"},{"location":"reference/core/auth/#core.auth.api_permissions.IsRoot","title":"IsRoot","text":"

Bases: BasePermission

Check that the user is root.

"},{"location":"reference/core/auth/#core.auth.api_permissions.IsSubscriber","title":"IsSubscriber","text":"

Bases: BasePermission

Check that the user is currently subscribed.

"},{"location":"reference/core/auth/#core.auth.api_permissions.IsOldSubscriber","title":"IsOldSubscriber","text":"

Bases: BasePermission

Check that the user has at least one subscription in its history.

"},{"location":"reference/core/auth/#core.auth.api_permissions.CanView","title":"CanView","text":"

Bases: BasePermission

Check that this user has the permission to view the object of this route.

Wrap the user.can_view(obj) method. To see an example, look at the example in the module docstring.

"},{"location":"reference/core/auth/#core.auth.api_permissions.CanEdit","title":"CanEdit","text":"

Bases: BasePermission

Check that this user has the permission to edit the object of this route.

Wrap the user.can_edit(obj) method. To see an example, look at the example in the module docstring.

"},{"location":"reference/core/auth/#core.auth.api_permissions.IsOwner","title":"IsOwner","text":"

Bases: BasePermission

Check that this user owns the object of this route.

Wrap the user.is_owner(obj) method. To see an example, look at the example in the module docstring.

"},{"location":"reference/core/auth/#core.auth.api_permissions.IsLoggedInCounter","title":"IsLoggedInCounter","text":"

Bases: BasePermission

Check that a user is logged in a counter.

"},{"location":"reference/core/model_fields/","title":"Champs de mod\u00e8le","text":""},{"location":"reference/core/model_fields/#core.fields.ResizedImageFieldFile","title":"ResizedImageFieldFile","text":"

Bases: ImageFieldFile

"},{"location":"reference/core/model_fields/#core.fields.ResizedImageFieldFile.get_resized_dimensions","title":"get_resized_dimensions(image)","text":"

Get the dimensions of the resized image.

If the width and height are given, they are used. If only one is given, the other is calculated to keep the same ratio.

Returns:

Type Description tuple[int, int]

Tuple of width and height

Source code in core/fields.py
def get_resized_dimensions(self, image: Image.Image) -> tuple[int, int]:\n    \"\"\"Get the dimensions of the resized image.\n\n    If the width and height are given, they are used.\n    If only one is given, the other is calculated to keep the same ratio.\n\n    Returns:\n        Tuple of width and height\n    \"\"\"\n    width = self.field.width\n    height = self.field.height\n    if width is not None and height is not None:\n        return self.field.width, self.field.height\n    if width is None:\n        width = int(image.width * height / image.height)\n    elif height is None:\n        height = int(image.height * width / image.width)\n    return width, height\n
"},{"location":"reference/core/model_fields/#core.fields.ResizedImageFieldFile.get_name","title":"get_name()","text":"

Get the name of the resized image.

If the field has a force_format attribute, the extension of the file will be changed to match it. Otherwise, the name is left unchanged.

Raises:

Type Description ValueError

If the image format is unknown

Source code in core/fields.py
def get_name(self) -> str:\n    \"\"\"Get the name of the resized image.\n\n    If the field has a force_format attribute,\n    the extension of the file will be changed to match it.\n    Otherwise, the name is left unchanged.\n\n    Raises:\n        ValueError: If the image format is unknown\n    \"\"\"\n    if not self.field.force_format:\n        return self.name\n    formats = {val: key for key, val in Image.registered_extensions().items()}\n    new_format = self.field.force_format\n    if new_format in formats:\n        extension = formats[new_format]\n    else:\n        raise ValueError(f\"Unknown format {new_format}\")\n    return str(Path(self.file.name).with_suffix(extension))\n
"},{"location":"reference/core/model_fields/#core.fields.ResizedImageField","title":"ResizedImageField(width=None, height=None, force_format=None, **kwargs)","text":"

Bases: ImageField

A field that automatically resizes images to a given size.

This field is useful for profile pictures or product icons, for example.

The final size of the image is determined by the width and height parameters :

If the force_format parameter is given, the image will be converted to this format.

Examples:

To resize an image with a height of 100px, without changing the ratio, and a format of WEBP :

class Product(models.Model):\n    icon = ResizedImageField(height=100, force_format=\"WEBP\")\n

To explicitly resize an image to 100x100px (but possibly change the ratio) :

class Product(models.Model):\n    icon = ResizedImageField(width=100, height=100)\n

Raises:

Type Description FieldError

If neither width nor height is given

Parameters:

Name Type Description Default width int | None

If given, the width of the resized image

None height int | None

If given, the height of the resized image

None force_format str | None

If given, the image will be converted to this format

None Source code in core/fields.py
def __init__(\n    self,\n    width: int | None = None,\n    height: int | None = None,\n    force_format: str | None = None,\n    **kwargs,\n):\n    if width is None and height is None:\n        raise FieldError(\n            f\"{self.__class__.__name__} requires \"\n            \"width, height or both, but got neither\"\n        )\n    self.width = width\n    self.height = height\n    self.force_format = force_format\n    super().__init__(**kwargs)\n
"},{"location":"reference/core/models/","title":"Models","text":""},{"location":"reference/core/models/#core.models.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/core/models/#core.models.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/core/models/#core.models.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/core/models/#core.models.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/core/models/#core.models.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/core/models/#core.models.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/core/models/#core.models.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/core/models/#core.models.Group","title":"Group","text":"

Bases: Group

Wrapper around django.auth.Group

"},{"location":"reference/core/models/#core.models.BanGroup","title":"BanGroup","text":"

Bases: Group

An anti-group, that removes permissions instead of giving them.

Users are linked to BanGroups through UserBan objects.

Example
user = User.objects.get(username=\"...\")\nban_group = BanGroup.objects.first()\nUserBan.objects.create(user=user, ban_group=ban_group, reason=\"...\")\n\nassert user.ban_groups.contains(ban_group)\n
"},{"location":"reference/core/models/#core.models.UserQuerySet","title":"UserQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/core/models/#core.models.CustomUserManager","title":"CustomUserManager","text":"

Bases: from_queryset(UserQuerySet)

"},{"location":"reference/core/models/#core.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/core/models/#core.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/core/models/#core.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/core/models/#core.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/core/models/#core.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/core/models/#core.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/core/models/#core.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/core/models/#core.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/core/models/#core.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/core/models/#core.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/core/models/#core.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/core/models/#core.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/core/models/#core.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/core/models/#core.models.AnonymousUser","title":"AnonymousUser()","text":"

Bases: AnonymousUser

Source code in core/models.py
def __init__(self):\n    super().__init__()\n
"},{"location":"reference/core/models/#core.models.AnonymousUser.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

The anonymous user is only in the public group.

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"The anonymous user is only in the public group.\"\"\"\n    allowed_id = settings.SITH_GROUP_PUBLIC_ID\n    if pk is not None:\n        return pk == allowed_id\n    elif name is not None:\n        group = get_group(name=name)\n        return group is not None and group.id == allowed_id\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n
"},{"location":"reference/core/models/#core.models.UserBan","title":"UserBan","text":"

Bases: Model

A ban of a user.

A user can be banned for a specific reason, for a specific duration. The expiration date is indicative, and the ban should be removed manually.

"},{"location":"reference/core/models/#core.models.Preferences","title":"Preferences","text":"

Bases: Model

"},{"location":"reference/core/models/#core.models.SithFile","title":"SithFile","text":"

Bases: Model

"},{"location":"reference/core/models/#core.models.SithFile.clean","title":"clean()","text":"

Cleans up the file.

Source code in core/models.py
def clean(self):\n    \"\"\"Cleans up the file.\"\"\"\n    super().clean()\n    if \"/\" in self.name:\n        raise ValidationError(_(\"Character '/' not authorized in name\"))\n    if self == self.parent:\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self == self.parent or (\n        self.parent is not None and self in self.get_parent_list()\n    ):\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self.parent and self.parent.is_file:\n        raise ValidationError(\n            _(\"You can not make a file be a children of a non folder file\")\n        )\n    if (\n        self.parent is None\n        and SithFile.objects.exclude(id=self.id)\n        .filter(parent=None, name=self.name)\n        .exists()\n    ) or (\n        self.parent\n        and self.parent.children.exclude(id=self.id).filter(name=self.name).exists()\n    ):\n        raise ValidationError(_(\"Duplicate file\"), code=\"duplicate\")\n    if self.is_folder:\n        if self.file:\n            try:\n                Image.open(BytesIO(self.file.read()))\n            except Image.UnidentifiedImageError as e:\n                raise ValidationError(\n                    _(\"This is not a valid folder thumbnail\")\n                ) from e\n        self.mime_type = \"inode/directory\"\n    if self.is_file and (self.file is None or self.file == \"\"):\n        raise ValidationError(_(\"You must provide a file\"))\n
"},{"location":"reference/core/models/#core.models.SithFile.apply_rights_recursively","title":"apply_rights_recursively(*, only_folders=False)","text":"

Apply the rights of this file to all children recursively.

Parameters:

Name Type Description Default only_folders bool

If True, only apply the rights to SithFiles that are folders.

False Source code in core/models.py
def apply_rights_recursively(self, *, only_folders: bool = False) -> None:\n    \"\"\"Apply the rights of this file to all children recursively.\n\n    Args:\n        only_folders: If True, only apply the rights to SithFiles that are folders.\n    \"\"\"\n    file_ids = []\n    explored_ids = [self.id]\n    while len(explored_ids) > 0:  # find all children recursively\n        file_ids.extend(explored_ids)\n        next_level = SithFile.objects.filter(parent_id__in=explored_ids)\n        if only_folders:\n            next_level = next_level.filter(is_folder=True)\n        explored_ids = list(next_level.values_list(\"id\", flat=True))\n    for through in (SithFile.view_groups.through, SithFile.edit_groups.through):\n        # force evaluation. Without this, the iterator yields nothing\n        groups = list(\n            through.objects.filter(sithfile_id=self.id).values_list(\n                \"group_id\", flat=True\n            )\n        )\n        # delete previous rights\n        through.objects.filter(sithfile_id__in=file_ids).delete()\n        through.objects.bulk_create(  # create new rights\n            [through(sithfile_id=f, group_id=g) for f in file_ids for g in groups]\n        )\n
"},{"location":"reference/core/models/#core.models.SithFile.copy_rights","title":"copy_rights()","text":"

Copy, if possible, the rights of the parent folder.

Source code in core/models.py
def copy_rights(self):\n    \"\"\"Copy, if possible, the rights of the parent folder.\"\"\"\n    if self.parent is not None:\n        self.edit_groups.set(self.parent.edit_groups.all())\n        self.view_groups.set(self.parent.view_groups.all())\n
"},{"location":"reference/core/models/#core.models.SithFile.move_to","title":"move_to(parent)","text":"

Move a file to a new parent. parent must be a SithFile with the is_folder=True property. Otherwise, this function doesn't change anything. This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify SithFiles recursively, so it stays efficient even with top-level folders.

Source code in core/models.py
def move_to(self, parent):\n    \"\"\"Move a file to a new parent.\n    `parent` must be a SithFile with the `is_folder=True` property. Otherwise, this function doesn't change\n    anything.\n    This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify\n    SithFiles recursively, so it stays efficient even with top-level folders.\n    \"\"\"\n    if not parent.is_folder:\n        return\n    self.parent = parent\n    self.clean()\n    self.save()\n
"},{"location":"reference/core/models/#core.models.LockError","title":"LockError","text":"

Bases: Exception

There was a lock error on the object.

"},{"location":"reference/core/models/#core.models.AlreadyLocked","title":"AlreadyLocked","text":"

Bases: LockError

The object is already locked.

"},{"location":"reference/core/models/#core.models.NotLocked","title":"NotLocked","text":"

Bases: LockError

The object is not locked.

"},{"location":"reference/core/models/#core.models.Page","title":"Page","text":"

Bases: Model

The page class to build a Wiki Each page may have a parent and it's URL is of the form my.site/page/// It has an ID field, but don't use it, since it's only there for DB part, and because compound primary key is awkward! Prefere querying pages with Page.get_page_by_full_name().

Be careful with the _full_name attribute: this field may not be valid until you call save(). It's made for fast query, but don't rely on it when playing with a Page object, use get_full_name() instead!

"},{"location":"reference/core/models/#core.models.Page.save","title":"save(*args, **kwargs)","text":"

Performs some needed actions before and after saving a page in database.

Source code in core/models.py
def save(self, *args, **kwargs):\n    \"\"\"Performs some needed actions before and after saving a page in database.\"\"\"\n    locked = kwargs.pop(\"force_lock\", False)\n    if not locked:\n        locked = self.is_locked()\n    if not locked:\n        raise NotLocked(\"The page is not locked and thus can not be saved\")\n    self.full_clean()\n    if not self.id:\n        super().save(\n            *args, **kwargs\n        )  # Save a first time to correctly set _full_name\n    # This reset the _full_name just before saving to maintain a coherent field quicker for queries than the\n    # recursive method\n    # It also update all the children to maintain correct names\n    self._full_name = self.get_full_name()\n    for c in self.children.all():\n        c.save()\n    super().save(*args, **kwargs)\n    self.unset_lock()\n
"},{"location":"reference/core/models/#core.models.Page.get_page_by_full_name","title":"get_page_by_full_name(name) staticmethod","text":"

Quicker to get a page with that method rather than building the request every time.

Source code in core/models.py
@staticmethod\ndef get_page_by_full_name(name):\n    \"\"\"Quicker to get a page with that method rather than building the request every time.\"\"\"\n    return Page.objects.filter(_full_name=name).first()\n
"},{"location":"reference/core/models/#core.models.Page.clean","title":"clean()","text":"

Cleans up only the name for the moment, but this can be used to make any treatment before saving the object.

Source code in core/models.py
def clean(self):\n    \"\"\"Cleans up only the name for the moment, but this can be used to make any treatment before saving the object.\"\"\"\n    if \"/\" in self.name:\n        self.name = self.name.split(\"/\")[-1]\n    if (\n        Page.objects.exclude(pk=self.pk)\n        .filter(_full_name=self.get_full_name())\n        .exists()\n    ):\n        raise ValidationError(_(\"Duplicate page\"), code=\"duplicate\")\n    super().clean()\n    if self.parent is not None and self in self.get_parent_list():\n        raise ValidationError(_(\"Loop in page tree\"), code=\"loop\")\n
"},{"location":"reference/core/models/#core.models.Page.is_locked","title":"is_locked()","text":"

Is True if the page is locked, False otherwise.

This is where the timeout is handled, so a locked page for which the timeout is reach will be unlocked and this function will return False.

Source code in core/models.py
def is_locked(self):\n    \"\"\"Is True if the page is locked, False otherwise.\n\n    This is where the timeout is handled,\n    so a locked page for which the timeout is reach will be unlocked and this\n    function will return False.\n    \"\"\"\n    if self.lock_timeout and (\n        timezone.now() - self.lock_timeout > timedelta(minutes=5)\n    ):\n        self.unset_lock()\n    return (\n        self.lock_user\n        and self.lock_timeout\n        and (timezone.now() - self.lock_timeout < timedelta(minutes=5))\n    )\n
"},{"location":"reference/core/models/#core.models.Page.set_lock","title":"set_lock(user)","text":"

Sets a lock on the current page or raise an AlreadyLocked exception.

Source code in core/models.py
def set_lock(self, user):\n    \"\"\"Sets a lock on the current page or raise an AlreadyLocked exception.\"\"\"\n    if self.is_locked() and self.get_lock() != user:\n        raise AlreadyLocked(\"The page is already locked by someone else\")\n    self.lock_user = user\n    self.lock_timeout = timezone.now()\n    super().save()\n
"},{"location":"reference/core/models/#core.models.Page.set_lock_recursive","title":"set_lock_recursive(user)","text":"

Locks recursively all the child pages for editing properties.

Source code in core/models.py
def set_lock_recursive(self, user):\n    \"\"\"Locks recursively all the child pages for editing properties.\"\"\"\n    for p in self.children.all():\n        p.set_lock_recursive(user)\n    self.set_lock(user)\n
"},{"location":"reference/core/models/#core.models.Page.unset_lock_recursive","title":"unset_lock_recursive()","text":"

Unlocks recursively all the child pages.

Source code in core/models.py
def unset_lock_recursive(self):\n    \"\"\"Unlocks recursively all the child pages.\"\"\"\n    for p in self.children.all():\n        p.unset_lock_recursive()\n    self.unset_lock()\n
"},{"location":"reference/core/models/#core.models.Page.unset_lock","title":"unset_lock()","text":"

Always try to unlock, even if there is no lock.

Source code in core/models.py
def unset_lock(self):\n    \"\"\"Always try to unlock, even if there is no lock.\"\"\"\n    self.lock_user = None\n    self.lock_timeout = None\n    super().save()\n
"},{"location":"reference/core/models/#core.models.Page.get_lock","title":"get_lock()","text":"

Returns the page's mutex containing the time and the user in a dict.

Source code in core/models.py
def get_lock(self):\n    \"\"\"Returns the page's mutex containing the time and the user in a dict.\"\"\"\n    if self.lock_user:\n        return self.lock_user\n    raise NotLocked(\"The page is not locked and thus can not return its user\")\n
"},{"location":"reference/core/models/#core.models.Page.get_full_name","title":"get_full_name()","text":"

Computes the real full_name of the page based on its name and its parent's name You can and must rely on this function when working on a page object that is not freshly fetched from the DB (For example when treating a Page object coming from a form).

Source code in core/models.py
def get_full_name(self):\n    \"\"\"Computes the real full_name of the page based on its name and its parent's name\n    You can and must rely on this function when working on a page object that is not freshly fetched from the DB\n    (For example when treating a Page object coming from a form).\n    \"\"\"\n    if self.parent is None:\n        return self.name\n    return f\"{self.parent.get_full_name()}/{self.name}\"\n
"},{"location":"reference/core/models/#core.models.PageRev","title":"PageRev","text":"

Bases: Model

True content of the page.

Each page object has a revisions field that is a list of PageRev, ordered by date. my_page.revisions.last() gives the PageRev object that is the most up-to-date, and thus, is the real content of the page. The content is in PageRev.title and PageRev.content .

"},{"location":"reference/core/models/#core.models.Notification","title":"Notification","text":"

Bases: Model

"},{"location":"reference/core/models/#core.models.Gift","title":"Gift","text":"

Bases: Model

"},{"location":"reference/core/models/#core.models.OperationLog","title":"OperationLog","text":"

Bases: Model

General purpose log object to register operations.

"},{"location":"reference/core/models/#core.models.validate_promo","title":"validate_promo(value)","text":"Source code in core/models.py
def validate_promo(value: int) -> None:\n    start_year = settings.SITH_SCHOOL_START_YEAR\n    delta = (localdate() + timedelta(days=180)).year - start_year\n    if value < 0 or delta < value:\n        raise ValidationError(\n            _(\"%(value)s is not a valid promo (between 0 and %(end)s)\"),\n            params={\"value\": value, \"end\": delta},\n        )\n
"},{"location":"reference/core/models/#core.models.get_group","title":"get_group(*, pk=None, name=None)","text":"

Search for a group by its primary key or its name. Either one of the two must be set.

The result is cached for the default duration (should be 5 minutes).

Parameters:

Name Type Description Default pk int | None

The primary key of the group

None name str | None

The name of the group

None

Returns:

Type Description Group | None

The group if it exists, else None

Raises:

Type Description ValueError

If no group matches the criteria

Source code in core/models.py
def get_group(*, pk: int | None = None, name: str | None = None) -> Group | None:\n    \"\"\"Search for a group by its primary key or its name.\n    Either one of the two must be set.\n\n    The result is cached for the default duration (should be 5 minutes).\n\n    Args:\n        pk: The primary key of the group\n        name: The name of the group\n\n    Returns:\n        The group if it exists, else None\n\n    Raises:\n        ValueError: If no group matches the criteria\n    \"\"\"\n    if pk is None and name is None:\n        raise ValueError(\"Either pk or name must be set\")\n\n    # replace space characters to hide warnings with memcached backend\n    pk_or_name: str | int = pk if pk is not None else name.replace(\" \", \"_\")\n    group = cache.get(f\"sith_group_{pk_or_name}\")\n\n    if group == \"not_found\":\n        # Using None as a cache value is a little bit tricky,\n        # so we use a special string to represent None\n        return None\n    elif group is not None:\n        return group\n    # if this point is reached, the group is not in cache\n    if pk is not None:\n        group = Group.objects.filter(pk=pk).first()\n    else:\n        group = Group.objects.filter(name=name).first()\n    if group is not None:\n        name = group.name.replace(\" \", \"_\")\n        cache.set_many({f\"sith_group_{group.id}\": group, f\"sith_group_{name}\": group})\n    else:\n        cache.set(f\"sith_group_{pk_or_name}\", \"not_found\")\n    return group\n
"},{"location":"reference/core/models/#core.models.get_directory","title":"get_directory(instance, filename)","text":"Source code in core/models.py
def get_directory(instance, filename):\n    return \".{0}/{1}\".format(instance.get_parent_path(), filename)\n
"},{"location":"reference/core/models/#core.models.get_compressed_directory","title":"get_compressed_directory(instance, filename)","text":"Source code in core/models.py
def get_compressed_directory(instance, filename):\n    return \"./.compressed/{0}/{1}\".format(instance.get_parent_path(), filename)\n
"},{"location":"reference/core/models/#core.models.get_thumbnail_directory","title":"get_thumbnail_directory(instance, filename)","text":"Source code in core/models.py
def get_thumbnail_directory(instance, filename):\n    return \"./.thumbnails/{0}/{1}\".format(instance.get_parent_path(), filename)\n
"},{"location":"reference/core/models/#core.models.get_default_owner_group","title":"get_default_owner_group()","text":"Source code in core/models.py
def get_default_owner_group():\n    return settings.SITH_GROUP_ROOT_ID\n
"},{"location":"reference/core/schemas/","title":"Schemas","text":""},{"location":"reference/core/schemas/#core.schemas.Group","title":"Group","text":"

Bases: Group

Wrapper around django.auth.Group

"},{"location":"reference/core/schemas/#core.schemas.SithFile","title":"SithFile","text":"

Bases: Model

"},{"location":"reference/core/schemas/#core.schemas.SithFile.clean","title":"clean()","text":"

Cleans up the file.

Source code in core/models.py
def clean(self):\n    \"\"\"Cleans up the file.\"\"\"\n    super().clean()\n    if \"/\" in self.name:\n        raise ValidationError(_(\"Character '/' not authorized in name\"))\n    if self == self.parent:\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self == self.parent or (\n        self.parent is not None and self in self.get_parent_list()\n    ):\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self.parent and self.parent.is_file:\n        raise ValidationError(\n            _(\"You can not make a file be a children of a non folder file\")\n        )\n    if (\n        self.parent is None\n        and SithFile.objects.exclude(id=self.id)\n        .filter(parent=None, name=self.name)\n        .exists()\n    ) or (\n        self.parent\n        and self.parent.children.exclude(id=self.id).filter(name=self.name).exists()\n    ):\n        raise ValidationError(_(\"Duplicate file\"), code=\"duplicate\")\n    if self.is_folder:\n        if self.file:\n            try:\n                Image.open(BytesIO(self.file.read()))\n            except Image.UnidentifiedImageError as e:\n                raise ValidationError(\n                    _(\"This is not a valid folder thumbnail\")\n                ) from e\n        self.mime_type = \"inode/directory\"\n    if self.is_file and (self.file is None or self.file == \"\"):\n        raise ValidationError(_(\"You must provide a file\"))\n
"},{"location":"reference/core/schemas/#core.schemas.SithFile.apply_rights_recursively","title":"apply_rights_recursively(*, only_folders=False)","text":"

Apply the rights of this file to all children recursively.

Parameters:

Name Type Description Default only_folders bool

If True, only apply the rights to SithFiles that are folders.

False Source code in core/models.py
def apply_rights_recursively(self, *, only_folders: bool = False) -> None:\n    \"\"\"Apply the rights of this file to all children recursively.\n\n    Args:\n        only_folders: If True, only apply the rights to SithFiles that are folders.\n    \"\"\"\n    file_ids = []\n    explored_ids = [self.id]\n    while len(explored_ids) > 0:  # find all children recursively\n        file_ids.extend(explored_ids)\n        next_level = SithFile.objects.filter(parent_id__in=explored_ids)\n        if only_folders:\n            next_level = next_level.filter(is_folder=True)\n        explored_ids = list(next_level.values_list(\"id\", flat=True))\n    for through in (SithFile.view_groups.through, SithFile.edit_groups.through):\n        # force evaluation. Without this, the iterator yields nothing\n        groups = list(\n            through.objects.filter(sithfile_id=self.id).values_list(\n                \"group_id\", flat=True\n            )\n        )\n        # delete previous rights\n        through.objects.filter(sithfile_id__in=file_ids).delete()\n        through.objects.bulk_create(  # create new rights\n            [through(sithfile_id=f, group_id=g) for f in file_ids for g in groups]\n        )\n
"},{"location":"reference/core/schemas/#core.schemas.SithFile.copy_rights","title":"copy_rights()","text":"

Copy, if possible, the rights of the parent folder.

Source code in core/models.py
def copy_rights(self):\n    \"\"\"Copy, if possible, the rights of the parent folder.\"\"\"\n    if self.parent is not None:\n        self.edit_groups.set(self.parent.edit_groups.all())\n        self.view_groups.set(self.parent.view_groups.all())\n
"},{"location":"reference/core/schemas/#core.schemas.SithFile.move_to","title":"move_to(parent)","text":"

Move a file to a new parent. parent must be a SithFile with the is_folder=True property. Otherwise, this function doesn't change anything. This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify SithFiles recursively, so it stays efficient even with top-level folders.

Source code in core/models.py
def move_to(self, parent):\n    \"\"\"Move a file to a new parent.\n    `parent` must be a SithFile with the `is_folder=True` property. Otherwise, this function doesn't change\n    anything.\n    This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify\n    SithFiles recursively, so it stays efficient even with top-level folders.\n    \"\"\"\n    if not parent.is_folder:\n        return\n    self.parent = parent\n    self.clean()\n    self.save()\n
"},{"location":"reference/core/schemas/#core.schemas.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/core/schemas/#core.schemas.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/core/schemas/#core.schemas.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/core/schemas/#core.schemas.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/core/schemas/#core.schemas.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/core/schemas/#core.schemas.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/core/schemas/#core.schemas.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/core/schemas/#core.schemas.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/core/schemas/#core.schemas.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/core/schemas/#core.schemas.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/core/schemas/#core.schemas.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/core/schemas/#core.schemas.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/core/schemas/#core.schemas.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/core/schemas/#core.schemas.SimpleUserSchema","title":"SimpleUserSchema","text":"

Bases: ModelSchema

A schema with the minimum amount of information to represent a user.

"},{"location":"reference/core/schemas/#core.schemas.UserProfileSchema","title":"UserProfileSchema","text":"

Bases: ModelSchema

The necessary information to show a user profile

"},{"location":"reference/core/schemas/#core.schemas.SithFileSchema","title":"SithFileSchema","text":"

Bases: ModelSchema

"},{"location":"reference/core/schemas/#core.schemas.GroupSchema","title":"GroupSchema","text":"

Bases: ModelSchema

"},{"location":"reference/core/schemas/#core.schemas.UserFilterSchema","title":"UserFilterSchema","text":"

Bases: FilterSchema

"},{"location":"reference/core/schemas/#core.schemas.MarkdownSchema","title":"MarkdownSchema","text":"

Bases: Schema

"},{"location":"reference/core/schemas/#core.schemas.FamilyGodfatherSchema","title":"FamilyGodfatherSchema","text":"

Bases: Schema

"},{"location":"reference/core/schemas/#core.schemas.UserFamilySchema","title":"UserFamilySchema","text":"

Bases: Schema

Represent a graph of a user's family

"},{"location":"reference/core/views/","title":"Views","text":""},{"location":"reference/core/views/#core.views.DetailFormView","title":"DetailFormView","text":"

Bases: SingleObjectMixin, FormView

Class that allow both a detail view and a form view.

"},{"location":"reference/core/views/#core.views.DetailFormView.get_object","title":"get_object()","text":"

Get current group from id in url.

Source code in core/views/__init__.py
def get_object(self):\n    \"\"\"Get current group from id in url.\"\"\"\n    return self.cached_object\n
"},{"location":"reference/core/views/#core.views.DetailFormView.cached_object","title":"cached_object()","text":"

Optimisation on group retrieval.

Source code in core/views/__init__.py
@cached_property\ndef cached_object(self):\n    \"\"\"Optimisation on group retrieval.\"\"\"\n    return super().get_object()\n
"},{"location":"reference/core/views/#core.views.SithFile","title":"SithFile","text":"

Bases: Model

"},{"location":"reference/core/views/#core.views.SithFile.clean","title":"clean()","text":"

Cleans up the file.

Source code in core/models.py
def clean(self):\n    \"\"\"Cleans up the file.\"\"\"\n    super().clean()\n    if \"/\" in self.name:\n        raise ValidationError(_(\"Character '/' not authorized in name\"))\n    if self == self.parent:\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self == self.parent or (\n        self.parent is not None and self in self.get_parent_list()\n    ):\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self.parent and self.parent.is_file:\n        raise ValidationError(\n            _(\"You can not make a file be a children of a non folder file\")\n        )\n    if (\n        self.parent is None\n        and SithFile.objects.exclude(id=self.id)\n        .filter(parent=None, name=self.name)\n        .exists()\n    ) or (\n        self.parent\n        and self.parent.children.exclude(id=self.id).filter(name=self.name).exists()\n    ):\n        raise ValidationError(_(\"Duplicate file\"), code=\"duplicate\")\n    if self.is_folder:\n        if self.file:\n            try:\n                Image.open(BytesIO(self.file.read()))\n            except Image.UnidentifiedImageError as e:\n                raise ValidationError(\n                    _(\"This is not a valid folder thumbnail\")\n                ) from e\n        self.mime_type = \"inode/directory\"\n    if self.is_file and (self.file is None or self.file == \"\"):\n        raise ValidationError(_(\"You must provide a file\"))\n
"},{"location":"reference/core/views/#core.views.SithFile.apply_rights_recursively","title":"apply_rights_recursively(*, only_folders=False)","text":"

Apply the rights of this file to all children recursively.

Parameters:

Name Type Description Default only_folders bool

If True, only apply the rights to SithFiles that are folders.

False Source code in core/models.py
def apply_rights_recursively(self, *, only_folders: bool = False) -> None:\n    \"\"\"Apply the rights of this file to all children recursively.\n\n    Args:\n        only_folders: If True, only apply the rights to SithFiles that are folders.\n    \"\"\"\n    file_ids = []\n    explored_ids = [self.id]\n    while len(explored_ids) > 0:  # find all children recursively\n        file_ids.extend(explored_ids)\n        next_level = SithFile.objects.filter(parent_id__in=explored_ids)\n        if only_folders:\n            next_level = next_level.filter(is_folder=True)\n        explored_ids = list(next_level.values_list(\"id\", flat=True))\n    for through in (SithFile.view_groups.through, SithFile.edit_groups.through):\n        # force evaluation. Without this, the iterator yields nothing\n        groups = list(\n            through.objects.filter(sithfile_id=self.id).values_list(\n                \"group_id\", flat=True\n            )\n        )\n        # delete previous rights\n        through.objects.filter(sithfile_id__in=file_ids).delete()\n        through.objects.bulk_create(  # create new rights\n            [through(sithfile_id=f, group_id=g) for f in file_ids for g in groups]\n        )\n
"},{"location":"reference/core/views/#core.views.SithFile.copy_rights","title":"copy_rights()","text":"

Copy, if possible, the rights of the parent folder.

Source code in core/models.py
def copy_rights(self):\n    \"\"\"Copy, if possible, the rights of the parent folder.\"\"\"\n    if self.parent is not None:\n        self.edit_groups.set(self.parent.edit_groups.all())\n        self.view_groups.set(self.parent.view_groups.all())\n
"},{"location":"reference/core/views/#core.views.SithFile.move_to","title":"move_to(parent)","text":"

Move a file to a new parent. parent must be a SithFile with the is_folder=True property. Otherwise, this function doesn't change anything. This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify SithFiles recursively, so it stays efficient even with top-level folders.

Source code in core/models.py
def move_to(self, parent):\n    \"\"\"Move a file to a new parent.\n    `parent` must be a SithFile with the `is_folder=True` property. Otherwise, this function doesn't change\n    anything.\n    This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify\n    SithFiles recursively, so it stays efficient even with top-level folders.\n    \"\"\"\n    if not parent.is_folder:\n        return\n    self.parent = parent\n    self.clean()\n    self.save()\n
"},{"location":"reference/core/views/#core.views.AllowFragment","title":"AllowFragment","text":"

Add is_fragment to templates. It's only True if the request is emitted by htmx

"},{"location":"reference/core/views/#core.views.MultipleFileInput","title":"MultipleFileInput","text":"

Bases: ClearableFileInput

"},{"location":"reference/core/views/#core.views.MultipleFileField","title":"MultipleFileField(*args, **kwargs)","text":"

Bases: _MultipleFieldMixin, FileField

Source code in core/views/files.py
def __init__(self, *args, **kwargs):\n    kwargs.setdefault(\"widget\", MultipleFileInput())\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/core/views/#core.views.MultipleImageField","title":"MultipleImageField(*args, **kwargs)","text":"

Bases: _MultipleFieldMixin, ImageField

Source code in core/views/files.py
def __init__(self, *args, **kwargs):\n    kwargs.setdefault(\"widget\", MultipleFileInput())\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/core/views/#core.views.AddFilesForm","title":"AddFilesForm","text":"

Bases: Form

"},{"location":"reference/core/views/#core.views.FileListView","title":"FileListView","text":"

Bases: ListView

"},{"location":"reference/core/views/#core.views.FileEditView","title":"FileEditView","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/core/views/#core.views.FileEditPropForm","title":"FileEditPropForm","text":"

Bases: ModelForm

"},{"location":"reference/core/views/#core.views.FileEditPropView","title":"FileEditPropView","text":"

Bases: CanEditPropMixin, UpdateView

"},{"location":"reference/core/views/#core.views.FileView","title":"FileView","text":"

Bases: CanViewMixin, DetailView, FormMixin

Handle the upload of new files into a folder.

"},{"location":"reference/core/views/#core.views.FileView.handle_clipboard","title":"handle_clipboard(request, obj) staticmethod","text":"

Handle the clipboard in the view.

This method can fail, since it does not catch the exceptions coming from below, allowing proper handling in the calling view. Use this method like this:

FileView.handle_clipboard(request, self.object)\n

request is usually the self.request obj in your view obj is the SithFile object you want to put in the clipboard, or where you want to paste the clipboard

Source code in core/views/files.py
@staticmethod\ndef handle_clipboard(request, obj):\n    \"\"\"Handle the clipboard in the view.\n\n    This method can fail, since it does not catch the exceptions coming from\n    below, allowing proper handling in the calling view.\n    Use this method like this:\n\n        FileView.handle_clipboard(request, self.object)\n\n    `request` is usually the self.request obj in your view\n    `obj` is the SithFile object you want to put in the clipboard, or\n             where you want to paste the clipboard\n    \"\"\"\n    if \"delete\" in request.POST:\n        for f_id in request.POST.getlist(\"file_list\"):\n            file = SithFile.objects.filter(id=f_id).first()\n            if file:\n                file.delete()\n    if \"clear\" in request.POST:\n        request.session[\"clipboard\"] = []\n    if \"cut\" in request.POST:\n        for f_id_str in request.POST.getlist(\"file_list\"):\n            f_id = int(f_id_str)\n            if (\n                f_id in [c.id for c in obj.children.all()]\n                and f_id not in request.session[\"clipboard\"]\n            ):\n                request.session[\"clipboard\"].append(f_id)\n    if \"paste\" in request.POST:\n        for f_id in request.session[\"clipboard\"]:\n            file = SithFile.objects.filter(id=f_id).first()\n            if file:\n                file.move_to(obj)\n        request.session[\"clipboard\"] = []\n    request.session.modified = True\n
"},{"location":"reference/core/views/#core.views.FileDeleteView","title":"FileDeleteView","text":"

Bases: AllowFragment, CanEditPropMixin, DeleteView

"},{"location":"reference/core/views/#core.views.FileModerationView","title":"FileModerationView","text":"

Bases: AllowFragment, ListView

"},{"location":"reference/core/views/#core.views.FileModerateView","title":"FileModerateView","text":"

Bases: CanEditPropMixin, SingleObjectMixin

"},{"location":"reference/core/views/#core.views.Group","title":"Group","text":"

Bases: Group

Wrapper around django.auth.Group

"},{"location":"reference/core/views/#core.views.EditMembersForm","title":"EditMembersForm(*args, **kwargs)","text":"

Bases: Form

Add and remove members from a Group.

Source code in core/views/group.py
def __init__(self, *args, **kwargs):\n    self.current_users = kwargs.pop(\"users\", [])\n    super().__init__(*args, **kwargs)\n\n    self.fields[\"users_added\"] = forms.ModelMultipleChoiceField(\n        label=_(\"Users to add to group\"),\n        help_text=_(\"Search users to add (one or more).\"),\n        required=False,\n        widget=AutoCompleteSelectMultipleUser,\n        queryset=User.objects.exclude(id__in=self.current_users).all(),\n    )\n\n    self.fields[\"users_removed\"] = forms.ModelMultipleChoiceField(\n        User.objects.filter(id__in=self.current_users).all(),\n        label=_(\"Users to remove from group\"),\n        required=False,\n        widget=forms.CheckboxSelectMultiple,\n    )\n
"},{"location":"reference/core/views/#core.views.GroupListView","title":"GroupListView","text":"

Bases: CanEditMixin, ListView

Displays the Group list.

"},{"location":"reference/core/views/#core.views.GroupEditView","title":"GroupEditView","text":"

Bases: CanEditMixin, UpdateView

Edit infos of a Group.

"},{"location":"reference/core/views/#core.views.GroupCreateView","title":"GroupCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

Add a new Group.

"},{"location":"reference/core/views/#core.views.GroupTemplateView","title":"GroupTemplateView","text":"

Bases: CanEditMixin, DetailFormView

Display all users in a given Group Allow adding and removing users from it.

"},{"location":"reference/core/views/#core.views.GroupDeleteView","title":"GroupDeleteView","text":"

Bases: CanEditMixin, DeleteView

Delete a Group.

"},{"location":"reference/core/views/#core.views.CanCreateMixin","title":"CanCreateMixin(*args, **kwargs)","text":"

Bases: View

Protect any child view that would create an object.

Raises:

Type Description PermissionDenied

If the user has not the necessary permission to create the object of the view.

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/core/views/#core.views.LockError","title":"LockError","text":"

Bases: Exception

There was a lock error on the object.

"},{"location":"reference/core/views/#core.views.Page","title":"Page","text":"

Bases: Model

The page class to build a Wiki Each page may have a parent and it's URL is of the form my.site/page/// It has an ID field, but don't use it, since it's only there for DB part, and because compound primary key is awkward! Prefere querying pages with Page.get_page_by_full_name().

Be careful with the _full_name attribute: this field may not be valid until you call save(). It's made for fast query, but don't rely on it when playing with a Page object, use get_full_name() instead!

"},{"location":"reference/core/views/#core.views.Page.save","title":"save(*args, **kwargs)","text":"

Performs some needed actions before and after saving a page in database.

Source code in core/models.py
def save(self, *args, **kwargs):\n    \"\"\"Performs some needed actions before and after saving a page in database.\"\"\"\n    locked = kwargs.pop(\"force_lock\", False)\n    if not locked:\n        locked = self.is_locked()\n    if not locked:\n        raise NotLocked(\"The page is not locked and thus can not be saved\")\n    self.full_clean()\n    if not self.id:\n        super().save(\n            *args, **kwargs\n        )  # Save a first time to correctly set _full_name\n    # This reset the _full_name just before saving to maintain a coherent field quicker for queries than the\n    # recursive method\n    # It also update all the children to maintain correct names\n    self._full_name = self.get_full_name()\n    for c in self.children.all():\n        c.save()\n    super().save(*args, **kwargs)\n    self.unset_lock()\n
"},{"location":"reference/core/views/#core.views.Page.get_page_by_full_name","title":"get_page_by_full_name(name) staticmethod","text":"

Quicker to get a page with that method rather than building the request every time.

Source code in core/models.py
@staticmethod\ndef get_page_by_full_name(name):\n    \"\"\"Quicker to get a page with that method rather than building the request every time.\"\"\"\n    return Page.objects.filter(_full_name=name).first()\n
"},{"location":"reference/core/views/#core.views.Page.clean","title":"clean()","text":"

Cleans up only the name for the moment, but this can be used to make any treatment before saving the object.

Source code in core/models.py
def clean(self):\n    \"\"\"Cleans up only the name for the moment, but this can be used to make any treatment before saving the object.\"\"\"\n    if \"/\" in self.name:\n        self.name = self.name.split(\"/\")[-1]\n    if (\n        Page.objects.exclude(pk=self.pk)\n        .filter(_full_name=self.get_full_name())\n        .exists()\n    ):\n        raise ValidationError(_(\"Duplicate page\"), code=\"duplicate\")\n    super().clean()\n    if self.parent is not None and self in self.get_parent_list():\n        raise ValidationError(_(\"Loop in page tree\"), code=\"loop\")\n
"},{"location":"reference/core/views/#core.views.Page.is_locked","title":"is_locked()","text":"

Is True if the page is locked, False otherwise.

This is where the timeout is handled, so a locked page for which the timeout is reach will be unlocked and this function will return False.

Source code in core/models.py
def is_locked(self):\n    \"\"\"Is True if the page is locked, False otherwise.\n\n    This is where the timeout is handled,\n    so a locked page for which the timeout is reach will be unlocked and this\n    function will return False.\n    \"\"\"\n    if self.lock_timeout and (\n        timezone.now() - self.lock_timeout > timedelta(minutes=5)\n    ):\n        self.unset_lock()\n    return (\n        self.lock_user\n        and self.lock_timeout\n        and (timezone.now() - self.lock_timeout < timedelta(minutes=5))\n    )\n
"},{"location":"reference/core/views/#core.views.Page.set_lock","title":"set_lock(user)","text":"

Sets a lock on the current page or raise an AlreadyLocked exception.

Source code in core/models.py
def set_lock(self, user):\n    \"\"\"Sets a lock on the current page or raise an AlreadyLocked exception.\"\"\"\n    if self.is_locked() and self.get_lock() != user:\n        raise AlreadyLocked(\"The page is already locked by someone else\")\n    self.lock_user = user\n    self.lock_timeout = timezone.now()\n    super().save()\n
"},{"location":"reference/core/views/#core.views.Page.set_lock_recursive","title":"set_lock_recursive(user)","text":"

Locks recursively all the child pages for editing properties.

Source code in core/models.py
def set_lock_recursive(self, user):\n    \"\"\"Locks recursively all the child pages for editing properties.\"\"\"\n    for p in self.children.all():\n        p.set_lock_recursive(user)\n    self.set_lock(user)\n
"},{"location":"reference/core/views/#core.views.Page.unset_lock_recursive","title":"unset_lock_recursive()","text":"

Unlocks recursively all the child pages.

Source code in core/models.py
def unset_lock_recursive(self):\n    \"\"\"Unlocks recursively all the child pages.\"\"\"\n    for p in self.children.all():\n        p.unset_lock_recursive()\n    self.unset_lock()\n
"},{"location":"reference/core/views/#core.views.Page.unset_lock","title":"unset_lock()","text":"

Always try to unlock, even if there is no lock.

Source code in core/models.py
def unset_lock(self):\n    \"\"\"Always try to unlock, even if there is no lock.\"\"\"\n    self.lock_user = None\n    self.lock_timeout = None\n    super().save()\n
"},{"location":"reference/core/views/#core.views.Page.get_lock","title":"get_lock()","text":"

Returns the page's mutex containing the time and the user in a dict.

Source code in core/models.py
def get_lock(self):\n    \"\"\"Returns the page's mutex containing the time and the user in a dict.\"\"\"\n    if self.lock_user:\n        return self.lock_user\n    raise NotLocked(\"The page is not locked and thus can not return its user\")\n
"},{"location":"reference/core/views/#core.views.Page.get_full_name","title":"get_full_name()","text":"

Computes the real full_name of the page based on its name and its parent's name You can and must rely on this function when working on a page object that is not freshly fetched from the DB (For example when treating a Page object coming from a form).

Source code in core/models.py
def get_full_name(self):\n    \"\"\"Computes the real full_name of the page based on its name and its parent's name\n    You can and must rely on this function when working on a page object that is not freshly fetched from the DB\n    (For example when treating a Page object coming from a form).\n    \"\"\"\n    if self.parent is None:\n        return self.name\n    return f\"{self.parent.get_full_name()}/{self.name}\"\n
"},{"location":"reference/core/views/#core.views.PageRev","title":"PageRev","text":"

Bases: Model

True content of the page.

Each page object has a revisions field that is a list of PageRev, ordered by date. my_page.revisions.last() gives the PageRev object that is the most up-to-date, and thus, is the real content of the page. The content is in PageRev.title and PageRev.content .

"},{"location":"reference/core/views/#core.views.CanEditPagePropMixin","title":"CanEditPagePropMixin","text":"

Bases: CanEditPropMixin

"},{"location":"reference/core/views/#core.views.PageListView","title":"PageListView","text":"

Bases: CanViewMixin, ListView

"},{"location":"reference/core/views/#core.views.PageView","title":"PageView","text":"

Bases: CanViewMixin, DetailView

"},{"location":"reference/core/views/#core.views.PageHistView","title":"PageHistView","text":"

Bases: CanViewMixin, DetailView

"},{"location":"reference/core/views/#core.views.PageRevView","title":"PageRevView","text":"

Bases: CanViewMixin, DetailView

"},{"location":"reference/core/views/#core.views.PageCreateView","title":"PageCreateView(*args, **kwargs)","text":"

Bases: CanCreateMixin, CreateView

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/core/views/#core.views.PagePropView","title":"PagePropView","text":"

Bases: CanEditPagePropMixin, UpdateView

"},{"location":"reference/core/views/#core.views.PageEditViewBase","title":"PageEditViewBase","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/core/views/#core.views.PageEditView","title":"PageEditView","text":"

Bases: PageEditViewBase

"},{"location":"reference/core/views/#core.views.PageDeleteView","title":"PageDeleteView","text":"

Bases: CanEditPagePropMixin, DeleteView

"},{"location":"reference/core/views/#core.views.Notification","title":"Notification","text":"

Bases: Model

"},{"location":"reference/core/views/#core.views.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/core/views/#core.views.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/core/views/#core.views.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/core/views/#core.views.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/core/views/#core.views.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/core/views/#core.views.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/core/views/#core.views.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/core/views/#core.views.NotificationList","title":"NotificationList","text":"

Bases: ListView

"},{"location":"reference/core/views/#core.views.CanEditMixin","title":"CanEditMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to edit this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/core/views/#core.views.CanEditPropMixin","title":"CanEditPropMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has owner permissions on the child view object.

In other word, you can make a view with this view as parent, and it will be retricted to the users that are in the object's owner_group or that pass the obj.can_be_viewed_by test.

Raises:

Type Description PermissionDenied

If the user cannot see the object

"},{"location":"reference/core/views/#core.views.CanViewMixin","title":"CanViewMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to view this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/core/views/#core.views.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/core/views/#core.views.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/core/views/#core.views.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/core/views/#core.views.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/core/views/#core.views.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/core/views/#core.views.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/core/views/#core.views.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/core/views/#core.views.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/core/views/#core.views.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/core/views/#core.views.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/core/views/#core.views.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/core/views/#core.views.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/core/views/#core.views.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/core/views/#core.views.Gift","title":"Gift","text":"

Bases: Model

"},{"location":"reference/core/views/#core.views.Preferences","title":"Preferences","text":"

Bases: Model

"},{"location":"reference/core/views/#core.views.QuickNotifMixin","title":"QuickNotifMixin","text":""},{"location":"reference/core/views/#core.views.QuickNotifMixin.get_context_data","title":"get_context_data(**kwargs)","text":"

Add quick notifications to context.

Source code in core/views/mixins.py
def get_context_data(self, **kwargs):\n    \"\"\"Add quick notifications to context.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"quick_notifs\"] = []\n    for n in self.quick_notif_list:\n        kwargs[\"quick_notifs\"].append(settings.SITH_QUICK_NOTIF[n])\n    for key, val in settings.SITH_QUICK_NOTIF.items():\n        for gk in self.request.GET:\n            if key == gk:\n                kwargs[\"quick_notifs\"].append(val)\n    return kwargs\n
"},{"location":"reference/core/views/#core.views.TabedViewMixin","title":"TabedViewMixin","text":"

Bases: View

Basic functions for displaying tabs in the template.

"},{"location":"reference/core/views/#core.views.SithLoginView","title":"SithLoginView","text":"

Bases: LoginView

The login View.

"},{"location":"reference/core/views/#core.views.SithPasswordChangeView","title":"SithPasswordChangeView","text":"

Bases: PasswordChangeView

Allows a user to change its password.

"},{"location":"reference/core/views/#core.views.SithPasswordChangeDoneView","title":"SithPasswordChangeDoneView","text":"

Bases: PasswordChangeDoneView

Allows a user to change its password.

"},{"location":"reference/core/views/#core.views.SithPasswordResetView","title":"SithPasswordResetView","text":"

Bases: PasswordResetView

Allows someone to enter an email address for resetting password.

"},{"location":"reference/core/views/#core.views.SithPasswordResetDoneView","title":"SithPasswordResetDoneView","text":"

Bases: PasswordResetDoneView

Confirm that the reset email has been sent.

"},{"location":"reference/core/views/#core.views.SithPasswordResetConfirmView","title":"SithPasswordResetConfirmView","text":"

Bases: PasswordResetConfirmView

Provide a reset password form.

"},{"location":"reference/core/views/#core.views.SithPasswordResetCompleteView","title":"SithPasswordResetCompleteView","text":"

Bases: PasswordResetCompleteView

Confirm the password has successfully been reset.

"},{"location":"reference/core/views/#core.views.UserCreationView","title":"UserCreationView","text":"

Bases: FormView

"},{"location":"reference/core/views/#core.views.UserTabsMixin","title":"UserTabsMixin","text":"

Bases: TabedViewMixin

"},{"location":"reference/core/views/#core.views.UserView","title":"UserView","text":"

Bases: UserTabsMixin, CanViewMixin, DetailView

Display a user's profile.

"},{"location":"reference/core/views/#core.views.UserPicturesView","title":"UserPicturesView","text":"

Bases: UserTabsMixin, CanViewMixin, DetailView

Display a user's pictures.

"},{"location":"reference/core/views/#core.views.UserGodfathersView","title":"UserGodfathersView","text":"

Bases: UserTabsMixin, CanViewMixin, DetailView, FormView

Display a user's godfathers.

"},{"location":"reference/core/views/#core.views.UserGodfathersTreeView","title":"UserGodfathersTreeView","text":"

Bases: UserTabsMixin, CanViewMixin, DetailView

Display a user's family tree.

"},{"location":"reference/core/views/#core.views.UserStatsView","title":"UserStatsView","text":"

Bases: UserTabsMixin, CanViewMixin, DetailView

Display a user's stats.

"},{"location":"reference/core/views/#core.views.UserMiniView","title":"UserMiniView","text":"

Bases: CanViewMixin, DetailView

Display a user's profile.

"},{"location":"reference/core/views/#core.views.UserListView","title":"UserListView","text":"

Bases: ListView, CanEditPropMixin

Displays the user list.

"},{"location":"reference/core/views/#core.views.UserUpdateProfileView","title":"UserUpdateProfileView","text":"

Bases: UserTabsMixin, CanEditMixin, UpdateView

Edit a user's profile.

"},{"location":"reference/core/views/#core.views.UserUpdateProfileView.remove_restricted_fields","title":"remove_restricted_fields(request)","text":"

Removes edit_once and board_only fields.

Source code in core/views/user.py
def remove_restricted_fields(self, request):\n    \"\"\"Removes edit_once and board_only fields.\"\"\"\n    for i in self.edit_once:\n        if getattr(self.form.instance, i) and not (\n            request.user.is_board_member or request.user.is_root\n        ):\n            self.form.fields.pop(i, None)\n    for i in self.board_only:\n        if not (request.user.is_board_member or request.user.is_root):\n            self.form.fields.pop(i, None)\n
"},{"location":"reference/core/views/#core.views.UserClubView","title":"UserClubView","text":"

Bases: UserTabsMixin, CanViewMixin, DetailView

Display the user's club(s).

"},{"location":"reference/core/views/#core.views.UserPreferencesView","title":"UserPreferencesView","text":"

Bases: UserTabsMixin, CanEditMixin, UpdateView

Edit a user's preferences.

"},{"location":"reference/core/views/#core.views.UserUpdateGroupView","title":"UserUpdateGroupView","text":"

Bases: UserTabsMixin, CanEditPropMixin, UpdateView

Edit a user's groups.

"},{"location":"reference/core/views/#core.views.UserToolsView","title":"UserToolsView","text":"

Bases: LoginRequiredMixin, QuickNotifMixin, UserTabsMixin, TemplateView

Displays the logged user's tools.

"},{"location":"reference/core/views/#core.views.UserAccountBase","title":"UserAccountBase","text":"

Bases: UserTabsMixin, DetailView

Base class for UserAccount.

"},{"location":"reference/core/views/#core.views.UserAccountView","title":"UserAccountView","text":"

Bases: UserAccountBase

Display a user's account.

"},{"location":"reference/core/views/#core.views.UserAccountDetailView","title":"UserAccountDetailView","text":"

Bases: UserAccountBase, YearMixin, MonthMixin

Display a user's account for month.

"},{"location":"reference/core/views/#core.views.GiftCreateView","title":"GiftCreateView","text":"

Bases: CreateView

"},{"location":"reference/core/views/#core.views.GiftDeleteView","title":"GiftDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

"},{"location":"reference/core/views/#core.views.forbidden","title":"forbidden(request, exception)","text":"Source code in core/views/__init__.py
def forbidden(request, exception):\n    context = {\"next\": request.path, \"form\": LoginForm()}\n    if popup := request.resolver_match.kwargs.get(\"popup\"):\n        context[\"popup\"] = popup\n    return HttpResponseForbidden(render(request, \"core/403.jinja\", context=context))\n
"},{"location":"reference/core/views/#core.views.not_found","title":"not_found(request, exception)","text":"Source code in core/views/__init__.py
def not_found(request, exception):\n    return HttpResponseNotFound(\n        render(request, \"core/404.jinja\", context={\"exception\": exception})\n    )\n
"},{"location":"reference/core/views/#core.views.internal_servor_error","title":"internal_servor_error(request)","text":"Source code in core/views/__init__.py
def internal_servor_error(request):\n    request.sentry_last_event_id = last_event_id\n    return HttpResponseServerError(render(request, \"core/500.jinja\"))\n
"},{"location":"reference/core/views/#core.views.can_view","title":"can_view(obj, user)","text":"

Can the user see the object.

Parameters:

Name Type Description Default obj Any

Object to test for permission

required user User

core.models.User to test permissions against

required

Returns:

Type Description bool

True if user is authorized to see object else False

Example
if not can_view(self.object ,request.user):\n    raise PermissionDenied\n
Source code in core/auth/mixins.py
def can_view(obj: Any, user: User) -> bool:\n    \"\"\"Can the user see the object.\n\n    Args:\n        obj: Object to test for permission\n        user: core.models.User to test permissions against\n\n    Returns:\n        True if user is authorized to see object else False\n\n    Example:\n        ```python\n        if not can_view(self.object ,request.user):\n            raise PermissionDenied\n        ```\n    \"\"\"\n    if obj is None or user.can_view(obj):\n        return True\n    return can_edit(obj, user)\n
"},{"location":"reference/core/views/#core.views.send_raw_file","title":"send_raw_file(path)","text":"

Send a file located in the MEDIA_ROOT

This handles all the logic of using production reverse proxy or debug server.

THIS DOESN'T CHECK ANY PERMISSIONS !

Source code in core/views/files.py
def send_raw_file(path: Path) -> HttpResponse:\n    \"\"\"Send a file located in the MEDIA_ROOT\n\n    This handles all the logic of using production reverse proxy or debug server.\n\n    THIS DOESN'T CHECK ANY PERMISSIONS !\n    \"\"\"\n    if not path.is_relative_to(settings.MEDIA_ROOT):\n        raise Http404\n\n    if not path.is_file() or not path.exists():\n        raise Http404\n\n    response = HttpResponse(\n        headers={\"Content-Disposition\": f'inline; filename=\"{quote(path.name)}\"'}\n    )\n    if not settings.DEBUG:\n        # When receiving a response with the Accel-Redirect header,\n        # the reverse proxy will automatically handle the file sending.\n        # This is really hard to test (thus isn't tested)\n        # so please do not mess with this.\n        response[\"Content-Type\"] = \"\"  # automatically set by nginx\n        response[\"X-Accel-Redirect\"] = quote(\n            urljoin(settings.MEDIA_URL, str(path.relative_to(settings.MEDIA_ROOT)))\n        )\n        return response\n\n    with open(path, \"rb\") as filename:\n        response.content = FileWrapper(filename)\n        response[\"Content-Type\"] = mimetypes.guess_type(path)[0]\n        response[\"Last-Modified\"] = http_date(path.stat().st_mtime)\n        response[\"Content-Length\"] = path.stat().st_size\n        return response\n
"},{"location":"reference/core/views/#core.views.send_file","title":"send_file(request, file_id, file_class=SithFile, file_attr='file')","text":"

Send a protected file, if the user can see it.

In prod, the server won't handle the download itself, but set the appropriate headers in the response to make the reverse-proxy deal with it. In debug mode, the server will directly send the file.

Source code in core/views/files.py
def send_file(\n    request: HttpRequest,\n    file_id: int,\n    file_class: type[SithFile] = SithFile,\n    file_attr: str = \"file\",\n) -> HttpResponse:\n    \"\"\"Send a protected file, if the user can see it.\n\n    In prod, the server won't handle the download itself,\n    but set the appropriate headers in the response to make the reverse-proxy\n    deal with it.\n    In debug mode, the server will directly send the file.\n    \"\"\"\n    f = get_object_or_404(file_class, id=file_id)\n    if not can_view(f, request.user) and not is_logged_in_counter(request):\n        raise PermissionDenied\n    name = getattr(f, file_attr).name\n\n    return send_raw_file(settings.MEDIA_ROOT / name)\n
"},{"location":"reference/core/views/#core.views.index","title":"index(request, context=None)","text":"Source code in core/views/site.py
def index(request, context=None):\n    from com.views import NewsListView\n\n    return NewsListView.as_view()(request)\n
"},{"location":"reference/core/views/#core.views.notification","title":"notification(request, notif_id)","text":"Source code in core/views/site.py
def notification(request, notif_id):\n    notif = Notification.objects.filter(id=notif_id).first()\n    if notif:\n        if notif.type not in settings.SITH_PERMANENT_NOTIFICATIONS:\n            notif.viewed = True\n        else:\n            notif.callback()\n        notif.save()\n        return redirect(notif.url)\n    return redirect(\"/\")\n
"},{"location":"reference/core/views/#core.views.search_user","title":"search_user(query)","text":"Source code in core/views/site.py
def search_user(query):\n    try:\n        # slugify turns everything into ascii and every whitespace into -\n        # it ends by removing duplicate - (so ' - ' will turn into '-')\n        # replace('-', ' ') because search is whitespace based\n        query = slugify(query).replace(\"-\", \" \")\n        # TODO: is this necessary?\n        query = html.escape(query)\n        res = (\n            SearchQuerySet()\n            .models(User)\n            .autocomplete(auto=query)\n            .order_by(\"-last_login\")\n            .load_all()[:20]\n        )\n        return [r.object for r in res]\n    except TypeError:\n        return []\n
"},{"location":"reference/core/views/#core.views.search_club","title":"search_club(query, *, as_json=False)","text":"Source code in core/views/site.py
def search_club(query, *, as_json=False):\n    clubs = []\n    if query:\n        clubs = Club.objects.filter(name__icontains=query).all()\n        clubs = clubs[:5]\n        if as_json:\n            # Re-loads json to avoid double encoding by JsonResponse, but still benefit from serializers\n            clubs = json.loads(serializers.serialize(\"json\", clubs, fields=(\"name\")))\n        else:\n            clubs = list(clubs)\n    return clubs\n
"},{"location":"reference/core/views/#core.views.search_view","title":"search_view(request)","text":"Source code in core/views/site.py
@login_required\ndef search_view(request):\n    result = {\n        \"users\": search_user(request.GET.get(\"query\", \"\")),\n        \"clubs\": search_club(request.GET.get(\"query\", \"\")),\n    }\n    return render(request, \"core/search.jinja\", context={\"result\": result})\n
"},{"location":"reference/core/views/#core.views.search_user_json","title":"search_user_json(request)","text":"Source code in core/views/site.py
@login_required\ndef search_user_json(request):\n    result = {\"users\": search_user(request.GET.get(\"query\", \"\"))}\n    return JsonResponse(result)\n
"},{"location":"reference/core/views/#core.views.search_json","title":"search_json(request)","text":"Source code in core/views/site.py
@login_required\ndef search_json(request):\n    result = {\n        \"users\": search_user(request.GET.get(\"query\", \"\")),\n        \"clubs\": search_club(request.GET.get(\"query\", \"\"), as_json=True),\n    }\n    return JsonResponse(result)\n
"},{"location":"reference/core/views/#core.views.logout","title":"logout(request)","text":"

The logout view.

Source code in core/views/user.py
def logout(request):\n    \"\"\"The logout view.\"\"\"\n    return views.logout_then_login(request)\n
"},{"location":"reference/core/views/#core.views.password_root_change","title":"password_root_change(request, user_id)","text":"

Allows a root user to change someone's password.

Source code in core/views/user.py
def password_root_change(request, user_id):\n    \"\"\"Allows a root user to change someone's password.\"\"\"\n    if not request.user.is_root:\n        raise PermissionDenied\n    user = User.objects.filter(id=user_id).first()\n    if not user:\n        raise Http404(\"User not found\")\n    if request.method == \"POST\":\n        form = views.SetPasswordForm(user=user, data=request.POST)\n        if form.is_valid():\n            form.save()\n            return redirect(\"core:password_change_done\")\n    else:\n        form = views.SetPasswordForm(user=user)\n    return TemplateResponse(\n        request, \"core/password_change.jinja\", {\"form\": form, \"target\": user}\n    )\n
"},{"location":"reference/core/views/#core.views.delete_user_godfather","title":"delete_user_godfather(request, user_id, godfather_id, is_father)","text":"Source code in core/views/user.py
def delete_user_godfather(request, user_id, godfather_id, is_father):\n    user_is_admin = request.user.is_root or request.user.is_board_member\n    if user_id != request.user.id and not user_is_admin:\n        raise PermissionDenied()\n    user = get_object_or_404(User, id=user_id)\n    to_remove = get_object_or_404(User, id=godfather_id)\n    if is_father:\n        user.godfathers.remove(to_remove)\n    else:\n        user.godchildren.remove(to_remove)\n    return redirect(\"core:user_godfathers\", user_id=user_id)\n
"},{"location":"reference/counter/models/","title":"Models","text":""},{"location":"reference/counter/models/#counter.models.PAYMENT_METHOD","title":"PAYMENT_METHOD = [('CHECK', _('Check')), ('CASH', _('Cash')), ('CARD', _('Credit card'))] module-attribute","text":""},{"location":"reference/counter/models/#counter.models.CurrencyField","title":"CurrencyField(*args, **kwargs)","text":"

Bases: DecimalField

Custom database field used for currency.

Source code in accounting/models.py
def __init__(self, *args, **kwargs):\n    kwargs[\"max_digits\"] = 12\n    kwargs[\"decimal_places\"] = 2\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/counter/models/#counter.models.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/counter/models/#counter.models.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/counter/models/#counter.models.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/counter/models/#counter.models.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/counter/models/#counter.models.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/counter/models/#counter.models.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/counter/models/#counter.models.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/counter/models/#counter.models.ResizedImageField","title":"ResizedImageField(width=None, height=None, force_format=None, **kwargs)","text":"

Bases: ImageField

A field that automatically resizes images to a given size.

This field is useful for profile pictures or product icons, for example.

The final size of the image is determined by the width and height parameters :

If the force_format parameter is given, the image will be converted to this format.

Examples:

To resize an image with a height of 100px, without changing the ratio, and a format of WEBP :

class Product(models.Model):\n    icon = ResizedImageField(height=100, force_format=\"WEBP\")\n

To explicitly resize an image to 100x100px (but possibly change the ratio) :

class Product(models.Model):\n    icon = ResizedImageField(width=100, height=100)\n

Raises:

Type Description FieldError

If neither width nor height is given

Parameters:

Name Type Description Default width int | None

If given, the width of the resized image

None height int | None

If given, the height of the resized image

None force_format str | None

If given, the image will be converted to this format

None Source code in core/fields.py
def __init__(\n    self,\n    width: int | None = None,\n    height: int | None = None,\n    force_format: str | None = None,\n    **kwargs,\n):\n    if width is None and height is None:\n        raise FieldError(\n            f\"{self.__class__.__name__} requires \"\n            \"width, height or both, but got neither\"\n        )\n    self.width = width\n    self.height = height\n    self.force_format = force_format\n    super().__init__(**kwargs)\n
"},{"location":"reference/counter/models/#counter.models.Group","title":"Group","text":"

Bases: Group

Wrapper around django.auth.Group

"},{"location":"reference/counter/models/#counter.models.Notification","title":"Notification","text":"

Bases: Model

"},{"location":"reference/counter/models/#counter.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/counter/models/#counter.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/counter/models/#counter.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/counter/models/#counter.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/counter/models/#counter.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/counter/models/#counter.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/counter/models/#counter.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/counter/models/#counter.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/counter/models/#counter.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/counter/models/#counter.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/counter/models/#counter.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/counter/models/#counter.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/counter/models/#counter.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/counter/models/#counter.models.CustomerQuerySet","title":"CustomerQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/counter/models/#counter.models.CustomerQuerySet.update_amount","title":"update_amount()","text":"

Update the amount of all customers selected by this queryset.

The result is given as the sum of all refills minus the sum of all purchases.

Returns:

Type Description int

The number of updated rows.

Source code in counter/models.py
def update_amount(self) -> int:\n    \"\"\"Update the amount of all customers selected by this queryset.\n\n    The result is given as the sum of all refills minus the sum of all purchases.\n\n    Returns:\n        The number of updated rows.\n\n    Warnings:\n        The execution time of this query grows really quickly.\n        When updating 500 customers, it may take around a second.\n        If you try to update all customers at once, the execution time\n        goes up to tens of seconds.\n        Use this either on a small subset of the `Customer` table,\n        or execute it inside an independent task\n        (like a Celery task or a management command).\n    \"\"\"\n    money_in = Subquery(\n        Refilling.objects.filter(customer=OuterRef(\"pk\"))\n        .values(\"customer_id\")  # group by customer\n        .annotate(res=Sum(F(\"amount\"), default=0))\n        .values(\"res\")\n    )\n    money_out = Subquery(\n        Selling.objects.filter(customer=OuterRef(\"pk\"))\n        .values(\"customer_id\")\n        .annotate(res=Sum(F(\"unit_price\") * F(\"quantity\"), default=0))\n        .values(\"res\")\n    )\n    return self.update(amount=Coalesce(money_in - money_out, Decimal(\"0\")))\n
"},{"location":"reference/counter/models/#counter.models.Customer","title":"Customer","text":"

Bases: Model

Customer data of a User.

It adds some basic customers' information, such as the account ID, and is used by other accounting classes as reference to the customer, rather than using User.

"},{"location":"reference/counter/models/#counter.models.Customer.can_buy","title":"can_buy property","text":"

Check if whether this customer has the right to purchase any item.

This must be not confused with the Product.can_be_sold_to(user) method as the present method returns an information about a customer whereas the other tells something about the relation between a User (not a Customer, don't mix them) and a Product.

"},{"location":"reference/counter/models/#counter.models.Customer.save","title":"save(*args, allow_negative=False, is_selling=False, **kwargs)","text":"

is_selling : tell if the current action is a selling allow_negative : ignored if not a selling. Allow a selling to put the account in negative Those two parameters avoid blocking the save method of a customer if his account is negative.

Source code in counter/models.py
def save(self, *args, allow_negative=False, is_selling=False, **kwargs):\n    \"\"\"is_selling : tell if the current action is a selling\n    allow_negative : ignored if not a selling. Allow a selling to put the account in negative\n    Those two parameters avoid blocking the save method of a customer if his account is negative.\n    \"\"\"\n    if self.amount < 0 and (is_selling and not allow_negative):\n        raise ValidationError(_(\"Not enough money\"))\n    super().save(*args, **kwargs)\n
"},{"location":"reference/counter/models/#counter.models.Customer.get_or_create","title":"get_or_create(user) classmethod","text":"

Work in pretty much the same way as the usual get_or_create method, but with the default field replaced by some under the hood.

If the user has an account, return it as is. Else create a new account with no money on it and a new unique account id

Example : ::

user = User.objects.get(pk=1)\naccount, created = Customer.get_or_create(user)\nif created:\n    print(f\"created a new account with id {account.id}\")\nelse:\n    print(f\"user has already an account, with {account.id} \u20ac on it\"\n
Source code in counter/models.py
@classmethod\ndef get_or_create(cls, user: User) -> tuple[Customer, bool]:\n    \"\"\"Work in pretty much the same way as the usual get_or_create method,\n    but with the default field replaced by some under the hood.\n\n    If the user has an account, return it as is.\n    Else create a new account with no money on it and a new unique account id\n\n    Example : ::\n\n        user = User.objects.get(pk=1)\n        account, created = Customer.get_or_create(user)\n        if created:\n            print(f\"created a new account with id {account.id}\")\n        else:\n            print(f\"user has already an account, with {account.id} \u20ac on it\"\n    \"\"\"\n    if hasattr(user, \"customer\"):\n        return user.customer, False\n\n    # account_id are always a number with a letter appended\n    account_id = (\n        Customer.objects.order_by(Length(\"account_id\"), \"account_id\")\n        .values(\"account_id\")\n        .last()\n    )\n    if account_id is None:\n        # legacy from the old site\n        account = cls.objects.create(user=user, account_id=\"1504a\")\n        return account, True\n\n    account_id = account_id[\"account_id\"]\n    account_num = int(account_id[:-1])\n    while Customer.objects.filter(account_id=account_id).exists():\n        # when entering the first iteration, we are using an already existing account id\n        # so the loop should always execute at least one time\n        account_num += 1\n        account_id = f\"{account_num}{random.choice(string.ascii_lowercase)}\"\n\n    account = cls.objects.create(user=user, account_id=account_id)\n    return account, True\n
"},{"location":"reference/counter/models/#counter.models.BillingInfo","title":"BillingInfo","text":"

Bases: Model

Represent the billing information of a user, which are required by the 3D-Secure v2 system used by the etransaction module.

"},{"location":"reference/counter/models/#counter.models.BillingInfo.to_3dsv2_xml","title":"to_3dsv2_xml()","text":"

Convert the data from this model into a xml usable by the online paying service of the Cr\u00e9dit Agricole bank. see : https://www.ca-moncommerce.com/espace-client-mon-commerce/up2pay-e-transactions/ma-documentation/manuel-dintegration-focus-3ds-v2/principes-generaux/#integration-3dsv2-developpeur-webmaster.

Source code in counter/models.py
def to_3dsv2_xml(self) -> str:\n    \"\"\"Convert the data from this model into a xml usable\n    by the online paying service of the Cr\u00e9dit Agricole bank.\n    see : `https://www.ca-moncommerce.com/espace-client-mon-commerce/up2pay-e-transactions/ma-documentation/manuel-dintegration-focus-3ds-v2/principes-generaux/#integration-3dsv2-developpeur-webmaster`.\n    \"\"\"\n    data = {\n        \"Address\": {\n            \"FirstName\": self.first_name,\n            \"LastName\": self.last_name,\n            \"Address1\": self.address_1,\n            \"ZipCode\": self.zip_code,\n            \"City\": self.city,\n            \"CountryCode\": self.country.numeric,  # ISO-3166-1 numeric code\n            \"MobilePhone\": self.phone_number.as_national.replace(\" \", \"\"),\n            \"CountryCodeMobilePhone\": f\"+{self.phone_number.country_code}\",\n        }\n    }\n    if self.address_2:\n        data[\"Address\"][\"Address2\"] = self.address_2\n    xml = dict2xml(data, wrap=\"Billing\", newlines=False)\n    return '<?xml version=\"1.0\" encoding=\"UTF-8\" ?>' + xml\n
"},{"location":"reference/counter/models/#counter.models.AccountDumpQuerySet","title":"AccountDumpQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/counter/models/#counter.models.AccountDumpQuerySet.ongoing","title":"ongoing()","text":"

Filter dump operations that are not completed yet.

Source code in counter/models.py
def ongoing(self) -> Self:\n    \"\"\"Filter dump operations that are not completed yet.\"\"\"\n    return self.filter(dump_operation=None)\n
"},{"location":"reference/counter/models/#counter.models.AccountDump","title":"AccountDump","text":"

Bases: Model

The process of dumping an account.

"},{"location":"reference/counter/models/#counter.models.ProductType","title":"ProductType","text":"

Bases: OrderedModel

A product type.

Useful only for categorizing.

"},{"location":"reference/counter/models/#counter.models.ProductType.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in counter/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID)\n
"},{"location":"reference/counter/models/#counter.models.Product","title":"Product","text":"

Bases: Model

A product, with all its related information.

"},{"location":"reference/counter/models/#counter.models.Product.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in counter/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(\n        pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID\n    ) or user.is_in_group(pk=settings.SITH_GROUP_COUNTER_ADMIN_ID)\n
"},{"location":"reference/counter/models/#counter.models.Product.can_be_sold_to","title":"can_be_sold_to(user)","text":"

Check if whether the user given in parameter has the right to buy this product or not.

This must be not confused with the Customer.can_buy() method as the present method returns an information about the relation between a User and a Product, whereas the other tells something about a Customer (and not a user, they are not the same model).

Returns:

Type Description bool

True if the user can buy this product else False

Warning

This performs a db query, thus you can quickly have a N+1 queries problem if you call it in a loop. Hopefully, you can avoid that if you prefetch the buying_groups :

user = User.objects.get(username=\"foobar\")\nproducts = [\n    p\n    for p in Product.objects.prefetch_related(\"buying_groups\")\n    if p.can_be_sold_to(user)\n]\n
Source code in counter/models.py
def can_be_sold_to(self, user: User) -> bool:\n    \"\"\"Check if whether the user given in parameter has the right to buy\n    this product or not.\n\n    This must be not confused with the Customer.can_buy()\n    method as the present method returns an information\n    about the relation between a User and a Product,\n    whereas the other tells something about a Customer\n    (and not a user, they are not the same model).\n\n    Returns:\n        True if the user can buy this product else False\n\n    Warning:\n        This performs a db query, thus you can quickly have\n        a N+1 queries problem if you call it in a loop.\n        Hopefully, you can avoid that if you prefetch the buying_groups :\n\n        ```python\n        user = User.objects.get(username=\"foobar\")\n        products = [\n            p\n            for p in Product.objects.prefetch_related(\"buying_groups\")\n            if p.can_be_sold_to(user)\n        ]\n        ```\n    \"\"\"\n    buying_groups = list(self.buying_groups.all())\n    if not buying_groups:\n        return True\n    return any(user.is_in_group(pk=group.id) for group in buying_groups)\n
"},{"location":"reference/counter/models/#counter.models.CounterQuerySet","title":"CounterQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/counter/models/#counter.models.CounterQuerySet.annotate_has_barman","title":"annotate_has_barman(user)","text":"

Annotate the queryset with the user_is_barman field.

For each counter, this field has value True if the user is a barman of this counter, else False.

Parameters:

Name Type Description Default user User

the user we want to check if he is a barman

required

Examples:

sli = User.objects.get(username=\"sli\")\ncounters = (\n    Counter.objects\n    .annotate_has_barman(sli)  # add the user_has_barman boolean field\n    .filter(has_annotated_barman=True)  # keep only counters where this user is barman\n)\nprint(\"Sli est barman dans les comptoirs suivants :\")\nfor counter in counters:\n    print(f\"- {counter.name}\")\n
Source code in counter/models.py
def annotate_has_barman(self, user: User) -> Self:\n    \"\"\"Annotate the queryset with the `user_is_barman` field.\n\n    For each counter, this field has value True if the user\n    is a barman of this counter, else False.\n\n    Args:\n        user: the user we want to check if he is a barman\n\n    Examples:\n        ```python\n        sli = User.objects.get(username=\"sli\")\n        counters = (\n            Counter.objects\n            .annotate_has_barman(sli)  # add the user_has_barman boolean field\n            .filter(has_annotated_barman=True)  # keep only counters where this user is barman\n        )\n        print(\"Sli est barman dans les comptoirs suivants :\")\n        for counter in counters:\n            print(f\"- {counter.name}\")\n        ```\n    \"\"\"\n    subquery = user.counters.filter(pk=OuterRef(\"pk\"))\n    return self.annotate(has_annotated_barman=Exists(subquery))\n
"},{"location":"reference/counter/models/#counter.models.CounterQuerySet.annotate_is_open","title":"annotate_is_open()","text":"

Annotate tue queryset with the is_open field.

For each counter, if is_open=True, then the counter is currently opened. Else the counter is closed.

Source code in counter/models.py
def annotate_is_open(self) -> Self:\n    \"\"\"Annotate tue queryset with the `is_open` field.\n\n    For each counter, if `is_open=True`, then the counter is currently opened.\n    Else the counter is closed.\n    \"\"\"\n    return self.annotate(\n        is_open=Exists(\n            Permanency.objects.filter(counter_id=OuterRef(\"pk\"), end=None)\n        )\n    )\n
"},{"location":"reference/counter/models/#counter.models.CounterQuerySet.handle_timeout","title":"handle_timeout()","text":"

Disconnect the barmen who are inactive in the given counters.

Returns:

Type Description int

The number of affected rows (ie, the number of timeouted permanences)

Source code in counter/models.py
def handle_timeout(self) -> int:\n    \"\"\"Disconnect the barmen who are inactive in the given counters.\n\n    Returns:\n        The number of affected rows (ie, the number of timeouted permanences)\n    \"\"\"\n    timeout = timezone.now() - timedelta(minutes=settings.SITH_BARMAN_TIMEOUT)\n    return Permanency.objects.filter(\n        counter__in=self, end=None, activity__lt=timeout\n    ).update(end=F(\"activity\"))\n
"},{"location":"reference/counter/models/#counter.models.Counter","title":"Counter","text":"

Bases: Model

"},{"location":"reference/counter/models/#counter.models.Counter.gen_token","title":"gen_token()","text":"

Generate a new token for this counter.

Source code in counter/models.py
def gen_token(self) -> None:\n    \"\"\"Generate a new token for this counter.\"\"\"\n    self.token = \"\".join(\n        random.choice(string.ascii_letters + string.digits) for _ in range(30)\n    )\n    self.save()\n
"},{"location":"reference/counter/models/#counter.models.Counter.barmen_list","title":"barmen_list()","text":"

Returns the barman list as list of User.

Source code in counter/models.py
@cached_property\ndef barmen_list(self) -> list[User]:\n    \"\"\"Returns the barman list as list of User.\"\"\"\n    return [\n        p.user for p in self.permanencies.filter(end=None).select_related(\"user\")\n    ]\n
"},{"location":"reference/counter/models/#counter.models.Counter.get_random_barman","title":"get_random_barman()","text":"

Return a random user being currently a barman.

Source code in counter/models.py
def get_random_barman(self) -> User:\n    \"\"\"Return a random user being currently a barman.\"\"\"\n    return random.choice(self.barmen_list)\n
"},{"location":"reference/counter/models/#counter.models.Counter.update_activity","title":"update_activity()","text":"

Update the barman activity to prevent timeout.

Source code in counter/models.py
def update_activity(self) -> None:\n    \"\"\"Update the barman activity to prevent timeout.\"\"\"\n    self.permanencies.filter(end=None).update(activity=timezone.now())\n
"},{"location":"reference/counter/models/#counter.models.Counter.can_refill","title":"can_refill()","text":"

Show if the counter authorize the refilling with physic money.

Source code in counter/models.py
def can_refill(self) -> bool:\n    \"\"\"Show if the counter authorize the refilling with physic money.\"\"\"\n    if self.type != \"BAR\":\n        return False\n    # at least one of the barmen is in the AE board\n    ae = Club.objects.get(unix_name=SITH_MAIN_CLUB[\"unix_name\"])\n    return any(ae.get_membership_for(barman) for barman in self.barmen_list)\n
"},{"location":"reference/counter/models/#counter.models.Counter.get_top_barmen","title":"get_top_barmen()","text":"

Return a QuerySet querying the office hours stats of all the barmen of all time of this counter, ordered by descending number of hours.

Each element of the QuerySet corresponds to a barman and has the following data Source code in counter/models.py
def get_top_barmen(self) -> QuerySet:\n    \"\"\"Return a QuerySet querying the office hours stats of all the barmen of all time\n    of this counter, ordered by descending number of hours.\n\n    Each element of the QuerySet corresponds to a barman and has the following data :\n        - the full name (first name + last name) of the barman\n        - the nickname of the barman\n        - the promo of the barman\n        - the total number of office hours the barman did attend\n    \"\"\"\n    return (\n        self.permanencies.exclude(end=None)\n        .annotate(\n            name=Concat(F(\"user__first_name\"), Value(\" \"), F(\"user__last_name\"))\n        )\n        .annotate(nickname=F(\"user__nick_name\"))\n        .annotate(promo=F(\"user__promo\"))\n        .values(\"user\", \"name\", \"nickname\", \"promo\")\n        .annotate(perm_sum=Sum(F(\"end\") - F(\"start\")))\n        .exclude(perm_sum=None)\n        .order_by(\"-perm_sum\")\n    )\n
"},{"location":"reference/counter/models/#counter.models.Counter.get_top_customers","title":"get_top_customers(since=None)","text":"

Return a QuerySet querying the money spent by customers of this counter since the specified date, ordered by descending amount of money spent.

Each element of the QuerySet corresponds to a customer and has the following data :

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None Source code in counter/models.py
def get_top_customers(self, since: datetime | date | None = None) -> QuerySet:\n    \"\"\"Return a QuerySet querying the money spent by customers of this counter\n    since the specified date, ordered by descending amount of money spent.\n\n    Each element of the QuerySet corresponds to a customer and has the following data :\n\n    - the full name (first name + last name) of the customer\n    - the nickname of the customer\n    - the amount of money spent by the customer\n\n    Args:\n        since: timestamp from which to perform the calculation\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return (\n        self.sellings.filter(date__gte=since)\n        .annotate(\n            name=Concat(\n                F(\"customer__user__first_name\"),\n                Value(\" \"),\n                F(\"customer__user__last_name\"),\n            )\n        )\n        .annotate(nickname=F(\"customer__user__nick_name\"))\n        .annotate(promo=F(\"customer__user__promo\"))\n        .annotate(user=F(\"customer__user\"))\n        .values(\"user\", \"promo\", \"name\", \"nickname\")\n        .annotate(\n            selling_sum=Sum(\n                F(\"unit_price\") * F(\"quantity\"), output_field=CurrencyField()\n            )\n        )\n        .filter(selling_sum__gt=0)\n        .order_by(\"-selling_sum\")\n    )\n
"},{"location":"reference/counter/models/#counter.models.Counter.get_total_sales","title":"get_total_sales(since=None)","text":"

Compute and return the total turnover of this counter since the given date.

By default, the date is the start of the current semester.

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None

Returns:

Type Description CurrencyField

Total revenue earned at this counter.

Source code in counter/models.py
def get_total_sales(self, since: datetime | date | None = None) -> CurrencyField:\n    \"\"\"Compute and return the total turnover of this counter since the given date.\n\n    By default, the date is the start of the current semester.\n\n    Args:\n        since: timestamp from which to perform the calculation\n\n    Returns:\n        Total revenue earned at this counter.\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return self.sellings.filter(date__gte=since).aggregate(\n        total=Sum(\n            F(\"quantity\") * F(\"unit_price\"),\n            default=0,\n            output_field=CurrencyField(),\n        )\n    )[\"total\"]\n
"},{"location":"reference/counter/models/#counter.models.Counter.customer_is_barman","title":"customer_is_barman(customer)","text":"

Check if this counter is a bar and if the customer is currently logged in. This is useful to compute special prices.

Source code in counter/models.py
def customer_is_barman(self, customer: Customer | User) -> bool:\n    \"\"\"Check if this counter is a `bar` and if the customer is currently logged in.\n    This is useful to compute special prices.\"\"\"\n\n    # Customer and User are two different tables,\n    # but they share the same primary key\n    return self.type == \"BAR\" and any(b.pk == customer.pk for b in self.barmen_list)\n
"},{"location":"reference/counter/models/#counter.models.Counter.get_products_for","title":"get_products_for(customer)","text":"

Get all allowed products for the provided customer on this counter Prices will be annotated

Source code in counter/models.py
def get_products_for(self, customer: Customer) -> list[Product]:\n    \"\"\"\n    Get all allowed products for the provided customer on this counter\n    Prices will be annotated\n    \"\"\"\n\n    products = self.products.select_related(\"product_type\").prefetch_related(\n        \"buying_groups\"\n    )\n\n    # Only include age appropriate products\n    age = customer.user.age\n    if customer.user.is_banned_alcohol:\n        age = min(age, 17)\n    products = products.filter(limit_age__lte=age)\n\n    # Compute special price for customer if he is a barmen on that bar\n    if self.customer_is_barman(customer):\n        products = products.annotate(price=F(\"special_selling_price\"))\n    else:\n        products = products.annotate(price=F(\"selling_price\"))\n\n    return [\n        product\n        for product in products.all()\n        if product.can_be_sold_to(customer.user)\n    ]\n
"},{"location":"reference/counter/models/#counter.models.RefillingQuerySet","title":"RefillingQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/counter/models/#counter.models.RefillingQuerySet.annotate_total","title":"annotate_total()","text":"

Annotate the Queryset with the total amount.

The total is just the sum of the amounts for each row. If no grouping is involved (like in most queries), this is just the same as doing nothing and fetching the amount attribute.

However, it may be useful when there is a group by clause in the query, or when other models are queried and having a common interface is helpful (e.g. Selling.objects.annotate_total() and Refilling.objects.annotate_total() will both have the total field).

Source code in counter/models.py
def annotate_total(self) -> Self:\n    \"\"\"Annotate the Queryset with the total amount.\n\n    The total is just the sum of the amounts for each row.\n    If no grouping is involved (like in most queries),\n    this is just the same as doing nothing and fetching the\n    `amount` attribute.\n\n    However, it may be useful when there is a `group by` clause\n    in the query, or when other models are queried and having\n    a common interface is helpful (e.g. `Selling.objects.annotate_total()`\n    and `Refilling.objects.annotate_total()` will both have the `total` field).\n    \"\"\"\n    return self.annotate(total=Sum(\"amount\"))\n
"},{"location":"reference/counter/models/#counter.models.Refilling","title":"Refilling","text":"

Bases: Model

Handle the refilling.

"},{"location":"reference/counter/models/#counter.models.SellingQuerySet","title":"SellingQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/counter/models/#counter.models.SellingQuerySet.annotate_total","title":"annotate_total()","text":"

Annotate the Queryset with the total amount of the sales.

The total is considered as the sum of (unit_price * quantity).

Source code in counter/models.py
def annotate_total(self) -> Self:\n    \"\"\"Annotate the Queryset with the total amount of the sales.\n\n    The total is considered as the sum of (unit_price * quantity).\n    \"\"\"\n    return self.annotate(total=Sum(F(\"unit_price\") * F(\"quantity\")))\n
"},{"location":"reference/counter/models/#counter.models.Selling","title":"Selling","text":"

Bases: Model

Handle the sellings.

"},{"location":"reference/counter/models/#counter.models.Selling.save","title":"save(*args, allow_negative=False, **kwargs)","text":"

allow_negative : Allow this selling to use more money than available for this user.

Source code in counter/models.py
def save(self, *args, allow_negative=False, **kwargs):\n    \"\"\"allow_negative : Allow this selling to use more money than available for this user.\"\"\"\n    if not self.date:\n        self.date = timezone.now()\n    self.full_clean()\n    if not self.is_validated:\n        self.customer.amount -= self.quantity * self.unit_price\n        self.customer.save(allow_negative=allow_negative, is_selling=True)\n        self.is_validated = True\n    user = self.customer.user\n    if user.was_subscribed:\n        if (\n            self.product\n            and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_ONE_SEMESTER\n        ):\n            sub = Subscription(\n                member=user,\n                subscription_type=\"un-semestre\",\n                payment_method=\"EBOUTIC\",\n                location=\"EBOUTIC\",\n            )\n            sub.subscription_start = Subscription.compute_start()\n            sub.subscription_start = Subscription.compute_start(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ]\n            )\n            sub.subscription_end = Subscription.compute_end(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ],\n                start=sub.subscription_start,\n            )\n            sub.save()\n        elif (\n            self.product\n            and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_TWO_SEMESTERS\n        ):\n            sub = Subscription(\n                member=user,\n                subscription_type=\"deux-semestres\",\n                payment_method=\"EBOUTIC\",\n                location=\"EBOUTIC\",\n            )\n            sub.subscription_start = Subscription.compute_start()\n            sub.subscription_start = Subscription.compute_start(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ]\n            )\n            sub.subscription_end = Subscription.compute_end(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ],\n                start=sub.subscription_start,\n            )\n            sub.save()\n    if user.preferences.notify_on_click:\n        Notification(\n            user=user,\n            url=reverse(\n                \"core:user_account_detail\",\n                kwargs={\n                    \"user_id\": user.id,\n                    \"year\": self.date.year,\n                    \"month\": self.date.month,\n                },\n            ),\n            param=\"%d x %s\" % (self.quantity, self.label),\n            type=\"SELLING\",\n        ).save()\n    super().save(*args, **kwargs)\n    if hasattr(self.product, \"eticket\"):\n        self.send_mail_customer()\n
"},{"location":"reference/counter/models/#counter.models.Permanency","title":"Permanency","text":"

Bases: Model

A permanency of a barman, on a counter.

This aims at storing a traceability of who was barman where and when. Mainly for dick size contest establishing the top 10 barmen of the semester.

"},{"location":"reference/counter/models/#counter.models.CashRegisterSummary","title":"CashRegisterSummary","text":"

Bases: Model

"},{"location":"reference/counter/models/#counter.models.CashRegisterSummary.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in counter/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(pk=settings.SITH_GROUP_COUNTER_ADMIN_ID)\n
"},{"location":"reference/counter/models/#counter.models.CashRegisterSummaryItem","title":"CashRegisterSummaryItem","text":"

Bases: Model

"},{"location":"reference/counter/models/#counter.models.Eticket","title":"Eticket","text":"

Bases: Model

Eticket can be linked to a product an allows PDF generation.

"},{"location":"reference/counter/models/#counter.models.Eticket.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in counter/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(pk=settings.SITH_GROUP_COUNTER_ADMIN_ID)\n
"},{"location":"reference/counter/models/#counter.models.StudentCard","title":"StudentCard","text":"

Bases: Model

Alternative way to connect a customer into a counter.

We are using Mifare DESFire EV1 specs since it's used for izly cards https://www.nxp.com/docs/en/application-note/AN10927.pdf UID is 7 byte long that means 14 hexa characters.

"},{"location":"reference/counter/models/#counter.models.get_start_of_semester","title":"get_start_of_semester(today=None)","text":"

Return the date of the start of the semester of the given date. If no date is given, return the start date of the current semester.

The current semester is computed as follows:

Parameters:

Name Type Description Default today date | None

the date to use to compute the semester. If None, use today's date.

None

Returns:

Type Description date

the date of the start of the semester

Source code in core/utils.py
def get_start_of_semester(today: date | None = None) -> date:\n    \"\"\"Return the date of the start of the semester of the given date.\n    If no date is given, return the start date of the current semester.\n\n    The current semester is computed as follows:\n\n    - If the date is between 15/08 and 31/12  => Autumn semester.\n    - If the date is between 01/01 and 15/02  => Autumn semester of the previous year.\n    - If the date is between 15/02 and 15/08  => Spring semester\n\n    Args:\n        today: the date to use to compute the semester. If None, use today's date.\n\n    Returns:\n        the date of the start of the semester\n    \"\"\"\n    if today is None:\n        today = localdate()\n\n    autumn = date(today.year, *settings.SITH_SEMESTER_START_AUTUMN)\n    spring = date(today.year, *settings.SITH_SEMESTER_START_SPRING)\n\n    if today >= autumn:  # between 15/08 (included) and 31/12 -> autumn semester\n        return autumn\n    if today >= spring:  # between 15/02 (included) and 15/08 -> spring semester\n        return spring\n    # between 01/01 and 15/02 -> autumn semester of the previous year\n    return autumn.replace(year=autumn.year - 1)\n
"},{"location":"reference/counter/schemas/","title":"Schemas","text":""},{"location":"reference/counter/schemas/#counter.schemas.ClubSchema","title":"ClubSchema","text":"

Bases: ModelSchema

"},{"location":"reference/counter/schemas/#counter.schemas.GroupSchema","title":"GroupSchema","text":"

Bases: ModelSchema

"},{"location":"reference/counter/schemas/#counter.schemas.SimpleUserSchema","title":"SimpleUserSchema","text":"

Bases: ModelSchema

A schema with the minimum amount of information to represent a user.

"},{"location":"reference/counter/schemas/#counter.schemas.Counter","title":"Counter","text":"

Bases: Model

"},{"location":"reference/counter/schemas/#counter.schemas.Counter.gen_token","title":"gen_token()","text":"

Generate a new token for this counter.

Source code in counter/models.py
def gen_token(self) -> None:\n    \"\"\"Generate a new token for this counter.\"\"\"\n    self.token = \"\".join(\n        random.choice(string.ascii_letters + string.digits) for _ in range(30)\n    )\n    self.save()\n
"},{"location":"reference/counter/schemas/#counter.schemas.Counter.barmen_list","title":"barmen_list()","text":"

Returns the barman list as list of User.

Source code in counter/models.py
@cached_property\ndef barmen_list(self) -> list[User]:\n    \"\"\"Returns the barman list as list of User.\"\"\"\n    return [\n        p.user for p in self.permanencies.filter(end=None).select_related(\"user\")\n    ]\n
"},{"location":"reference/counter/schemas/#counter.schemas.Counter.get_random_barman","title":"get_random_barman()","text":"

Return a random user being currently a barman.

Source code in counter/models.py
def get_random_barman(self) -> User:\n    \"\"\"Return a random user being currently a barman.\"\"\"\n    return random.choice(self.barmen_list)\n
"},{"location":"reference/counter/schemas/#counter.schemas.Counter.update_activity","title":"update_activity()","text":"

Update the barman activity to prevent timeout.

Source code in counter/models.py
def update_activity(self) -> None:\n    \"\"\"Update the barman activity to prevent timeout.\"\"\"\n    self.permanencies.filter(end=None).update(activity=timezone.now())\n
"},{"location":"reference/counter/schemas/#counter.schemas.Counter.can_refill","title":"can_refill()","text":"

Show if the counter authorize the refilling with physic money.

Source code in counter/models.py
def can_refill(self) -> bool:\n    \"\"\"Show if the counter authorize the refilling with physic money.\"\"\"\n    if self.type != \"BAR\":\n        return False\n    # at least one of the barmen is in the AE board\n    ae = Club.objects.get(unix_name=SITH_MAIN_CLUB[\"unix_name\"])\n    return any(ae.get_membership_for(barman) for barman in self.barmen_list)\n
"},{"location":"reference/counter/schemas/#counter.schemas.Counter.get_top_barmen","title":"get_top_barmen()","text":"

Return a QuerySet querying the office hours stats of all the barmen of all time of this counter, ordered by descending number of hours.

Each element of the QuerySet corresponds to a barman and has the following data Source code in counter/models.py
def get_top_barmen(self) -> QuerySet:\n    \"\"\"Return a QuerySet querying the office hours stats of all the barmen of all time\n    of this counter, ordered by descending number of hours.\n\n    Each element of the QuerySet corresponds to a barman and has the following data :\n        - the full name (first name + last name) of the barman\n        - the nickname of the barman\n        - the promo of the barman\n        - the total number of office hours the barman did attend\n    \"\"\"\n    return (\n        self.permanencies.exclude(end=None)\n        .annotate(\n            name=Concat(F(\"user__first_name\"), Value(\" \"), F(\"user__last_name\"))\n        )\n        .annotate(nickname=F(\"user__nick_name\"))\n        .annotate(promo=F(\"user__promo\"))\n        .values(\"user\", \"name\", \"nickname\", \"promo\")\n        .annotate(perm_sum=Sum(F(\"end\") - F(\"start\")))\n        .exclude(perm_sum=None)\n        .order_by(\"-perm_sum\")\n    )\n
"},{"location":"reference/counter/schemas/#counter.schemas.Counter.get_top_customers","title":"get_top_customers(since=None)","text":"

Return a QuerySet querying the money spent by customers of this counter since the specified date, ordered by descending amount of money spent.

Each element of the QuerySet corresponds to a customer and has the following data :

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None Source code in counter/models.py
def get_top_customers(self, since: datetime | date | None = None) -> QuerySet:\n    \"\"\"Return a QuerySet querying the money spent by customers of this counter\n    since the specified date, ordered by descending amount of money spent.\n\n    Each element of the QuerySet corresponds to a customer and has the following data :\n\n    - the full name (first name + last name) of the customer\n    - the nickname of the customer\n    - the amount of money spent by the customer\n\n    Args:\n        since: timestamp from which to perform the calculation\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return (\n        self.sellings.filter(date__gte=since)\n        .annotate(\n            name=Concat(\n                F(\"customer__user__first_name\"),\n                Value(\" \"),\n                F(\"customer__user__last_name\"),\n            )\n        )\n        .annotate(nickname=F(\"customer__user__nick_name\"))\n        .annotate(promo=F(\"customer__user__promo\"))\n        .annotate(user=F(\"customer__user\"))\n        .values(\"user\", \"promo\", \"name\", \"nickname\")\n        .annotate(\n            selling_sum=Sum(\n                F(\"unit_price\") * F(\"quantity\"), output_field=CurrencyField()\n            )\n        )\n        .filter(selling_sum__gt=0)\n        .order_by(\"-selling_sum\")\n    )\n
"},{"location":"reference/counter/schemas/#counter.schemas.Counter.get_total_sales","title":"get_total_sales(since=None)","text":"

Compute and return the total turnover of this counter since the given date.

By default, the date is the start of the current semester.

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None

Returns:

Type Description CurrencyField

Total revenue earned at this counter.

Source code in counter/models.py
def get_total_sales(self, since: datetime | date | None = None) -> CurrencyField:\n    \"\"\"Compute and return the total turnover of this counter since the given date.\n\n    By default, the date is the start of the current semester.\n\n    Args:\n        since: timestamp from which to perform the calculation\n\n    Returns:\n        Total revenue earned at this counter.\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return self.sellings.filter(date__gte=since).aggregate(\n        total=Sum(\n            F(\"quantity\") * F(\"unit_price\"),\n            default=0,\n            output_field=CurrencyField(),\n        )\n    )[\"total\"]\n
"},{"location":"reference/counter/schemas/#counter.schemas.Counter.customer_is_barman","title":"customer_is_barman(customer)","text":"

Check if this counter is a bar and if the customer is currently logged in. This is useful to compute special prices.

Source code in counter/models.py
def customer_is_barman(self, customer: Customer | User) -> bool:\n    \"\"\"Check if this counter is a `bar` and if the customer is currently logged in.\n    This is useful to compute special prices.\"\"\"\n\n    # Customer and User are two different tables,\n    # but they share the same primary key\n    return self.type == \"BAR\" and any(b.pk == customer.pk for b in self.barmen_list)\n
"},{"location":"reference/counter/schemas/#counter.schemas.Counter.get_products_for","title":"get_products_for(customer)","text":"

Get all allowed products for the provided customer on this counter Prices will be annotated

Source code in counter/models.py
def get_products_for(self, customer: Customer) -> list[Product]:\n    \"\"\"\n    Get all allowed products for the provided customer on this counter\n    Prices will be annotated\n    \"\"\"\n\n    products = self.products.select_related(\"product_type\").prefetch_related(\n        \"buying_groups\"\n    )\n\n    # Only include age appropriate products\n    age = customer.user.age\n    if customer.user.is_banned_alcohol:\n        age = min(age, 17)\n    products = products.filter(limit_age__lte=age)\n\n    # Compute special price for customer if he is a barmen on that bar\n    if self.customer_is_barman(customer):\n        products = products.annotate(price=F(\"special_selling_price\"))\n    else:\n        products = products.annotate(price=F(\"selling_price\"))\n\n    return [\n        product\n        for product in products.all()\n        if product.can_be_sold_to(customer.user)\n    ]\n
"},{"location":"reference/counter/schemas/#counter.schemas.Product","title":"Product","text":"

Bases: Model

A product, with all its related information.

"},{"location":"reference/counter/schemas/#counter.schemas.Product.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in counter/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(\n        pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID\n    ) or user.is_in_group(pk=settings.SITH_GROUP_COUNTER_ADMIN_ID)\n
"},{"location":"reference/counter/schemas/#counter.schemas.Product.can_be_sold_to","title":"can_be_sold_to(user)","text":"

Check if whether the user given in parameter has the right to buy this product or not.

This must be not confused with the Customer.can_buy() method as the present method returns an information about the relation between a User and a Product, whereas the other tells something about a Customer (and not a user, they are not the same model).

Returns:

Type Description bool

True if the user can buy this product else False

Warning

This performs a db query, thus you can quickly have a N+1 queries problem if you call it in a loop. Hopefully, you can avoid that if you prefetch the buying_groups :

user = User.objects.get(username=\"foobar\")\nproducts = [\n    p\n    for p in Product.objects.prefetch_related(\"buying_groups\")\n    if p.can_be_sold_to(user)\n]\n
Source code in counter/models.py
def can_be_sold_to(self, user: User) -> bool:\n    \"\"\"Check if whether the user given in parameter has the right to buy\n    this product or not.\n\n    This must be not confused with the Customer.can_buy()\n    method as the present method returns an information\n    about the relation between a User and a Product,\n    whereas the other tells something about a Customer\n    (and not a user, they are not the same model).\n\n    Returns:\n        True if the user can buy this product else False\n\n    Warning:\n        This performs a db query, thus you can quickly have\n        a N+1 queries problem if you call it in a loop.\n        Hopefully, you can avoid that if you prefetch the buying_groups :\n\n        ```python\n        user = User.objects.get(username=\"foobar\")\n        products = [\n            p\n            for p in Product.objects.prefetch_related(\"buying_groups\")\n            if p.can_be_sold_to(user)\n        ]\n        ```\n    \"\"\"\n    buying_groups = list(self.buying_groups.all())\n    if not buying_groups:\n        return True\n    return any(user.is_in_group(pk=group.id) for group in buying_groups)\n
"},{"location":"reference/counter/schemas/#counter.schemas.ProductType","title":"ProductType","text":"

Bases: OrderedModel

A product type.

Useful only for categorizing.

"},{"location":"reference/counter/schemas/#counter.schemas.ProductType.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in counter/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID)\n
"},{"location":"reference/counter/schemas/#counter.schemas.CounterSchema","title":"CounterSchema","text":"

Bases: ModelSchema

"},{"location":"reference/counter/schemas/#counter.schemas.CounterFilterSchema","title":"CounterFilterSchema","text":"

Bases: FilterSchema

"},{"location":"reference/counter/schemas/#counter.schemas.SimplifiedCounterSchema","title":"SimplifiedCounterSchema","text":"

Bases: ModelSchema

"},{"location":"reference/counter/schemas/#counter.schemas.ProductTypeSchema","title":"ProductTypeSchema","text":"

Bases: ModelSchema

"},{"location":"reference/counter/schemas/#counter.schemas.SimpleProductTypeSchema","title":"SimpleProductTypeSchema","text":"

Bases: ModelSchema

"},{"location":"reference/counter/schemas/#counter.schemas.ReorderProductTypeSchema","title":"ReorderProductTypeSchema","text":"

Bases: Schema

"},{"location":"reference/counter/schemas/#counter.schemas.SimpleProductSchema","title":"SimpleProductSchema","text":"

Bases: ModelSchema

"},{"location":"reference/counter/schemas/#counter.schemas.ProductSchema","title":"ProductSchema","text":"

Bases: ModelSchema

"},{"location":"reference/counter/schemas/#counter.schemas.ProductFilterSchema","title":"ProductFilterSchema","text":"

Bases: FilterSchema

"},{"location":"reference/counter/views/","title":"Views","text":""},{"location":"reference/eboutic/models/","title":"Models","text":""},{"location":"reference/eboutic/models/#eboutic.models.CurrencyField","title":"CurrencyField(*args, **kwargs)","text":"

Bases: DecimalField

Custom database field used for currency.

Source code in accounting/models.py
def __init__(self, *args, **kwargs):\n    kwargs[\"max_digits\"] = 12\n    kwargs[\"decimal_places\"] = 2\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/eboutic/models/#eboutic.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/eboutic/models/#eboutic.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/eboutic/models/#eboutic.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/eboutic/models/#eboutic.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/eboutic/models/#eboutic.models.BillingInfo","title":"BillingInfo","text":"

Bases: Model

Represent the billing information of a user, which are required by the 3D-Secure v2 system used by the etransaction module.

"},{"location":"reference/eboutic/models/#eboutic.models.BillingInfo.to_3dsv2_xml","title":"to_3dsv2_xml()","text":"

Convert the data from this model into a xml usable by the online paying service of the Cr\u00e9dit Agricole bank. see : https://www.ca-moncommerce.com/espace-client-mon-commerce/up2pay-e-transactions/ma-documentation/manuel-dintegration-focus-3ds-v2/principes-generaux/#integration-3dsv2-developpeur-webmaster.

Source code in counter/models.py
def to_3dsv2_xml(self) -> str:\n    \"\"\"Convert the data from this model into a xml usable\n    by the online paying service of the Cr\u00e9dit Agricole bank.\n    see : `https://www.ca-moncommerce.com/espace-client-mon-commerce/up2pay-e-transactions/ma-documentation/manuel-dintegration-focus-3ds-v2/principes-generaux/#integration-3dsv2-developpeur-webmaster`.\n    \"\"\"\n    data = {\n        \"Address\": {\n            \"FirstName\": self.first_name,\n            \"LastName\": self.last_name,\n            \"Address1\": self.address_1,\n            \"ZipCode\": self.zip_code,\n            \"City\": self.city,\n            \"CountryCode\": self.country.numeric,  # ISO-3166-1 numeric code\n            \"MobilePhone\": self.phone_number.as_national.replace(\" \", \"\"),\n            \"CountryCodeMobilePhone\": f\"+{self.phone_number.country_code}\",\n        }\n    }\n    if self.address_2:\n        data[\"Address\"][\"Address2\"] = self.address_2\n    xml = dict2xml(data, wrap=\"Billing\", newlines=False)\n    return '<?xml version=\"1.0\" encoding=\"UTF-8\" ?>' + xml\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter","title":"Counter","text":"

Bases: Model

"},{"location":"reference/eboutic/models/#eboutic.models.Counter.gen_token","title":"gen_token()","text":"

Generate a new token for this counter.

Source code in counter/models.py
def gen_token(self) -> None:\n    \"\"\"Generate a new token for this counter.\"\"\"\n    self.token = \"\".join(\n        random.choice(string.ascii_letters + string.digits) for _ in range(30)\n    )\n    self.save()\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter.barmen_list","title":"barmen_list()","text":"

Returns the barman list as list of User.

Source code in counter/models.py
@cached_property\ndef barmen_list(self) -> list[User]:\n    \"\"\"Returns the barman list as list of User.\"\"\"\n    return [\n        p.user for p in self.permanencies.filter(end=None).select_related(\"user\")\n    ]\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter.get_random_barman","title":"get_random_barman()","text":"

Return a random user being currently a barman.

Source code in counter/models.py
def get_random_barman(self) -> User:\n    \"\"\"Return a random user being currently a barman.\"\"\"\n    return random.choice(self.barmen_list)\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter.update_activity","title":"update_activity()","text":"

Update the barman activity to prevent timeout.

Source code in counter/models.py
def update_activity(self) -> None:\n    \"\"\"Update the barman activity to prevent timeout.\"\"\"\n    self.permanencies.filter(end=None).update(activity=timezone.now())\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter.can_refill","title":"can_refill()","text":"

Show if the counter authorize the refilling with physic money.

Source code in counter/models.py
def can_refill(self) -> bool:\n    \"\"\"Show if the counter authorize the refilling with physic money.\"\"\"\n    if self.type != \"BAR\":\n        return False\n    # at least one of the barmen is in the AE board\n    ae = Club.objects.get(unix_name=SITH_MAIN_CLUB[\"unix_name\"])\n    return any(ae.get_membership_for(barman) for barman in self.barmen_list)\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter.get_top_barmen","title":"get_top_barmen()","text":"

Return a QuerySet querying the office hours stats of all the barmen of all time of this counter, ordered by descending number of hours.

Each element of the QuerySet corresponds to a barman and has the following data Source code in counter/models.py
def get_top_barmen(self) -> QuerySet:\n    \"\"\"Return a QuerySet querying the office hours stats of all the barmen of all time\n    of this counter, ordered by descending number of hours.\n\n    Each element of the QuerySet corresponds to a barman and has the following data :\n        - the full name (first name + last name) of the barman\n        - the nickname of the barman\n        - the promo of the barman\n        - the total number of office hours the barman did attend\n    \"\"\"\n    return (\n        self.permanencies.exclude(end=None)\n        .annotate(\n            name=Concat(F(\"user__first_name\"), Value(\" \"), F(\"user__last_name\"))\n        )\n        .annotate(nickname=F(\"user__nick_name\"))\n        .annotate(promo=F(\"user__promo\"))\n        .values(\"user\", \"name\", \"nickname\", \"promo\")\n        .annotate(perm_sum=Sum(F(\"end\") - F(\"start\")))\n        .exclude(perm_sum=None)\n        .order_by(\"-perm_sum\")\n    )\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter.get_top_customers","title":"get_top_customers(since=None)","text":"

Return a QuerySet querying the money spent by customers of this counter since the specified date, ordered by descending amount of money spent.

Each element of the QuerySet corresponds to a customer and has the following data :

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None Source code in counter/models.py
def get_top_customers(self, since: datetime | date | None = None) -> QuerySet:\n    \"\"\"Return a QuerySet querying the money spent by customers of this counter\n    since the specified date, ordered by descending amount of money spent.\n\n    Each element of the QuerySet corresponds to a customer and has the following data :\n\n    - the full name (first name + last name) of the customer\n    - the nickname of the customer\n    - the amount of money spent by the customer\n\n    Args:\n        since: timestamp from which to perform the calculation\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return (\n        self.sellings.filter(date__gte=since)\n        .annotate(\n            name=Concat(\n                F(\"customer__user__first_name\"),\n                Value(\" \"),\n                F(\"customer__user__last_name\"),\n            )\n        )\n        .annotate(nickname=F(\"customer__user__nick_name\"))\n        .annotate(promo=F(\"customer__user__promo\"))\n        .annotate(user=F(\"customer__user\"))\n        .values(\"user\", \"promo\", \"name\", \"nickname\")\n        .annotate(\n            selling_sum=Sum(\n                F(\"unit_price\") * F(\"quantity\"), output_field=CurrencyField()\n            )\n        )\n        .filter(selling_sum__gt=0)\n        .order_by(\"-selling_sum\")\n    )\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter.get_total_sales","title":"get_total_sales(since=None)","text":"

Compute and return the total turnover of this counter since the given date.

By default, the date is the start of the current semester.

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None

Returns:

Type Description CurrencyField

Total revenue earned at this counter.

Source code in counter/models.py
def get_total_sales(self, since: datetime | date | None = None) -> CurrencyField:\n    \"\"\"Compute and return the total turnover of this counter since the given date.\n\n    By default, the date is the start of the current semester.\n\n    Args:\n        since: timestamp from which to perform the calculation\n\n    Returns:\n        Total revenue earned at this counter.\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return self.sellings.filter(date__gte=since).aggregate(\n        total=Sum(\n            F(\"quantity\") * F(\"unit_price\"),\n            default=0,\n            output_field=CurrencyField(),\n        )\n    )[\"total\"]\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter.customer_is_barman","title":"customer_is_barman(customer)","text":"

Check if this counter is a bar and if the customer is currently logged in. This is useful to compute special prices.

Source code in counter/models.py
def customer_is_barman(self, customer: Customer | User) -> bool:\n    \"\"\"Check if this counter is a `bar` and if the customer is currently logged in.\n    This is useful to compute special prices.\"\"\"\n\n    # Customer and User are two different tables,\n    # but they share the same primary key\n    return self.type == \"BAR\" and any(b.pk == customer.pk for b in self.barmen_list)\n
"},{"location":"reference/eboutic/models/#eboutic.models.Counter.get_products_for","title":"get_products_for(customer)","text":"

Get all allowed products for the provided customer on this counter Prices will be annotated

Source code in counter/models.py
def get_products_for(self, customer: Customer) -> list[Product]:\n    \"\"\"\n    Get all allowed products for the provided customer on this counter\n    Prices will be annotated\n    \"\"\"\n\n    products = self.products.select_related(\"product_type\").prefetch_related(\n        \"buying_groups\"\n    )\n\n    # Only include age appropriate products\n    age = customer.user.age\n    if customer.user.is_banned_alcohol:\n        age = min(age, 17)\n    products = products.filter(limit_age__lte=age)\n\n    # Compute special price for customer if he is a barmen on that bar\n    if self.customer_is_barman(customer):\n        products = products.annotate(price=F(\"special_selling_price\"))\n    else:\n        products = products.annotate(price=F(\"selling_price\"))\n\n    return [\n        product\n        for product in products.all()\n        if product.can_be_sold_to(customer.user)\n    ]\n
"},{"location":"reference/eboutic/models/#eboutic.models.Customer","title":"Customer","text":"

Bases: Model

Customer data of a User.

It adds some basic customers' information, such as the account ID, and is used by other accounting classes as reference to the customer, rather than using User.

"},{"location":"reference/eboutic/models/#eboutic.models.Customer.can_buy","title":"can_buy property","text":"

Check if whether this customer has the right to purchase any item.

This must be not confused with the Product.can_be_sold_to(user) method as the present method returns an information about a customer whereas the other tells something about the relation between a User (not a Customer, don't mix them) and a Product.

"},{"location":"reference/eboutic/models/#eboutic.models.Customer.save","title":"save(*args, allow_negative=False, is_selling=False, **kwargs)","text":"

is_selling : tell if the current action is a selling allow_negative : ignored if not a selling. Allow a selling to put the account in negative Those two parameters avoid blocking the save method of a customer if his account is negative.

Source code in counter/models.py
def save(self, *args, allow_negative=False, is_selling=False, **kwargs):\n    \"\"\"is_selling : tell if the current action is a selling\n    allow_negative : ignored if not a selling. Allow a selling to put the account in negative\n    Those two parameters avoid blocking the save method of a customer if his account is negative.\n    \"\"\"\n    if self.amount < 0 and (is_selling and not allow_negative):\n        raise ValidationError(_(\"Not enough money\"))\n    super().save(*args, **kwargs)\n
"},{"location":"reference/eboutic/models/#eboutic.models.Customer.get_or_create","title":"get_or_create(user) classmethod","text":"

Work in pretty much the same way as the usual get_or_create method, but with the default field replaced by some under the hood.

If the user has an account, return it as is. Else create a new account with no money on it and a new unique account id

Example : ::

user = User.objects.get(pk=1)\naccount, created = Customer.get_or_create(user)\nif created:\n    print(f\"created a new account with id {account.id}\")\nelse:\n    print(f\"user has already an account, with {account.id} \u20ac on it\"\n
Source code in counter/models.py
@classmethod\ndef get_or_create(cls, user: User) -> tuple[Customer, bool]:\n    \"\"\"Work in pretty much the same way as the usual get_or_create method,\n    but with the default field replaced by some under the hood.\n\n    If the user has an account, return it as is.\n    Else create a new account with no money on it and a new unique account id\n\n    Example : ::\n\n        user = User.objects.get(pk=1)\n        account, created = Customer.get_or_create(user)\n        if created:\n            print(f\"created a new account with id {account.id}\")\n        else:\n            print(f\"user has already an account, with {account.id} \u20ac on it\"\n    \"\"\"\n    if hasattr(user, \"customer\"):\n        return user.customer, False\n\n    # account_id are always a number with a letter appended\n    account_id = (\n        Customer.objects.order_by(Length(\"account_id\"), \"account_id\")\n        .values(\"account_id\")\n        .last()\n    )\n    if account_id is None:\n        # legacy from the old site\n        account = cls.objects.create(user=user, account_id=\"1504a\")\n        return account, True\n\n    account_id = account_id[\"account_id\"]\n    account_num = int(account_id[:-1])\n    while Customer.objects.filter(account_id=account_id).exists():\n        # when entering the first iteration, we are using an already existing account id\n        # so the loop should always execute at least one time\n        account_num += 1\n        account_id = f\"{account_num}{random.choice(string.ascii_lowercase)}\"\n\n    account = cls.objects.create(user=user, account_id=account_id)\n    return account, True\n
"},{"location":"reference/eboutic/models/#eboutic.models.Product","title":"Product","text":"

Bases: Model

A product, with all its related information.

"},{"location":"reference/eboutic/models/#eboutic.models.Product.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in counter/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(\n        pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID\n    ) or user.is_in_group(pk=settings.SITH_GROUP_COUNTER_ADMIN_ID)\n
"},{"location":"reference/eboutic/models/#eboutic.models.Product.can_be_sold_to","title":"can_be_sold_to(user)","text":"

Check if whether the user given in parameter has the right to buy this product or not.

This must be not confused with the Customer.can_buy() method as the present method returns an information about the relation between a User and a Product, whereas the other tells something about a Customer (and not a user, they are not the same model).

Returns:

Type Description bool

True if the user can buy this product else False

Warning

This performs a db query, thus you can quickly have a N+1 queries problem if you call it in a loop. Hopefully, you can avoid that if you prefetch the buying_groups :

user = User.objects.get(username=\"foobar\")\nproducts = [\n    p\n    for p in Product.objects.prefetch_related(\"buying_groups\")\n    if p.can_be_sold_to(user)\n]\n
Source code in counter/models.py
def can_be_sold_to(self, user: User) -> bool:\n    \"\"\"Check if whether the user given in parameter has the right to buy\n    this product or not.\n\n    This must be not confused with the Customer.can_buy()\n    method as the present method returns an information\n    about the relation between a User and a Product,\n    whereas the other tells something about a Customer\n    (and not a user, they are not the same model).\n\n    Returns:\n        True if the user can buy this product else False\n\n    Warning:\n        This performs a db query, thus you can quickly have\n        a N+1 queries problem if you call it in a loop.\n        Hopefully, you can avoid that if you prefetch the buying_groups :\n\n        ```python\n        user = User.objects.get(username=\"foobar\")\n        products = [\n            p\n            for p in Product.objects.prefetch_related(\"buying_groups\")\n            if p.can_be_sold_to(user)\n        ]\n        ```\n    \"\"\"\n    buying_groups = list(self.buying_groups.all())\n    if not buying_groups:\n        return True\n    return any(user.is_in_group(pk=group.id) for group in buying_groups)\n
"},{"location":"reference/eboutic/models/#eboutic.models.Refilling","title":"Refilling","text":"

Bases: Model

Handle the refilling.

"},{"location":"reference/eboutic/models/#eboutic.models.Selling","title":"Selling","text":"

Bases: Model

Handle the sellings.

"},{"location":"reference/eboutic/models/#eboutic.models.Selling.save","title":"save(*args, allow_negative=False, **kwargs)","text":"

allow_negative : Allow this selling to use more money than available for this user.

Source code in counter/models.py
def save(self, *args, allow_negative=False, **kwargs):\n    \"\"\"allow_negative : Allow this selling to use more money than available for this user.\"\"\"\n    if not self.date:\n        self.date = timezone.now()\n    self.full_clean()\n    if not self.is_validated:\n        self.customer.amount -= self.quantity * self.unit_price\n        self.customer.save(allow_negative=allow_negative, is_selling=True)\n        self.is_validated = True\n    user = self.customer.user\n    if user.was_subscribed:\n        if (\n            self.product\n            and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_ONE_SEMESTER\n        ):\n            sub = Subscription(\n                member=user,\n                subscription_type=\"un-semestre\",\n                payment_method=\"EBOUTIC\",\n                location=\"EBOUTIC\",\n            )\n            sub.subscription_start = Subscription.compute_start()\n            sub.subscription_start = Subscription.compute_start(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ]\n            )\n            sub.subscription_end = Subscription.compute_end(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ],\n                start=sub.subscription_start,\n            )\n            sub.save()\n        elif (\n            self.product\n            and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_TWO_SEMESTERS\n        ):\n            sub = Subscription(\n                member=user,\n                subscription_type=\"deux-semestres\",\n                payment_method=\"EBOUTIC\",\n                location=\"EBOUTIC\",\n            )\n            sub.subscription_start = Subscription.compute_start()\n            sub.subscription_start = Subscription.compute_start(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ]\n            )\n            sub.subscription_end = Subscription.compute_end(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ],\n                start=sub.subscription_start,\n            )\n            sub.save()\n    if user.preferences.notify_on_click:\n        Notification(\n            user=user,\n            url=reverse(\n                \"core:user_account_detail\",\n                kwargs={\n                    \"user_id\": user.id,\n                    \"year\": self.date.year,\n                    \"month\": self.date.month,\n                },\n            ),\n            param=\"%d x %s\" % (self.quantity, self.label),\n            type=\"SELLING\",\n        ).save()\n    super().save(*args, **kwargs)\n    if hasattr(self.product, \"eticket\"):\n        self.send_mail_customer()\n
"},{"location":"reference/eboutic/models/#eboutic.models.Basket","title":"Basket","text":"

Bases: Model

Basket is built when the user connects to an eboutic page.

"},{"location":"reference/eboutic/models/#eboutic.models.Basket.from_session","title":"from_session(session) classmethod","text":"

The basket stored in the session object, if it exists.

Source code in eboutic/models.py
@classmethod\ndef from_session(cls, session) -> Basket | None:\n    \"\"\"The basket stored in the session object, if it exists.\"\"\"\n    if \"basket_id\" in session:\n        return cls.objects.filter(id=session[\"basket_id\"]).first()\n    return None\n
"},{"location":"reference/eboutic/models/#eboutic.models.Basket.generate_sales","title":"generate_sales(counter, seller, payment_method)","text":"

Generate a list of sold items corresponding to the items of this basket WITHOUT saving them NOR deleting the basket.

Example
counter = Counter.objects.get(name=\"Eboutic\")\nsales = basket.generate_sales(counter, \"SITH_ACCOUNT\")\n# here the basket is in the same state as before the method call\n\nwith transaction.atomic():\n    for sale in sales:\n        sale.save()\n    basket.delete()\n    # all the basket items are deleted by the on_delete=CASCADE relation\n    # thus only the sales remain\n
Source code in eboutic/models.py
def generate_sales(self, counter, seller: User, payment_method: str):\n    \"\"\"Generate a list of sold items corresponding to the items\n    of this basket WITHOUT saving them NOR deleting the basket.\n\n    Example:\n        ```python\n        counter = Counter.objects.get(name=\"Eboutic\")\n        sales = basket.generate_sales(counter, \"SITH_ACCOUNT\")\n        # here the basket is in the same state as before the method call\n\n        with transaction.atomic():\n            for sale in sales:\n                sale.save()\n            basket.delete()\n            # all the basket items are deleted by the on_delete=CASCADE relation\n            # thus only the sales remain\n        ```\n    \"\"\"\n    # I must proceed with two distinct requests instead of\n    # only one with a join because the AbstractBaseItem model has been\n    # poorly designed. If you refactor the model, please refactor this too.\n    items = self.items.order_by(\"product_id\")\n    ids = [item.product_id for item in items]\n    products = Product.objects.filter(id__in=ids).order_by(\"id\")\n    # items and products are sorted in the same order\n    sales = []\n    for item, product in zip(items, products, strict=False):\n        sales.append(\n            Selling(\n                label=product.name,\n                counter=counter,\n                club=product.club,\n                product=product,\n                seller=seller,\n                customer=self.user.customer,\n                unit_price=item.product_unit_price,\n                quantity=item.quantity,\n                payment_method=payment_method,\n            )\n        )\n    return sales\n
"},{"location":"reference/eboutic/models/#eboutic.models.InvoiceQueryset","title":"InvoiceQueryset","text":"

Bases: QuerySet

"},{"location":"reference/eboutic/models/#eboutic.models.InvoiceQueryset.annotate_total","title":"annotate_total()","text":"

Annotate the queryset with the total amount of each invoice.

The total amount is the sum of (product_unit_price * quantity) for all items related to the invoice.

Source code in eboutic/models.py
def annotate_total(self) -> Self:\n    \"\"\"Annotate the queryset with the total amount of each invoice.\n\n    The total amount is the sum of (product_unit_price * quantity)\n    for all items related to the invoice.\n    \"\"\"\n    # aggregates within subqueries require a little bit of black magic,\n    # but hopefully, django gives a comprehensive documentation for that :\n    # https://docs.djangoproject.com/en/stable/ref/models/expressions/#using-aggregates-within-a-subquery-expression\n    return self.annotate(\n        total=Subquery(\n            InvoiceItem.objects.filter(invoice_id=OuterRef(\"pk\"))\n            .values(\"invoice_id\")\n            .annotate(total=Sum(F(\"product_unit_price\") * F(\"quantity\")))\n            .values(\"total\")\n        )\n    )\n
"},{"location":"reference/eboutic/models/#eboutic.models.Invoice","title":"Invoice","text":"

Bases: Model

Invoices are generated once the payment has been validated.

"},{"location":"reference/eboutic/models/#eboutic.models.AbstractBaseItem","title":"AbstractBaseItem","text":"

Bases: Model

"},{"location":"reference/eboutic/models/#eboutic.models.BasketItem","title":"BasketItem","text":"

Bases: AbstractBaseItem

"},{"location":"reference/eboutic/models/#eboutic.models.BasketItem.from_product","title":"from_product(product, quantity, basket) classmethod","text":"

Create a BasketItem with the same characteristics as the product passed in parameters, with the specified quantity.

Warning

the basket field is not filled, so you must set it yourself before saving the model.

Source code in eboutic/models.py
@classmethod\ndef from_product(cls, product: Product, quantity: int, basket: Basket):\n    \"\"\"Create a BasketItem with the same characteristics as the\n    product passed in parameters, with the specified quantity.\n\n    Warning:\n        the basket field is not filled, so you must set\n        it yourself before saving the model.\n    \"\"\"\n    return cls(\n        basket=basket,\n        product_id=product.id,\n        product_name=product.name,\n        type_id=product.product_type_id,\n        quantity=quantity,\n        product_unit_price=product.selling_price,\n    )\n
"},{"location":"reference/eboutic/models/#eboutic.models.InvoiceItem","title":"InvoiceItem","text":"

Bases: AbstractBaseItem

"},{"location":"reference/eboutic/models/#eboutic.models.get_eboutic_products","title":"get_eboutic_products(user)","text":"Source code in eboutic/models.py
def get_eboutic_products(user: User) -> list[Product]:\n    products = (\n        Counter.objects.get(type=\"EBOUTIC\")\n        .products.filter(product_type__isnull=False)\n        .filter(archived=False)\n        .filter(limit_age__lte=user.age)\n        .annotate(order=F(\"product_type__order\"))\n        .annotate(category=F(\"product_type__name\"))\n        .annotate(category_comment=F(\"product_type__comment\"))\n        .prefetch_related(\"buying_groups\")  # <-- used in `Product.can_be_sold_to`\n    )\n    return [p for p in products if p.can_be_sold_to(user)]\n
"},{"location":"reference/eboutic/views/","title":"Views","text":""},{"location":"reference/eboutic/views/#eboutic.views.PurchaseItemList","title":"PurchaseItemList = TypeAdapter(list[PurchaseItemSchema]) module-attribute","text":""},{"location":"reference/eboutic/views/#eboutic.views.BillingInfoForm","title":"BillingInfoForm","text":"

Bases: ModelForm

"},{"location":"reference/eboutic/views/#eboutic.views.Counter","title":"Counter","text":"

Bases: Model

"},{"location":"reference/eboutic/views/#eboutic.views.Counter.gen_token","title":"gen_token()","text":"

Generate a new token for this counter.

Source code in counter/models.py
def gen_token(self) -> None:\n    \"\"\"Generate a new token for this counter.\"\"\"\n    self.token = \"\".join(\n        random.choice(string.ascii_letters + string.digits) for _ in range(30)\n    )\n    self.save()\n
"},{"location":"reference/eboutic/views/#eboutic.views.Counter.barmen_list","title":"barmen_list()","text":"

Returns the barman list as list of User.

Source code in counter/models.py
@cached_property\ndef barmen_list(self) -> list[User]:\n    \"\"\"Returns the barman list as list of User.\"\"\"\n    return [\n        p.user for p in self.permanencies.filter(end=None).select_related(\"user\")\n    ]\n
"},{"location":"reference/eboutic/views/#eboutic.views.Counter.get_random_barman","title":"get_random_barman()","text":"

Return a random user being currently a barman.

Source code in counter/models.py
def get_random_barman(self) -> User:\n    \"\"\"Return a random user being currently a barman.\"\"\"\n    return random.choice(self.barmen_list)\n
"},{"location":"reference/eboutic/views/#eboutic.views.Counter.update_activity","title":"update_activity()","text":"

Update the barman activity to prevent timeout.

Source code in counter/models.py
def update_activity(self) -> None:\n    \"\"\"Update the barman activity to prevent timeout.\"\"\"\n    self.permanencies.filter(end=None).update(activity=timezone.now())\n
"},{"location":"reference/eboutic/views/#eboutic.views.Counter.can_refill","title":"can_refill()","text":"

Show if the counter authorize the refilling with physic money.

Source code in counter/models.py
def can_refill(self) -> bool:\n    \"\"\"Show if the counter authorize the refilling with physic money.\"\"\"\n    if self.type != \"BAR\":\n        return False\n    # at least one of the barmen is in the AE board\n    ae = Club.objects.get(unix_name=SITH_MAIN_CLUB[\"unix_name\"])\n    return any(ae.get_membership_for(barman) for barman in self.barmen_list)\n
"},{"location":"reference/eboutic/views/#eboutic.views.Counter.get_top_barmen","title":"get_top_barmen()","text":"

Return a QuerySet querying the office hours stats of all the barmen of all time of this counter, ordered by descending number of hours.

Each element of the QuerySet corresponds to a barman and has the following data Source code in counter/models.py
def get_top_barmen(self) -> QuerySet:\n    \"\"\"Return a QuerySet querying the office hours stats of all the barmen of all time\n    of this counter, ordered by descending number of hours.\n\n    Each element of the QuerySet corresponds to a barman and has the following data :\n        - the full name (first name + last name) of the barman\n        - the nickname of the barman\n        - the promo of the barman\n        - the total number of office hours the barman did attend\n    \"\"\"\n    return (\n        self.permanencies.exclude(end=None)\n        .annotate(\n            name=Concat(F(\"user__first_name\"), Value(\" \"), F(\"user__last_name\"))\n        )\n        .annotate(nickname=F(\"user__nick_name\"))\n        .annotate(promo=F(\"user__promo\"))\n        .values(\"user\", \"name\", \"nickname\", \"promo\")\n        .annotate(perm_sum=Sum(F(\"end\") - F(\"start\")))\n        .exclude(perm_sum=None)\n        .order_by(\"-perm_sum\")\n    )\n
"},{"location":"reference/eboutic/views/#eboutic.views.Counter.get_top_customers","title":"get_top_customers(since=None)","text":"

Return a QuerySet querying the money spent by customers of this counter since the specified date, ordered by descending amount of money spent.

Each element of the QuerySet corresponds to a customer and has the following data :

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None Source code in counter/models.py
def get_top_customers(self, since: datetime | date | None = None) -> QuerySet:\n    \"\"\"Return a QuerySet querying the money spent by customers of this counter\n    since the specified date, ordered by descending amount of money spent.\n\n    Each element of the QuerySet corresponds to a customer and has the following data :\n\n    - the full name (first name + last name) of the customer\n    - the nickname of the customer\n    - the amount of money spent by the customer\n\n    Args:\n        since: timestamp from which to perform the calculation\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return (\n        self.sellings.filter(date__gte=since)\n        .annotate(\n            name=Concat(\n                F(\"customer__user__first_name\"),\n                Value(\" \"),\n                F(\"customer__user__last_name\"),\n            )\n        )\n        .annotate(nickname=F(\"customer__user__nick_name\"))\n        .annotate(promo=F(\"customer__user__promo\"))\n        .annotate(user=F(\"customer__user\"))\n        .values(\"user\", \"promo\", \"name\", \"nickname\")\n        .annotate(\n            selling_sum=Sum(\n                F(\"unit_price\") * F(\"quantity\"), output_field=CurrencyField()\n            )\n        )\n        .filter(selling_sum__gt=0)\n        .order_by(\"-selling_sum\")\n    )\n
"},{"location":"reference/eboutic/views/#eboutic.views.Counter.get_total_sales","title":"get_total_sales(since=None)","text":"

Compute and return the total turnover of this counter since the given date.

By default, the date is the start of the current semester.

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None

Returns:

Type Description CurrencyField

Total revenue earned at this counter.

Source code in counter/models.py
def get_total_sales(self, since: datetime | date | None = None) -> CurrencyField:\n    \"\"\"Compute and return the total turnover of this counter since the given date.\n\n    By default, the date is the start of the current semester.\n\n    Args:\n        since: timestamp from which to perform the calculation\n\n    Returns:\n        Total revenue earned at this counter.\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return self.sellings.filter(date__gte=since).aggregate(\n        total=Sum(\n            F(\"quantity\") * F(\"unit_price\"),\n            default=0,\n            output_field=CurrencyField(),\n        )\n    )[\"total\"]\n
"},{"location":"reference/eboutic/views/#eboutic.views.Counter.customer_is_barman","title":"customer_is_barman(customer)","text":"

Check if this counter is a bar and if the customer is currently logged in. This is useful to compute special prices.

Source code in counter/models.py
def customer_is_barman(self, customer: Customer | User) -> bool:\n    \"\"\"Check if this counter is a `bar` and if the customer is currently logged in.\n    This is useful to compute special prices.\"\"\"\n\n    # Customer and User are two different tables,\n    # but they share the same primary key\n    return self.type == \"BAR\" and any(b.pk == customer.pk for b in self.barmen_list)\n
"},{"location":"reference/eboutic/views/#eboutic.views.Counter.get_products_for","title":"get_products_for(customer)","text":"

Get all allowed products for the provided customer on this counter Prices will be annotated

Source code in counter/models.py
def get_products_for(self, customer: Customer) -> list[Product]:\n    \"\"\"\n    Get all allowed products for the provided customer on this counter\n    Prices will be annotated\n    \"\"\"\n\n    products = self.products.select_related(\"product_type\").prefetch_related(\n        \"buying_groups\"\n    )\n\n    # Only include age appropriate products\n    age = customer.user.age\n    if customer.user.is_banned_alcohol:\n        age = min(age, 17)\n    products = products.filter(limit_age__lte=age)\n\n    # Compute special price for customer if he is a barmen on that bar\n    if self.customer_is_barman(customer):\n        products = products.annotate(price=F(\"special_selling_price\"))\n    else:\n        products = products.annotate(price=F(\"selling_price\"))\n\n    return [\n        product\n        for product in products.all()\n        if product.can_be_sold_to(customer.user)\n    ]\n
"},{"location":"reference/eboutic/views/#eboutic.views.Customer","title":"Customer","text":"

Bases: Model

Customer data of a User.

It adds some basic customers' information, such as the account ID, and is used by other accounting classes as reference to the customer, rather than using User.

"},{"location":"reference/eboutic/views/#eboutic.views.Customer.can_buy","title":"can_buy property","text":"

Check if whether this customer has the right to purchase any item.

This must be not confused with the Product.can_be_sold_to(user) method as the present method returns an information about a customer whereas the other tells something about the relation between a User (not a Customer, don't mix them) and a Product.

"},{"location":"reference/eboutic/views/#eboutic.views.Customer.save","title":"save(*args, allow_negative=False, is_selling=False, **kwargs)","text":"

is_selling : tell if the current action is a selling allow_negative : ignored if not a selling. Allow a selling to put the account in negative Those two parameters avoid blocking the save method of a customer if his account is negative.

Source code in counter/models.py
def save(self, *args, allow_negative=False, is_selling=False, **kwargs):\n    \"\"\"is_selling : tell if the current action is a selling\n    allow_negative : ignored if not a selling. Allow a selling to put the account in negative\n    Those two parameters avoid blocking the save method of a customer if his account is negative.\n    \"\"\"\n    if self.amount < 0 and (is_selling and not allow_negative):\n        raise ValidationError(_(\"Not enough money\"))\n    super().save(*args, **kwargs)\n
"},{"location":"reference/eboutic/views/#eboutic.views.Customer.get_or_create","title":"get_or_create(user) classmethod","text":"

Work in pretty much the same way as the usual get_or_create method, but with the default field replaced by some under the hood.

If the user has an account, return it as is. Else create a new account with no money on it and a new unique account id

Example : ::

user = User.objects.get(pk=1)\naccount, created = Customer.get_or_create(user)\nif created:\n    print(f\"created a new account with id {account.id}\")\nelse:\n    print(f\"user has already an account, with {account.id} \u20ac on it\"\n
Source code in counter/models.py
@classmethod\ndef get_or_create(cls, user: User) -> tuple[Customer, bool]:\n    \"\"\"Work in pretty much the same way as the usual get_or_create method,\n    but with the default field replaced by some under the hood.\n\n    If the user has an account, return it as is.\n    Else create a new account with no money on it and a new unique account id\n\n    Example : ::\n\n        user = User.objects.get(pk=1)\n        account, created = Customer.get_or_create(user)\n        if created:\n            print(f\"created a new account with id {account.id}\")\n        else:\n            print(f\"user has already an account, with {account.id} \u20ac on it\"\n    \"\"\"\n    if hasattr(user, \"customer\"):\n        return user.customer, False\n\n    # account_id are always a number with a letter appended\n    account_id = (\n        Customer.objects.order_by(Length(\"account_id\"), \"account_id\")\n        .values(\"account_id\")\n        .last()\n    )\n    if account_id is None:\n        # legacy from the old site\n        account = cls.objects.create(user=user, account_id=\"1504a\")\n        return account, True\n\n    account_id = account_id[\"account_id\"]\n    account_num = int(account_id[:-1])\n    while Customer.objects.filter(account_id=account_id).exists():\n        # when entering the first iteration, we are using an already existing account id\n        # so the loop should always execute at least one time\n        account_num += 1\n        account_id = f\"{account_num}{random.choice(string.ascii_lowercase)}\"\n\n    account = cls.objects.create(user=user, account_id=account_id)\n    return account, True\n
"},{"location":"reference/eboutic/views/#eboutic.views.Product","title":"Product","text":"

Bases: Model

A product, with all its related information.

"},{"location":"reference/eboutic/views/#eboutic.views.Product.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in counter/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_in_group(\n        pk=settings.SITH_GROUP_ACCOUNTING_ADMIN_ID\n    ) or user.is_in_group(pk=settings.SITH_GROUP_COUNTER_ADMIN_ID)\n
"},{"location":"reference/eboutic/views/#eboutic.views.Product.can_be_sold_to","title":"can_be_sold_to(user)","text":"

Check if whether the user given in parameter has the right to buy this product or not.

This must be not confused with the Customer.can_buy() method as the present method returns an information about the relation between a User and a Product, whereas the other tells something about a Customer (and not a user, they are not the same model).

Returns:

Type Description bool

True if the user can buy this product else False

Warning

This performs a db query, thus you can quickly have a N+1 queries problem if you call it in a loop. Hopefully, you can avoid that if you prefetch the buying_groups :

user = User.objects.get(username=\"foobar\")\nproducts = [\n    p\n    for p in Product.objects.prefetch_related(\"buying_groups\")\n    if p.can_be_sold_to(user)\n]\n
Source code in counter/models.py
def can_be_sold_to(self, user: User) -> bool:\n    \"\"\"Check if whether the user given in parameter has the right to buy\n    this product or not.\n\n    This must be not confused with the Customer.can_buy()\n    method as the present method returns an information\n    about the relation between a User and a Product,\n    whereas the other tells something about a Customer\n    (and not a user, they are not the same model).\n\n    Returns:\n        True if the user can buy this product else False\n\n    Warning:\n        This performs a db query, thus you can quickly have\n        a N+1 queries problem if you call it in a loop.\n        Hopefully, you can avoid that if you prefetch the buying_groups :\n\n        ```python\n        user = User.objects.get(username=\"foobar\")\n        products = [\n            p\n            for p in Product.objects.prefetch_related(\"buying_groups\")\n            if p.can_be_sold_to(user)\n        ]\n        ```\n    \"\"\"\n    buying_groups = list(self.buying_groups.all())\n    if not buying_groups:\n        return True\n    return any(user.is_in_group(pk=group.id) for group in buying_groups)\n
"},{"location":"reference/eboutic/views/#eboutic.views.BasketForm","title":"BasketForm(request)","text":"

Class intended to perform checks on the request sended to the server when the user submits his basket from /eboutic/.

Because it must check an unknown number of fields, coming from a cookie and needing some databases checks to be performed, inheriting from forms.Form or using formset would have been likely to end in a big ball of wibbly-wobbly hacky stuff. Thus this class is a pure standalone and performs its operations by its own means. However, it still tries to share some similarities with a standard django Form.

Examples:

::

def my_view(request):\n    form = BasketForm(request)\n    form.clean()\n    if form.is_valid():\n        # perform operations\n    else:\n        errors = form.get_error_messages()\n\n        # return the cookie that was in the request, but with all\n        # incorrects elements removed\n        cookie = form.get_cleaned_cookie()\n

You can also use a little shortcut by directly calling form.is_valid() without calling form.clean(). In this case, the latter method shall be implicitly called.

Source code in eboutic/forms.py
def __init__(self, request: HttpRequest):\n    self.user = request.user\n    self.cookies = request.COOKIES\n    self.error_messages = set()\n    self.correct_items = []\n
"},{"location":"reference/eboutic/views/#eboutic.views.BasketForm.clean","title":"clean()","text":"

Perform all the checks, but return nothing. To know if the form is valid, the is_valid() method must be used.

The form shall be considered as valid if it meets all the following conditions Source code in eboutic/forms.py
def clean(self) -> None:\n    \"\"\"Perform all the checks, but return nothing.\n    To know if the form is valid, the `is_valid()` method must be used.\n\n    The form shall be considered as valid if it meets all the following conditions :\n        - it contains a \"basket_items\" key in the cookies of the request given in the constructor\n        - this cookie is a list of objects formatted this way : `[{'id': <int>, 'quantity': <int>,\n         'name': <str>, 'unit_price': <float>}, ...]`. The order of the fields in each object does not matter\n        - all the ids are positive integers\n        - all the ids refer to products available in the EBOUTIC\n        - all the ids refer to products the user is allowed to buy\n        - all the quantities are positive integers\n    \"\"\"\n    try:\n        basket = PurchaseItemList.validate_json(\n            unquote(self.cookies.get(\"basket_items\", \"[]\"))\n        )\n    except ValidationError:\n        self.error_messages.add(_(\"The request was badly formatted.\"))\n        return\n    if len(basket) == 0:\n        self.error_messages.add(_(\"Your basket is empty.\"))\n        return\n    existing_ids = {product.id for product in get_eboutic_products(self.user)}\n    for item in basket:\n        # check a product with this id does exist\n        if item.product_id in existing_ids:\n            self.correct_items.append(item)\n        else:\n            self.error_messages.add(\n                _(\n                    \"%(name)s : this product does not exist or may no longer be available.\"\n                )\n                % {\"name\": item.name}\n            )\n            continue\n
"},{"location":"reference/eboutic/views/#eboutic.views.BasketForm.is_valid","title":"is_valid()","text":"

Return True if the form is correct else False.

If the clean() method has not been called beforehand, call it.

Source code in eboutic/forms.py
def is_valid(self) -> bool:\n    \"\"\"Return True if the form is correct else False.\n\n    If the `clean()` method has not been called beforehand, call it.\n    \"\"\"\n    if not self.error_messages and not self.correct_items:\n        self.clean()\n    return not self.error_messages\n
"},{"location":"reference/eboutic/views/#eboutic.views.Basket","title":"Basket","text":"

Bases: Model

Basket is built when the user connects to an eboutic page.

"},{"location":"reference/eboutic/views/#eboutic.views.Basket.from_session","title":"from_session(session) classmethod","text":"

The basket stored in the session object, if it exists.

Source code in eboutic/models.py
@classmethod\ndef from_session(cls, session) -> Basket | None:\n    \"\"\"The basket stored in the session object, if it exists.\"\"\"\n    if \"basket_id\" in session:\n        return cls.objects.filter(id=session[\"basket_id\"]).first()\n    return None\n
"},{"location":"reference/eboutic/views/#eboutic.views.Basket.generate_sales","title":"generate_sales(counter, seller, payment_method)","text":"

Generate a list of sold items corresponding to the items of this basket WITHOUT saving them NOR deleting the basket.

Example
counter = Counter.objects.get(name=\"Eboutic\")\nsales = basket.generate_sales(counter, \"SITH_ACCOUNT\")\n# here the basket is in the same state as before the method call\n\nwith transaction.atomic():\n    for sale in sales:\n        sale.save()\n    basket.delete()\n    # all the basket items are deleted by the on_delete=CASCADE relation\n    # thus only the sales remain\n
Source code in eboutic/models.py
def generate_sales(self, counter, seller: User, payment_method: str):\n    \"\"\"Generate a list of sold items corresponding to the items\n    of this basket WITHOUT saving them NOR deleting the basket.\n\n    Example:\n        ```python\n        counter = Counter.objects.get(name=\"Eboutic\")\n        sales = basket.generate_sales(counter, \"SITH_ACCOUNT\")\n        # here the basket is in the same state as before the method call\n\n        with transaction.atomic():\n            for sale in sales:\n                sale.save()\n            basket.delete()\n            # all the basket items are deleted by the on_delete=CASCADE relation\n            # thus only the sales remain\n        ```\n    \"\"\"\n    # I must proceed with two distinct requests instead of\n    # only one with a join because the AbstractBaseItem model has been\n    # poorly designed. If you refactor the model, please refactor this too.\n    items = self.items.order_by(\"product_id\")\n    ids = [item.product_id for item in items]\n    products = Product.objects.filter(id__in=ids).order_by(\"id\")\n    # items and products are sorted in the same order\n    sales = []\n    for item, product in zip(items, products, strict=False):\n        sales.append(\n            Selling(\n                label=product.name,\n                counter=counter,\n                club=product.club,\n                product=product,\n                seller=seller,\n                customer=self.user.customer,\n                unit_price=item.product_unit_price,\n                quantity=item.quantity,\n                payment_method=payment_method,\n            )\n        )\n    return sales\n
"},{"location":"reference/eboutic/views/#eboutic.views.BasketItem","title":"BasketItem","text":"

Bases: AbstractBaseItem

"},{"location":"reference/eboutic/views/#eboutic.views.BasketItem.from_product","title":"from_product(product, quantity, basket) classmethod","text":"

Create a BasketItem with the same characteristics as the product passed in parameters, with the specified quantity.

Warning

the basket field is not filled, so you must set it yourself before saving the model.

Source code in eboutic/models.py
@classmethod\ndef from_product(cls, product: Product, quantity: int, basket: Basket):\n    \"\"\"Create a BasketItem with the same characteristics as the\n    product passed in parameters, with the specified quantity.\n\n    Warning:\n        the basket field is not filled, so you must set\n        it yourself before saving the model.\n    \"\"\"\n    return cls(\n        basket=basket,\n        product_id=product.id,\n        product_name=product.name,\n        type_id=product.product_type_id,\n        quantity=quantity,\n        product_unit_price=product.selling_price,\n    )\n
"},{"location":"reference/eboutic/views/#eboutic.views.Invoice","title":"Invoice","text":"

Bases: Model

Invoices are generated once the payment has been validated.

"},{"location":"reference/eboutic/views/#eboutic.views.InvoiceItem","title":"InvoiceItem","text":"

Bases: AbstractBaseItem

"},{"location":"reference/eboutic/views/#eboutic.views.PurchaseItemSchema","title":"PurchaseItemSchema","text":"

Bases: Schema

"},{"location":"reference/eboutic/views/#eboutic.views.BillingInfoState","title":"BillingInfoState","text":"

Bases: Enum

"},{"location":"reference/eboutic/views/#eboutic.views.EbouticCommand","title":"EbouticCommand","text":"

Bases: LoginRequiredMixin, TemplateView

"},{"location":"reference/eboutic/views/#eboutic.views.EtransactionAutoAnswer","title":"EtransactionAutoAnswer","text":"

Bases: View

"},{"location":"reference/eboutic/views/#eboutic.views.get_eboutic_products","title":"get_eboutic_products(user)","text":"Source code in eboutic/models.py
def get_eboutic_products(user: User) -> list[Product]:\n    products = (\n        Counter.objects.get(type=\"EBOUTIC\")\n        .products.filter(product_type__isnull=False)\n        .filter(archived=False)\n        .filter(limit_age__lte=user.age)\n        .annotate(order=F(\"product_type__order\"))\n        .annotate(category=F(\"product_type__name\"))\n        .annotate(category_comment=F(\"product_type__comment\"))\n        .prefetch_related(\"buying_groups\")  # <-- used in `Product.can_be_sold_to`\n    )\n    return [p for p in products if p.can_be_sold_to(user)]\n
"},{"location":"reference/eboutic/views/#eboutic.views.eboutic_main","title":"eboutic_main(request)","text":"

Main view of the eboutic application.

Return an Http response whose content is of type text/html. The latter represents the page from which a user can see the catalogue of products that he can buy and fill his shopping cart.

The purchasable products are those of the eboutic which belong to a category of products of a product category (orphan products are inaccessible).

If the session contains a key-value pair that associates \"errors\" with a list of strings, this pair is removed from the session and its value displayed to the user when the page is rendered.

Source code in eboutic/views.py
@login_required\n@require_GET\ndef eboutic_main(request: HttpRequest) -> HttpResponse:\n    \"\"\"Main view of the eboutic application.\n\n    Return an Http response whose content is of type text/html.\n    The latter represents the page from which a user can see\n    the catalogue of products that he can buy and fill\n    his shopping cart.\n\n    The purchasable products are those of the eboutic which\n    belong to a category of products of a product category\n    (orphan products are inaccessible).\n\n    If the session contains a key-value pair that associates \"errors\"\n    with a list of strings, this pair is removed from the session\n    and its value displayed to the user when the page is rendered.\n    \"\"\"\n    errors = request.session.pop(\"errors\", None)\n    products = get_eboutic_products(request.user)\n    context = {\n        \"errors\": errors,\n        \"products\": products,\n        \"customer_amount\": request.user.account_balance,\n    }\n    return render(request, \"eboutic/eboutic_main.jinja\", context)\n
"},{"location":"reference/eboutic/views/#eboutic.views.payment_result","title":"payment_result(request, result)","text":"Source code in eboutic/views.py
@require_GET\n@login_required\ndef payment_result(request, result: str) -> HttpResponse:\n    context = {\"success\": result == \"success\"}\n    return render(request, \"eboutic/eboutic_payment_result.jinja\", context)\n
"},{"location":"reference/eboutic/views/#eboutic.views.e_transaction_data","title":"e_transaction_data(request)","text":"Source code in eboutic/views.py
@login_required\n@require_GET\ndef e_transaction_data(request):\n    basket = Basket.from_session(request.session)\n    if basket is None:\n        return HttpResponse(status=404, content=json.dumps({\"data\": []}))\n    data = basket.get_e_transaction_data()\n    data = {\"data\": [{\"key\": key, \"value\": val} for key, val in data]}\n    return HttpResponse(status=200, content=json.dumps(data))\n
"},{"location":"reference/eboutic/views/#eboutic.views.pay_with_sith","title":"pay_with_sith(request)","text":"Source code in eboutic/views.py
@login_required\n@require_POST\ndef pay_with_sith(request):\n    basket = Basket.from_session(request.session)\n    refilling = settings.SITH_COUNTER_PRODUCTTYPE_REFILLING\n    if basket is None or basket.items.filter(type_id=refilling).exists():\n        return redirect(\"eboutic:main\")\n    c = Customer.objects.filter(user__id=basket.user_id).first()\n    if c is None:\n        return redirect(\"eboutic:main\")\n    if c.amount < basket.total:\n        res = redirect(\"eboutic:payment_result\", \"failure\")\n        res.delete_cookie(\"basket_items\", \"/eboutic\")\n        return res\n    eboutic = Counter.objects.get(type=\"EBOUTIC\")\n    sales = basket.generate_sales(eboutic, c.user, \"SITH_ACCOUNT\")\n    try:\n        with transaction.atomic():\n            # Selling.save has some important business logic in it.\n            # Do not bulk_create this\n            for sale in sales:\n                sale.save()\n            basket.delete()\n        request.session.pop(\"basket_id\", None)\n        res = redirect(\"eboutic:payment_result\", \"success\")\n    except DatabaseError as e:\n        with sentry_sdk.push_scope() as scope:\n            scope.user = {\"username\": request.user.username}\n            scope.set_extra(\"someVariable\", e.__repr__())\n            sentry_sdk.capture_message(\n                f\"Erreur le {datetime.now()} dans eboutic.pay_with_sith\"\n            )\n        res = redirect(\"eboutic:payment_result\", \"failure\")\n    res.delete_cookie(\"basket_items\", \"/eboutic\")\n    return res\n
"},{"location":"reference/election/models/","title":"Models","text":""},{"location":"reference/election/models/#election.models.Group","title":"Group","text":"

Bases: Group

Wrapper around django.auth.Group

"},{"location":"reference/election/models/#election.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/election/models/#election.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/election/models/#election.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/election/models/#election.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/election/models/#election.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/election/models/#election.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/election/models/#election.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/election/models/#election.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/election/models/#election.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/election/models/#election.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/election/models/#election.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/election/models/#election.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/election/models/#election.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/election/models/#election.models.Election","title":"Election","text":"

Bases: Model

This class allows to create a new election.

"},{"location":"reference/election/models/#election.models.Role","title":"Role","text":"

Bases: OrderedModel

This class allows to create a new role avaliable for a candidature.

"},{"location":"reference/election/models/#election.models.ElectionList","title":"ElectionList","text":"

Bases: Model

To allow per list vote.

"},{"location":"reference/election/models/#election.models.Candidature","title":"Candidature","text":"

Bases: Model

This class is a component of responsability.

"},{"location":"reference/election/models/#election.models.Vote","title":"Vote","text":"

Bases: Model

This class allows to vote for candidates.

"},{"location":"reference/election/views/","title":"Views","text":""},{"location":"reference/election/views/#election.views.CanCreateMixin","title":"CanCreateMixin(*args, **kwargs)","text":"

Bases: View

Protect any child view that would create an object.

Raises:

Type Description PermissionDenied

If the user has not the necessary permission to create the object of the view.

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/election/views/#election.views.CanEditMixin","title":"CanEditMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to edit this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/election/views/#election.views.CanViewMixin","title":"CanViewMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to view this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/election/views/#election.views.Candidature","title":"Candidature","text":"

Bases: Model

This class is a component of responsability.

"},{"location":"reference/election/views/#election.views.Election","title":"Election","text":"

Bases: Model

This class allows to create a new election.

"},{"location":"reference/election/views/#election.views.ElectionList","title":"ElectionList","text":"

Bases: Model

To allow per list vote.

"},{"location":"reference/election/views/#election.views.Role","title":"Role","text":"

Bases: OrderedModel

This class allows to create a new role avaliable for a candidature.

"},{"location":"reference/election/views/#election.views.Vote","title":"Vote","text":"

Bases: Model

This class allows to vote for candidates.

"},{"location":"reference/election/views/#election.views.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/election/views/#election.views.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/election/views/#election.views.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/election/views/#election.views.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/election/views/#election.views.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/election/views/#election.views.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/election/views/#election.views.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/election/views/#election.views.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/election/views/#election.views.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/election/views/#election.views.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/election/views/#election.views.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/election/views/#election.views.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/election/views/#election.views.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/election/views/#election.views.LimitedCheckboxField","title":"LimitedCheckboxField(queryset, max_choice, **kwargs)","text":"

Bases: ModelMultipleChoiceField

A ModelMultipleChoiceField, with a max limit of selectable inputs.

Source code in election/views.py
def __init__(self, queryset, max_choice, **kwargs):\n    self.max_choice = max_choice\n    super().__init__(queryset, **kwargs)\n
"},{"location":"reference/election/views/#election.views.CandidateForm","title":"CandidateForm(*args, **kwargs)","text":"

Bases: ModelForm

Form to candidate.

Source code in election/views.py
def __init__(self, *args, **kwargs):\n    election_id = kwargs.pop(\"election_id\", None)\n    can_edit = kwargs.pop(\"can_edit\", False)\n    super().__init__(*args, **kwargs)\n    if election_id:\n        self.fields[\"role\"].queryset = Role.objects.filter(\n            election__id=election_id\n        ).all()\n        self.fields[\"election_list\"].queryset = ElectionList.objects.filter(\n            election__id=election_id\n        ).all()\n    if not can_edit:\n        self.fields[\"user\"].widget = forms.HiddenInput()\n
"},{"location":"reference/election/views/#election.views.VoteForm","title":"VoteForm(election, user, *args, **kwargs)","text":"

Bases: Form

Source code in election/views.py
def __init__(self, election, user, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n    if not election.has_voted(user):\n        for role in election.roles.all():\n            cand = role.candidatures\n            if role.max_choice > 1:\n                self.fields[role.title] = LimitedCheckboxField(\n                    cand, role.max_choice, required=False\n                )\n            else:\n                self.fields[role.title] = forms.ModelChoiceField(\n                    cand,\n                    required=False,\n                    widget=forms.RadioSelect(),\n                    empty_label=_(\"Blank vote\"),\n                )\n
"},{"location":"reference/election/views/#election.views.RoleForm","title":"RoleForm(*args, **kwargs)","text":"

Bases: ModelForm

Form for creating a role.

Source code in election/views.py
def __init__(self, *args, **kwargs):\n    election_id = kwargs.pop(\"election_id\", None)\n    super().__init__(*args, **kwargs)\n    if election_id:\n        self.fields[\"election\"].queryset = Election.objects.filter(\n            id=election_id\n        ).all()\n
"},{"location":"reference/election/views/#election.views.ElectionListForm","title":"ElectionListForm(*args, **kwargs)","text":"

Bases: ModelForm

Source code in election/views.py
def __init__(self, *args, **kwargs):\n    election_id = kwargs.pop(\"election_id\", None)\n    super().__init__(*args, **kwargs)\n    if election_id:\n        self.fields[\"election\"].queryset = Election.objects.filter(\n            id=election_id\n        ).all()\n
"},{"location":"reference/election/views/#election.views.ElectionForm","title":"ElectionForm","text":"

Bases: ModelForm

"},{"location":"reference/election/views/#election.views.ElectionsListView","title":"ElectionsListView","text":"

Bases: CanViewMixin, ListView

A list of all non archived elections visible.

"},{"location":"reference/election/views/#election.views.ElectionListArchivedView","title":"ElectionListArchivedView","text":"

Bases: CanViewMixin, ListView

A list of all archived elections visible.

"},{"location":"reference/election/views/#election.views.ElectionDetailView","title":"ElectionDetailView","text":"

Bases: CanViewMixin, DetailView

Details an election responsability by responsability.

"},{"location":"reference/election/views/#election.views.ElectionDetailView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add additionnal data to the template.

Source code in election/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add additionnal data to the template.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"election_form\"] = VoteForm(self.object, self.request.user)\n    kwargs[\"election_results\"] = self.object.results\n    return kwargs\n
"},{"location":"reference/election/views/#election.views.VoteFormView","title":"VoteFormView(*args, **kwargs)","text":"

Bases: CanCreateMixin, FormView

Alows users to vote.

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/election/views/#election.views.VoteFormView.form_valid","title":"form_valid(form)","text":"

Verify that the user is part in a vote group.

Source code in election/views.py
def form_valid(self, form):\n    \"\"\"Verify that the user is part in a vote group.\"\"\"\n    data = form.clean()\n    res = super(FormView, self).form_valid(form)\n    for grp_id in self.election.vote_groups.values_list(\"pk\", flat=True):\n        if self.request.user.is_in_group(pk=grp_id):\n            self.vote(data)\n            return res\n    return res\n
"},{"location":"reference/election/views/#election.views.VoteFormView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add additionnal data to the template.

Source code in election/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add additionnal data to the template.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"object\"] = self.election\n    kwargs[\"election\"] = self.election\n    kwargs[\"election_form\"] = self.get_form()\n    return kwargs\n
"},{"location":"reference/election/views/#election.views.CandidatureCreateView","title":"CandidatureCreateView","text":"

Bases: LoginRequiredMixin, CreateView

View dedicated to a cundidature creation.

"},{"location":"reference/election/views/#election.views.CandidatureCreateView.form_valid","title":"form_valid(form)","text":"

Verify that the selected user is in candidate group.

Source code in election/views.py
def form_valid(self, form):\n    \"\"\"Verify that the selected user is in candidate group.\"\"\"\n    obj = form.instance\n    obj.election = self.election\n    if not hasattr(obj, \"user\"):\n        obj.user = self.request.user\n    if (obj.election.can_candidate(obj.user)) and (\n        obj.user == self.request.user or self.can_edit\n    ):\n        return super().form_valid(form)\n    raise PermissionDenied\n
"},{"location":"reference/election/views/#election.views.ElectionCreateView","title":"ElectionCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

"},{"location":"reference/election/views/#election.views.RoleCreateView","title":"RoleCreateView(*args, **kwargs)","text":"

Bases: CanCreateMixin, CreateView

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/election/views/#election.views.RoleCreateView.form_valid","title":"form_valid(form)","text":"

Verify that the user can edit properly.

Source code in election/views.py
def form_valid(self, form):\n    \"\"\"Verify that the user can edit properly.\"\"\"\n    obj: Role = form.instance\n    user: User = self.request.user\n    if obj.election:\n        for grp_id in obj.election.edit_groups.values_list(\"pk\", flat=True):\n            if user.is_in_group(pk=grp_id):\n                return super(CreateView, self).form_valid(form)\n    raise PermissionDenied\n
"},{"location":"reference/election/views/#election.views.ElectionListCreateView","title":"ElectionListCreateView(*args, **kwargs)","text":"

Bases: CanCreateMixin, CreateView

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/election/views/#election.views.ElectionListCreateView.form_valid","title":"form_valid(form)","text":"

Verify that the user can vote on this election.

Source code in election/views.py
def form_valid(self, form):\n    \"\"\"Verify that the user can vote on this election.\"\"\"\n    obj: ElectionList = form.instance\n    user: User = self.request.user\n    if obj.election:\n        for grp_id in obj.election.candidature_groups.values_list(\"pk\", flat=True):\n            if user.is_in_group(pk=grp_id):\n                return super(CreateView, self).form_valid(form)\n        for grp_id in obj.election.edit_groups.values_list(\"pk\", flat=True):\n            if user.is_in_group(pk=grp_id):\n                return super(CreateView, self).form_valid(form)\n    raise PermissionDenied\n
"},{"location":"reference/election/views/#election.views.ElectionUpdateView","title":"ElectionUpdateView","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/election/views/#election.views.CandidatureUpdateView","title":"CandidatureUpdateView","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/election/views/#election.views.RoleUpdateView","title":"RoleUpdateView","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/election/views/#election.views.ElectionDeleteView","title":"ElectionDeleteView","text":"

Bases: DeleteView

"},{"location":"reference/election/views/#election.views.CandidatureDeleteView","title":"CandidatureDeleteView","text":"

Bases: CanEditMixin, DeleteView

"},{"location":"reference/election/views/#election.views.RoleDeleteView","title":"RoleDeleteView","text":"

Bases: CanEditMixin, DeleteView

"},{"location":"reference/election/views/#election.views.ElectionListDeleteView","title":"ElectionListDeleteView","text":"

Bases: CanEditMixin, DeleteView

"},{"location":"reference/forum/models/","title":"Models","text":""},{"location":"reference/forum/models/#forum.models.MESSAGE_META_ACTIONS","title":"MESSAGE_META_ACTIONS = [('EDIT', _('Message edited by')), ('DELETE', _('Message deleted by')), ('UNDELETE', _('Message undeleted by'))] module-attribute","text":""},{"location":"reference/forum/models/#forum.models.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/forum/models/#forum.models.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/forum/models/#forum.models.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/forum/models/#forum.models.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/forum/models/#forum.models.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/forum/models/#forum.models.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/forum/models/#forum.models.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/forum/models/#forum.models.Group","title":"Group","text":"

Bases: Group

Wrapper around django.auth.Group

"},{"location":"reference/forum/models/#forum.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/forum/models/#forum.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/forum/models/#forum.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/forum/models/#forum.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/forum/models/#forum.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/forum/models/#forum.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/forum/models/#forum.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/forum/models/#forum.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/forum/models/#forum.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/forum/models/#forum.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/forum/models/#forum.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/forum/models/#forum.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/forum/models/#forum.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/forum/models/#forum.models.Forum","title":"Forum","text":"

Bases: Model

The Forum class, made as a tree to allow nice tidy organization.

owner_club allows club members to moderate there own topics edit_groups allows to put any group as a forum admin view_groups allows some groups to view a forum

"},{"location":"reference/forum/models/#forum.models.Forum.copy_rights","title":"copy_rights()","text":"

Copy, if possible, the rights of the parent folder.

Source code in forum/models.py
def copy_rights(self):\n    \"\"\"Copy, if possible, the rights of the parent folder.\"\"\"\n    if self.parent is not None:\n        self.owner_club = self.parent.owner_club\n        self.edit_groups.set(self.parent.edit_groups.all())\n        self.view_groups.set(self.parent.view_groups.all())\n        self.save()\n
"},{"location":"reference/forum/models/#forum.models.Forum.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in forum/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in forums\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/forum/models/#forum.models.ForumTopic","title":"ForumTopic","text":"

Bases: Model

"},{"location":"reference/forum/models/#forum.models.ForumMessage","title":"ForumMessage","text":"

Bases: Model

A message in the forum (thx Cpt. Obvious.).

"},{"location":"reference/forum/models/#forum.models.ForumMessageMeta","title":"ForumMessageMeta","text":"

Bases: Model

"},{"location":"reference/forum/models/#forum.models.ForumUserInfo","title":"ForumUserInfo","text":"

Bases: Model

The forum infos of a user.

This currently stores only the last date a user clicked \"Mark all as read\". However, this can be extended with lot of user preferences dedicated to a user, such as the favourite topics, the signature, and so on...

"},{"location":"reference/forum/models/#forum.models.get_default_edit_group","title":"get_default_edit_group()","text":"Source code in forum/models.py
def get_default_edit_group():\n    return [settings.SITH_GROUP_OLD_SUBSCRIBERS_ID]\n
"},{"location":"reference/forum/models/#forum.models.get_default_view_group","title":"get_default_view_group()","text":"Source code in forum/models.py
def get_default_view_group():\n    return [settings.SITH_GROUP_PUBLIC_ID]\n
"},{"location":"reference/forum/views/","title":"Views","text":""},{"location":"reference/forum/views/#forum.views.CanCreateMixin","title":"CanCreateMixin(*args, **kwargs)","text":"

Bases: View

Protect any child view that would create an object.

Raises:

Type Description PermissionDenied

If the user has not the necessary permission to create the object of the view.

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/forum/views/#forum.views.CanEditMixin","title":"CanEditMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to edit this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/forum/views/#forum.views.CanEditPropMixin","title":"CanEditPropMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has owner permissions on the child view object.

In other word, you can make a view with this view as parent, and it will be retricted to the users that are in the object's owner_group or that pass the obj.can_be_viewed_by test.

Raises:

Type Description PermissionDenied

If the user cannot see the object

"},{"location":"reference/forum/views/#forum.views.CanViewMixin","title":"CanViewMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to view this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/forum/views/#forum.views.Forum","title":"Forum","text":"

Bases: Model

The Forum class, made as a tree to allow nice tidy organization.

owner_club allows club members to moderate there own topics edit_groups allows to put any group as a forum admin view_groups allows some groups to view a forum

"},{"location":"reference/forum/views/#forum.views.Forum.copy_rights","title":"copy_rights()","text":"

Copy, if possible, the rights of the parent folder.

Source code in forum/models.py
def copy_rights(self):\n    \"\"\"Copy, if possible, the rights of the parent folder.\"\"\"\n    if self.parent is not None:\n        self.owner_club = self.parent.owner_club\n        self.edit_groups.set(self.parent.edit_groups.all())\n        self.view_groups.set(self.parent.view_groups.all())\n        self.save()\n
"},{"location":"reference/forum/views/#forum.views.Forum.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in forum/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in forums\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/forum/views/#forum.views.ForumMessage","title":"ForumMessage","text":"

Bases: Model

A message in the forum (thx Cpt. Obvious.).

"},{"location":"reference/forum/views/#forum.views.ForumMessageMeta","title":"ForumMessageMeta","text":"

Bases: Model

"},{"location":"reference/forum/views/#forum.views.ForumTopic","title":"ForumTopic","text":"

Bases: Model

"},{"location":"reference/forum/views/#forum.views.ForumSearchView","title":"ForumSearchView","text":"

Bases: ListView

"},{"location":"reference/forum/views/#forum.views.ForumMainView","title":"ForumMainView","text":"

Bases: ListView

"},{"location":"reference/forum/views/#forum.views.ForumMarkAllAsRead","title":"ForumMarkAllAsRead","text":"

Bases: RedirectView

"},{"location":"reference/forum/views/#forum.views.ForumFavoriteTopics","title":"ForumFavoriteTopics","text":"

Bases: ListView

"},{"location":"reference/forum/views/#forum.views.ForumLastUnread","title":"ForumLastUnread","text":"

Bases: ListView

"},{"location":"reference/forum/views/#forum.views.ForumNameField","title":"ForumNameField","text":"

Bases: ModelChoiceField

"},{"location":"reference/forum/views/#forum.views.ForumForm","title":"ForumForm","text":"

Bases: ModelForm

"},{"location":"reference/forum/views/#forum.views.ForumCreateView","title":"ForumCreateView(*args, **kwargs)","text":"

Bases: CanCreateMixin, CreateView

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/forum/views/#forum.views.ForumEditForm","title":"ForumEditForm","text":"

Bases: ForumForm

"},{"location":"reference/forum/views/#forum.views.ForumEditView","title":"ForumEditView","text":"

Bases: CanEditPropMixin, UpdateView

"},{"location":"reference/forum/views/#forum.views.ForumDeleteView","title":"ForumDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

"},{"location":"reference/forum/views/#forum.views.ForumDetailView","title":"ForumDetailView","text":"

Bases: CanViewMixin, DetailView

"},{"location":"reference/forum/views/#forum.views.TopicForm","title":"TopicForm","text":"

Bases: ModelForm

"},{"location":"reference/forum/views/#forum.views.ForumTopicCreateView","title":"ForumTopicCreateView(*args, **kwargs)","text":"

Bases: CanCreateMixin, CreateView

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/forum/views/#forum.views.ForumTopicEditView","title":"ForumTopicEditView","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/forum/views/#forum.views.ForumTopicSubscribeView","title":"ForumTopicSubscribeView","text":"

Bases: LoginRequiredMixin, CanViewMixin, SingleObjectMixin, RedirectView

"},{"location":"reference/forum/views/#forum.views.ForumTopicDetailView","title":"ForumTopicDetailView","text":"

Bases: CanViewMixin, DetailView

"},{"location":"reference/forum/views/#forum.views.ForumMessageView","title":"ForumMessageView","text":"

Bases: SingleObjectMixin, RedirectView

"},{"location":"reference/forum/views/#forum.views.ForumMessageEditView","title":"ForumMessageEditView","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/forum/views/#forum.views.ForumMessageDeleteView","title":"ForumMessageDeleteView","text":"

Bases: SingleObjectMixin, RedirectView

"},{"location":"reference/forum/views/#forum.views.ForumMessageUndeleteView","title":"ForumMessageUndeleteView","text":"

Bases: SingleObjectMixin, RedirectView

"},{"location":"reference/forum/views/#forum.views.ForumMessageCreateView","title":"ForumMessageCreateView(*args, **kwargs)","text":"

Bases: CanCreateMixin, CreateView

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/forum/views/#forum.views.can_view","title":"can_view(obj, user)","text":"

Can the user see the object.

Parameters:

Name Type Description Default obj Any

Object to test for permission

required user User

core.models.User to test permissions against

required

Returns:

Type Description bool

True if user is authorized to see object else False

Example
if not can_view(self.object ,request.user):\n    raise PermissionDenied\n
Source code in core/auth/mixins.py
def can_view(obj: Any, user: User) -> bool:\n    \"\"\"Can the user see the object.\n\n    Args:\n        obj: Object to test for permission\n        user: core.models.User to test permissions against\n\n    Returns:\n        True if user is authorized to see object else False\n\n    Example:\n        ```python\n        if not can_view(self.object ,request.user):\n            raise PermissionDenied\n        ```\n    \"\"\"\n    if obj is None or user.can_view(obj):\n        return True\n    return can_edit(obj, user)\n
"},{"location":"reference/galaxy/models/","title":"Models","text":""},{"location":"reference/galaxy/models/#galaxy.models.current_star","title":"current_star property","text":"

The star of this user in the :class:Galaxy.

Only take into account the most recent active galaxy.

Returns:

Type Description GalaxyStar | None

The star of this user if there is an active Galaxy

GalaxyStar | None

and this user is a citizen of it, else None

"},{"location":"reference/galaxy/models/#galaxy.models.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/galaxy/models/#galaxy.models.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/galaxy/models/#galaxy.models.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/galaxy/models/#galaxy.models.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/galaxy/models/#galaxy.models.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/galaxy/models/#galaxy.models.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/galaxy/models/#galaxy.models.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/galaxy/models/#galaxy.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/galaxy/models/#galaxy.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/galaxy/models/#galaxy.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/galaxy/models/#galaxy.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/galaxy/models/#galaxy.models.GalaxyStar","title":"GalaxyStar","text":"

Bases: Model

Define a star (vertex -> user) in the galaxy graph.

Store a reference to its owner citizen.

Stars are linked to each others through the :class:GalaxyLane model.

Each GalaxyStar has a mass which push it towards the center of the galaxy. This mass is proportional to the number of pictures the owner of the star is tagged on.

"},{"location":"reference/galaxy/models/#galaxy.models.GalaxyLane","title":"GalaxyLane","text":"

Bases: Model

Define a lane (edge -> link between galaxy citizen) in the galaxy map.

Store a reference to both its ends and the distance it covers. Score details between citizen owning the stars is also stored here.

"},{"location":"reference/galaxy/models/#galaxy.models.StarDict","title":"StarDict","text":"

Bases: TypedDict

"},{"location":"reference/galaxy/models/#galaxy.models.GalaxyDict","title":"GalaxyDict","text":"

Bases: TypedDict

"},{"location":"reference/galaxy/models/#galaxy.models.RelationScore","title":"RelationScore","text":"

Bases: NamedTuple

"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy","title":"Galaxy","text":"

Bases: Model

The Galaxy, a graph linking the active users between each others.

The distance between two users is given by a relation score which takes into account a few parameter like the number of pictures they are both tagged on, the time during which they were in the same clubs and whether they are in the same family.

The citizens of the Galaxy are represented by :class:GalaxyStar and their relations by :class:GalaxyLane.

Several galaxies can coexist. In this case, only the most recent active one shall usually be taken into account. This is useful to keep the current galaxy while generating a new one and swapping them only at the very end.

Please take into account that generating the galaxy is a very expensive operation. For this reason, try not to call the :meth:rule method more than once a day in production.

To quickly access to the state of a galaxy, use the :attr:state attribute.

"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy.compute_user_score","title":"compute_user_score(user) classmethod","text":"

Compute an individual score for each citizen.

It will later be used by the graph algorithm to push higher scores towards the center of the galaxy.

Idea: This could be added to the computation:

Source code in galaxy/models.py
@classmethod\ndef compute_user_score(cls, user: User) -> int:\n    \"\"\"Compute an individual score for each citizen.\n\n    It will later be used by the graph algorithm to push\n    higher scores towards the center of the galaxy.\n\n    Idea: This could be added to the computation:\n\n    - Forum posts\n    - Picture count\n    - Counter consumption\n    - Barman time\n    - ...\n    \"\"\"\n    user_score = 1\n    user_score += cls.query_user_score(user)\n\n    # TODO:\n    # Scale that value with some magic number to accommodate to typical data\n    # Really active galaxy citizen after 5 years typically have a score of about XXX\n    # Citizen that were seen regularly without taking much part in organizations typically have a score of about XXX\n    # Citizen that only went to a few events typically score about XXX\n    user_score = int(math.log2(user_score))\n\n    return user_score\n
"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy.query_user_score","title":"query_user_score(user) classmethod","text":"

Get the individual score of the given user in the galaxy.

Source code in galaxy/models.py
@classmethod\ndef query_user_score(cls, user: User) -> int:\n    \"\"\"Get the individual score of the given user in the galaxy.\"\"\"\n    score_query = (\n        User.objects.filter(id=user.id)\n        .annotate(\n            godchildren_count=Count(\"godchildren\", distinct=True)\n            * cls.FAMILY_LINK_POINTS,\n            godfathers_count=Count(\"godfathers\", distinct=True)\n            * cls.FAMILY_LINK_POINTS,\n            pictures_score=Count(\"pictures\", distinct=True) * cls.PICTURE_POINTS,\n            clubs_score=Count(\"memberships\", distinct=True) * cls.CLUBS_POINTS,\n        )\n        .aggregate(\n            score=models.Sum(\n                F(\"godchildren_count\")\n                + F(\"godfathers_count\")\n                + F(\"pictures_score\")\n                + F(\"clubs_score\")\n            )\n        )\n    )\n    return score_query.get(\"score\")\n
"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy.compute_users_score","title":"compute_users_score(user1, user2) classmethod","text":"

Compute the relationship scores of the two given users.

The computation is done with the following fields :

Source code in galaxy/models.py
@classmethod\ndef compute_users_score(cls, user1: User, user2: User) -> RelationScore:\n    \"\"\"Compute the relationship scores of the two given users.\n\n    The computation is done with the following fields :\n\n    - family: if they have some godfather/godchild relation\n    - pictures: in how many pictures are both tagged\n    - clubs: during how many days they were members of the same clubs\n    \"\"\"\n    family = cls.compute_users_family_score(user1, user2)\n    pictures = cls.compute_users_pictures_score(user1, user2)\n    clubs = cls.compute_users_clubs_score(user1, user2)\n    return RelationScore(family=family, pictures=pictures, clubs=clubs)\n
"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy.compute_users_family_score","title":"compute_users_family_score(user1, user2) classmethod","text":"

Compute the family score of the relation between the given users.

This takes into account mutual godfathers.

Returns:

Type Description int

366 if user1 is the godfather of user2 (or vice versa) else 0

Source code in galaxy/models.py
@classmethod\ndef compute_users_family_score(cls, user1: User, user2: User) -> int:\n    \"\"\"Compute the family score of the relation between the given users.\n\n    This takes into account mutual godfathers.\n\n    Returns:\n         366 if user1 is the godfather of user2 (or vice versa) else 0\n    \"\"\"\n    link_count = User.objects.filter(\n        Q(id=user1.id, godfathers=user2) | Q(id=user2.id, godfathers=user1)\n    ).count()\n    if link_count > 0:\n        cls.logger.debug(\n            f\"\\t\\t- '{user1}' and '{user2}' have {link_count} direct family link\"\n        )\n    return link_count * cls.FAMILY_LINK_POINTS\n
"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy.compute_users_pictures_score","title":"compute_users_pictures_score(user1, user2) classmethod","text":"

Compute the pictures score of the relation between the given users.

The pictures score is obtained by counting the number of :class:Picture in which they have been both identified. This score is then multiplied by 2.

Returns:

Type Description int

The number of pictures both users have in common, times 2

Source code in galaxy/models.py
@classmethod\ndef compute_users_pictures_score(cls, user1: User, user2: User) -> int:\n    \"\"\"Compute the pictures score of the relation between the given users.\n\n    The pictures score is obtained by counting the number\n    of :class:`Picture` in which they have been both identified.\n    This score is then multiplied by 2.\n\n    Returns:\n         The number of pictures both users have in common, times 2\n    \"\"\"\n    picture_count = (\n        Picture.objects.filter(people__user__in=(user1,))\n        .filter(people__user__in=(user2,))\n        .count()\n    )\n    if picture_count:\n        cls.logger.debug(\n            f\"\\t\\t- '{user1}' was pictured with '{user2}' {picture_count} times\"\n        )\n    return picture_count * cls.PICTURE_POINTS\n
"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy.compute_users_clubs_score","title":"compute_users_clubs_score(user1, user2) classmethod","text":"

Compute the clubs score of the relation between the given users.

The club score is obtained by counting the number of days during which the memberships (see :class:club.models.Membership) of both users overlapped.

For example, if user1 was a member of Unitec from 01/01/2020 to 31/12/2021 (two years) and user2 was a member of the same club from 01/01/2021 to 31/12/2022 (also two years, but with an offset of one year), then their club score is 365.

Returns:

Type Description int

the number of days during which both users were in the same club

Source code in galaxy/models.py
@classmethod\ndef compute_users_clubs_score(cls, user1: User, user2: User) -> int:\n    \"\"\"Compute the clubs score of the relation between the given users.\n\n    The club score is obtained by counting the number of days\n    during which the memberships (see :class:`club.models.Membership`)\n    of both users overlapped.\n\n    For example, if user1 was a member of Unitec from 01/01/2020 to 31/12/2021\n    (two years) and user2 was a member of the same club from 01/01/2021 to\n    31/12/2022 (also two years, but with an offset of one year), then their\n    club score is 365.\n\n    Returns:\n        the number of days during which both users were in the same club\n    \"\"\"\n    common_clubs = Club.objects.filter(members__in=user1.memberships.all()).filter(\n        members__in=user2.memberships.all()\n    )\n    user1_memberships = user1.memberships.filter(club__in=common_clubs)\n    user2_memberships = user2.memberships.filter(club__in=common_clubs)\n\n    score = 0\n    for user1_membership in user1_memberships:\n        if user1_membership.end_date is None:\n            # user1_membership.save() is not called in this function, hence this is safe\n            user1_membership.end_date = localdate()\n        query = Q(  # start2 <= start1 <= end2\n            start_date__lte=user1_membership.start_date,\n            end_date__gte=user1_membership.start_date,\n        )\n        query |= Q(  # start2 <= start1 <= now\n            start_date__lte=user1_membership.start_date, end_date=None\n        )\n        query |= Q(  # start1 <= start2 <= end2\n            start_date__gte=user1_membership.start_date,\n            start_date__lte=user1_membership.end_date,\n        )\n        for user2_membership in user2_memberships.filter(\n            query, club=user1_membership.club\n        ):\n            if user2_membership.end_date is None:\n                user2_membership.end_date = localdate()\n            latest_start = max(\n                user1_membership.start_date, user2_membership.start_date\n            )\n            earliest_end = min(user1_membership.end_date, user2_membership.end_date)\n            cls.logger.debug(\n                \"\\t\\t- '%s' was with '%s' in %s starting on %s until %s (%s days)\"\n                % (\n                    user1,\n                    user2,\n                    user2_membership.club,\n                    latest_start,\n                    earliest_end,\n                    (earliest_end - latest_start).days,\n                )\n            )\n            score += cls.CLUBS_POINTS * (earliest_end - latest_start).days\n    return score\n
"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy.scale_distance","title":"scale_distance(value) classmethod","text":"

Given a numeric value, return a scaled value which can be used in the Galaxy's graphical interface to set the distance between two stars.

Returns:

Type Description int

the scaled value usable in the Galaxy's 3d graph

Source code in galaxy/models.py
@classmethod\ndef scale_distance(cls, value: int | float) -> int:\n    \"\"\"Given a numeric value, return a scaled value which can\n    be used in the Galaxy's graphical interface to set the distance\n    between two stars.\n\n    Returns:\n        the scaled value usable in the Galaxy's 3d graph\n    \"\"\"\n    # TODO: this will need adjustements with the real, typical data on Taiste\n    if value == 0:\n        return 4000  # Following calculus would give us +\u221e, we cap it to 4000\n\n    cls.logger.debug(f\"\\t\\t> Score: {value}\")\n    # Invert score to draw close users together\n    value = 1 / value  # Cannot be 0\n    value += 2  # We use log2 just below and need to stay above 1\n    value = (  # Let's get something in the range ]0; log2(3)-1\u22480.58[ that we can multiply later\n        math.log2(value) - 1\n    )\n    value *= (  # Scale that value with a magic number to accommodate to typical data\n        # Really close galaxy citizen after 5 years typically have a score of about XXX\n        # Citizen that were in the same year without being really friends typically have a score of about XXX\n        # Citizen that have met once or twice only have a couple of pictures together typically score about XXX\n        cls.GALAXY_SCALE_FACTOR\n    )\n    cls.logger.debug(f\"\\t\\t> Scaled distance: {value}\")\n    return int(value)\n
"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy.rule","title":"rule(picture_count_threshold=10)","text":"

Main function of the Galaxy.

Iterate over all the rulable users to promote them to citizens. A citizen is a user who has a corresponding star in the Galaxy. Also build up the lanes, which are the links between the different citizen.

Users who can be ruled are defined with the picture_count_threshold: all users who are identified in a strictly lower number of pictures won't be promoted to citizens. This does very effectively limit the quantity of computing to do and only includes users who have had a minimum of activity.

This method still remains very expensive, so think thoroughly before you call it, especially in production.

:param picture_count_threshold: the minimum number of picture to have to be included in the galaxy

Source code in galaxy/models.py
def rule(self, picture_count_threshold=10) -> None:\n    \"\"\"Main function of the Galaxy.\n\n    Iterate over all the rulable users to promote them to citizens.\n    A citizen is a user who has a corresponding star in the Galaxy.\n    Also build up the lanes, which are the links between the different citizen.\n\n    Users who can be ruled are defined with the `picture_count_threshold`:\n    all users who are identified in a strictly lower number of pictures\n    won't be promoted to citizens.\n    This does very effectively limit the quantity of computing to do\n    and only includes users who have had a minimum of activity.\n\n    This method still remains very expensive, so think thoroughly before\n    you call it, especially in production.\n\n    :param picture_count_threshold: the minimum number of picture to have to be\n                                    included in the galaxy\n    \"\"\"\n    total_time = time.time()\n    self.logger.info(\"Listing rulable citizen.\")\n    rulable_users = (\n        User.objects.filter(subscriptions__isnull=False)\n        .annotate(pictures_count=Count(\"pictures\"))\n        .filter(pictures_count__gt=picture_count_threshold)\n        .distinct()\n    )\n\n    # force fetch of the whole query to make sure there won't\n    # be any more db hits\n    # this is memory expensive but prevents a lot of db hits, therefore\n    # is far more time efficient\n\n    rulable_users = list(rulable_users)\n    rulable_users_count = len(rulable_users)\n    user1_count = 0\n    self.logger.info(\n        f\"{rulable_users_count} citizen have been listed. Starting to rule.\"\n    )\n\n    stars = []\n    self.logger.info(\"Creating stars for all citizen\")\n    for user in rulable_users:\n        star = GalaxyStar(\n            owner=user, galaxy=self, mass=self.compute_user_score(user)\n        )\n        stars.append(star)\n    GalaxyStar.objects.bulk_create(stars)\n\n    stars = {}\n    for star in GalaxyStar.objects.filter(galaxy=self):\n        stars[star.owner.id] = star\n\n    self.logger.info(\"Creating lanes between stars\")\n    # Display current speed every $speed_count_frequency users\n    speed_count_frequency = max(rulable_users_count // 10, 1)  # ten time at most\n    global_avg_speed_accumulator = 0\n    global_avg_speed_count = 0\n    t_global_start = time.time()\n    while len(rulable_users) > 0:\n        user1 = rulable_users.pop()\n        user1_count += 1\n        rulable_users_count2 = len(rulable_users)\n\n        star1 = stars[user1.id]\n\n        user_avg_speed = 0\n        user_avg_speed_count = 0\n\n        tstart = time.time()\n        lanes = []\n        for user2_count, user2 in enumerate(rulable_users, start=1):\n            self.logger.debug(\"\")\n            self.logger.debug(\n                f\"\\t> Examining '{user1}' ({user1_count}/{rulable_users_count}) with '{user2}' ({user2_count}/{rulable_users_count2})\"\n            )\n\n            star2 = stars[user2.id]\n\n            score = Galaxy.compute_users_score(user1, user2)\n            distance = self.scale_distance(sum(score))\n            if distance < 30:  # TODO: this needs tuning with real-world data\n                lanes.append(\n                    GalaxyLane(\n                        star1=star1,\n                        star2=star2,\n                        distance=distance,\n                        family=score.family,\n                        pictures=score.pictures,\n                        clubs=score.clubs,\n                    )\n                )\n\n            if user2_count % speed_count_frequency == 0:\n                tend = time.time()\n                delta = tend - tstart\n                speed = float(speed_count_frequency) / delta\n                user_avg_speed += speed\n                user_avg_speed_count += 1\n                self.logger.debug(\n                    f\"\\tSpeed: {speed:.2f} users per second (time for last {speed_count_frequency} citizens: {delta:.2f} second)\"\n                )\n                tstart = time.time()\n\n        GalaxyLane.objects.bulk_create(lanes)\n\n        self.logger.info(\"\")\n\n        t_global_end = time.time()\n        global_delta = t_global_end - t_global_start\n        speed = 1.0 / global_delta\n        global_avg_speed_accumulator += speed\n        global_avg_speed_count += 1\n        global_avg_speed = global_avg_speed_accumulator / global_avg_speed_count\n\n        self.logger.info(f\" Ruling of {self} \".center(60, \"#\"))\n        self.logger.info(\n            f\"Progression: {user1_count}/{rulable_users_count} citizen -- {rulable_users_count - user1_count} remaining\"\n        )\n        self.logger.info(f\"Speed: {60.0*global_avg_speed:.2f} citizen per minute\")\n\n        # We can divide the computed ETA by 2 because each loop, there is one citizen less to check, and maths tell\n        # us that this averages to a division by two\n        eta = rulable_users_count2 / global_avg_speed / 2\n        eta_hours = int(eta // 3600)\n        eta_minutes = int(eta // 60 % 60)\n        self.logger.info(\n            f\"ETA: {eta_hours} hours {eta_minutes} minutes ({eta / 3600 / 24:.2f} days)\"\n        )\n        self.logger.info(\"#\" * 60)\n        t_global_start = time.time()\n\n    # Here, we get the IDs of the old galaxies that we'll need to delete. In normal operation, only one galaxy\n    # should be returned, and we can't delete it yet, as it's the one still displayed by the Sith.\n    old_galaxies_pks = list(\n        Galaxy.objects.filter(state__isnull=False).values_list(\"pk\", flat=True)\n    )\n    self.logger.info(\n        f\"These old galaxies will be deleted once the new one is ready: {old_galaxies_pks}\"\n    )\n\n    # Making the state sets this new galaxy as being ready. From now on, the Sith will show us to the world.\n    self.make_state()\n\n    # Avoid accident if there is nothing to delete\n    if len(old_galaxies_pks) > 0:\n        # Former galaxies can now be deleted.\n        Galaxy.objects.filter(pk__in=old_galaxies_pks).delete()\n\n    total_time = time.time() - total_time\n    total_time_hours = int(total_time // 3600)\n    total_time_minutes = int(total_time // 60 % 60)\n    total_time_seconds = int(total_time % 60)\n    self.logger.info(\n        f\"{self} ruled in {total_time:.2f} seconds ({total_time_hours} hours, {total_time_minutes} minutes, {total_time_seconds} seconds)\"\n    )\n
"},{"location":"reference/galaxy/models/#galaxy.models.Galaxy.make_state","title":"make_state()","text":"

Compute JSON structure to send to 3d-force-graph: https://github.com/vasturiano/3d-force-graph/.

Source code in galaxy/models.py
def make_state(self) -> None:\n    \"\"\"Compute JSON structure to send to 3d-force-graph: https://github.com/vasturiano/3d-force-graph/.\"\"\"\n    self.logger.info(\n        \"Caching current Galaxy state for a quicker display of the Empire's power.\"\n    )\n\n    without_nickname = Concat(\n        F(\"owner__first_name\"), Value(\" \"), F(\"owner__last_name\")\n    )\n    with_nickname = Concat(\n        F(\"owner__first_name\"),\n        Value(\" \"),\n        F(\"owner__last_name\"),\n        Value(\" (\"),\n        F(\"owner__nick_name\"),\n        Value(\")\"),\n    )\n    stars = (\n        GalaxyStar.objects.filter(galaxy=self)\n        .order_by(\n            \"owner\"\n        )  # This helps determinism for the tests and doesn't cost much\n        .annotate(\n            owner_name=Case(\n                When(owner__nick_name=None, then=without_nickname),\n                default=with_nickname,\n            )\n        )\n    )\n    lanes = (\n        GalaxyLane.objects.filter(star1__galaxy=self)\n        .order_by(\n            \"star1\"\n        )  # This helps determinism for the tests and doesn't cost much\n        .annotate(\n            star1_owner=F(\"star1__owner__id\"),\n            star2_owner=F(\"star2__owner__id\"),\n        )\n    )\n    json = GalaxyDict(\n        nodes=[\n            StarDict(\n                id=star.owner_id,\n                name=star.owner_name,\n                mass=star.mass,\n            )\n            for star in stars\n        ],\n        links=[],\n    )\n    for path in lanes:\n        json[\"links\"].append(\n            {\n                \"source\": path.star1_owner,\n                \"target\": path.star2_owner,\n                \"value\": path.distance,\n            }\n        )\n    self.state = json\n    self.save()\n    self.logger.info(f\"{self} is now ready!\")\n
"},{"location":"reference/galaxy/views/","title":"Views","text":""},{"location":"reference/galaxy/views/#galaxy.views.CanViewMixin","title":"CanViewMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to view this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/galaxy/views/#galaxy.views.FormerSubscriberMixin","title":"FormerSubscriberMixin","text":"

Bases: AccessMixin

Check if the user was at least an old subscriber.

Raises:

Type Description PermissionDenied

if the user never subscribed.

"},{"location":"reference/galaxy/views/#galaxy.views.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/galaxy/views/#galaxy.views.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/galaxy/views/#galaxy.views.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/galaxy/views/#galaxy.views.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/galaxy/views/#galaxy.views.UserTabsMixin","title":"UserTabsMixin","text":"

Bases: TabedViewMixin

"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy","title":"Galaxy","text":"

Bases: Model

The Galaxy, a graph linking the active users between each others.

The distance between two users is given by a relation score which takes into account a few parameter like the number of pictures they are both tagged on, the time during which they were in the same clubs and whether they are in the same family.

The citizens of the Galaxy are represented by :class:GalaxyStar and their relations by :class:GalaxyLane.

Several galaxies can coexist. In this case, only the most recent active one shall usually be taken into account. This is useful to keep the current galaxy while generating a new one and swapping them only at the very end.

Please take into account that generating the galaxy is a very expensive operation. For this reason, try not to call the :meth:rule method more than once a day in production.

To quickly access to the state of a galaxy, use the :attr:state attribute.

"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy.compute_user_score","title":"compute_user_score(user) classmethod","text":"

Compute an individual score for each citizen.

It will later be used by the graph algorithm to push higher scores towards the center of the galaxy.

Idea: This could be added to the computation:

Source code in galaxy/models.py
@classmethod\ndef compute_user_score(cls, user: User) -> int:\n    \"\"\"Compute an individual score for each citizen.\n\n    It will later be used by the graph algorithm to push\n    higher scores towards the center of the galaxy.\n\n    Idea: This could be added to the computation:\n\n    - Forum posts\n    - Picture count\n    - Counter consumption\n    - Barman time\n    - ...\n    \"\"\"\n    user_score = 1\n    user_score += cls.query_user_score(user)\n\n    # TODO:\n    # Scale that value with some magic number to accommodate to typical data\n    # Really active galaxy citizen after 5 years typically have a score of about XXX\n    # Citizen that were seen regularly without taking much part in organizations typically have a score of about XXX\n    # Citizen that only went to a few events typically score about XXX\n    user_score = int(math.log2(user_score))\n\n    return user_score\n
"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy.query_user_score","title":"query_user_score(user) classmethod","text":"

Get the individual score of the given user in the galaxy.

Source code in galaxy/models.py
@classmethod\ndef query_user_score(cls, user: User) -> int:\n    \"\"\"Get the individual score of the given user in the galaxy.\"\"\"\n    score_query = (\n        User.objects.filter(id=user.id)\n        .annotate(\n            godchildren_count=Count(\"godchildren\", distinct=True)\n            * cls.FAMILY_LINK_POINTS,\n            godfathers_count=Count(\"godfathers\", distinct=True)\n            * cls.FAMILY_LINK_POINTS,\n            pictures_score=Count(\"pictures\", distinct=True) * cls.PICTURE_POINTS,\n            clubs_score=Count(\"memberships\", distinct=True) * cls.CLUBS_POINTS,\n        )\n        .aggregate(\n            score=models.Sum(\n                F(\"godchildren_count\")\n                + F(\"godfathers_count\")\n                + F(\"pictures_score\")\n                + F(\"clubs_score\")\n            )\n        )\n    )\n    return score_query.get(\"score\")\n
"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy.compute_users_score","title":"compute_users_score(user1, user2) classmethod","text":"

Compute the relationship scores of the two given users.

The computation is done with the following fields :

Source code in galaxy/models.py
@classmethod\ndef compute_users_score(cls, user1: User, user2: User) -> RelationScore:\n    \"\"\"Compute the relationship scores of the two given users.\n\n    The computation is done with the following fields :\n\n    - family: if they have some godfather/godchild relation\n    - pictures: in how many pictures are both tagged\n    - clubs: during how many days they were members of the same clubs\n    \"\"\"\n    family = cls.compute_users_family_score(user1, user2)\n    pictures = cls.compute_users_pictures_score(user1, user2)\n    clubs = cls.compute_users_clubs_score(user1, user2)\n    return RelationScore(family=family, pictures=pictures, clubs=clubs)\n
"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy.compute_users_family_score","title":"compute_users_family_score(user1, user2) classmethod","text":"

Compute the family score of the relation between the given users.

This takes into account mutual godfathers.

Returns:

Type Description int

366 if user1 is the godfather of user2 (or vice versa) else 0

Source code in galaxy/models.py
@classmethod\ndef compute_users_family_score(cls, user1: User, user2: User) -> int:\n    \"\"\"Compute the family score of the relation between the given users.\n\n    This takes into account mutual godfathers.\n\n    Returns:\n         366 if user1 is the godfather of user2 (or vice versa) else 0\n    \"\"\"\n    link_count = User.objects.filter(\n        Q(id=user1.id, godfathers=user2) | Q(id=user2.id, godfathers=user1)\n    ).count()\n    if link_count > 0:\n        cls.logger.debug(\n            f\"\\t\\t- '{user1}' and '{user2}' have {link_count} direct family link\"\n        )\n    return link_count * cls.FAMILY_LINK_POINTS\n
"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy.compute_users_pictures_score","title":"compute_users_pictures_score(user1, user2) classmethod","text":"

Compute the pictures score of the relation between the given users.

The pictures score is obtained by counting the number of :class:Picture in which they have been both identified. This score is then multiplied by 2.

Returns:

Type Description int

The number of pictures both users have in common, times 2

Source code in galaxy/models.py
@classmethod\ndef compute_users_pictures_score(cls, user1: User, user2: User) -> int:\n    \"\"\"Compute the pictures score of the relation between the given users.\n\n    The pictures score is obtained by counting the number\n    of :class:`Picture` in which they have been both identified.\n    This score is then multiplied by 2.\n\n    Returns:\n         The number of pictures both users have in common, times 2\n    \"\"\"\n    picture_count = (\n        Picture.objects.filter(people__user__in=(user1,))\n        .filter(people__user__in=(user2,))\n        .count()\n    )\n    if picture_count:\n        cls.logger.debug(\n            f\"\\t\\t- '{user1}' was pictured with '{user2}' {picture_count} times\"\n        )\n    return picture_count * cls.PICTURE_POINTS\n
"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy.compute_users_clubs_score","title":"compute_users_clubs_score(user1, user2) classmethod","text":"

Compute the clubs score of the relation between the given users.

The club score is obtained by counting the number of days during which the memberships (see :class:club.models.Membership) of both users overlapped.

For example, if user1 was a member of Unitec from 01/01/2020 to 31/12/2021 (two years) and user2 was a member of the same club from 01/01/2021 to 31/12/2022 (also two years, but with an offset of one year), then their club score is 365.

Returns:

Type Description int

the number of days during which both users were in the same club

Source code in galaxy/models.py
@classmethod\ndef compute_users_clubs_score(cls, user1: User, user2: User) -> int:\n    \"\"\"Compute the clubs score of the relation between the given users.\n\n    The club score is obtained by counting the number of days\n    during which the memberships (see :class:`club.models.Membership`)\n    of both users overlapped.\n\n    For example, if user1 was a member of Unitec from 01/01/2020 to 31/12/2021\n    (two years) and user2 was a member of the same club from 01/01/2021 to\n    31/12/2022 (also two years, but with an offset of one year), then their\n    club score is 365.\n\n    Returns:\n        the number of days during which both users were in the same club\n    \"\"\"\n    common_clubs = Club.objects.filter(members__in=user1.memberships.all()).filter(\n        members__in=user2.memberships.all()\n    )\n    user1_memberships = user1.memberships.filter(club__in=common_clubs)\n    user2_memberships = user2.memberships.filter(club__in=common_clubs)\n\n    score = 0\n    for user1_membership in user1_memberships:\n        if user1_membership.end_date is None:\n            # user1_membership.save() is not called in this function, hence this is safe\n            user1_membership.end_date = localdate()\n        query = Q(  # start2 <= start1 <= end2\n            start_date__lte=user1_membership.start_date,\n            end_date__gte=user1_membership.start_date,\n        )\n        query |= Q(  # start2 <= start1 <= now\n            start_date__lte=user1_membership.start_date, end_date=None\n        )\n        query |= Q(  # start1 <= start2 <= end2\n            start_date__gte=user1_membership.start_date,\n            start_date__lte=user1_membership.end_date,\n        )\n        for user2_membership in user2_memberships.filter(\n            query, club=user1_membership.club\n        ):\n            if user2_membership.end_date is None:\n                user2_membership.end_date = localdate()\n            latest_start = max(\n                user1_membership.start_date, user2_membership.start_date\n            )\n            earliest_end = min(user1_membership.end_date, user2_membership.end_date)\n            cls.logger.debug(\n                \"\\t\\t- '%s' was with '%s' in %s starting on %s until %s (%s days)\"\n                % (\n                    user1,\n                    user2,\n                    user2_membership.club,\n                    latest_start,\n                    earliest_end,\n                    (earliest_end - latest_start).days,\n                )\n            )\n            score += cls.CLUBS_POINTS * (earliest_end - latest_start).days\n    return score\n
"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy.scale_distance","title":"scale_distance(value) classmethod","text":"

Given a numeric value, return a scaled value which can be used in the Galaxy's graphical interface to set the distance between two stars.

Returns:

Type Description int

the scaled value usable in the Galaxy's 3d graph

Source code in galaxy/models.py
@classmethod\ndef scale_distance(cls, value: int | float) -> int:\n    \"\"\"Given a numeric value, return a scaled value which can\n    be used in the Galaxy's graphical interface to set the distance\n    between two stars.\n\n    Returns:\n        the scaled value usable in the Galaxy's 3d graph\n    \"\"\"\n    # TODO: this will need adjustements with the real, typical data on Taiste\n    if value == 0:\n        return 4000  # Following calculus would give us +\u221e, we cap it to 4000\n\n    cls.logger.debug(f\"\\t\\t> Score: {value}\")\n    # Invert score to draw close users together\n    value = 1 / value  # Cannot be 0\n    value += 2  # We use log2 just below and need to stay above 1\n    value = (  # Let's get something in the range ]0; log2(3)-1\u22480.58[ that we can multiply later\n        math.log2(value) - 1\n    )\n    value *= (  # Scale that value with a magic number to accommodate to typical data\n        # Really close galaxy citizen after 5 years typically have a score of about XXX\n        # Citizen that were in the same year without being really friends typically have a score of about XXX\n        # Citizen that have met once or twice only have a couple of pictures together typically score about XXX\n        cls.GALAXY_SCALE_FACTOR\n    )\n    cls.logger.debug(f\"\\t\\t> Scaled distance: {value}\")\n    return int(value)\n
"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy.rule","title":"rule(picture_count_threshold=10)","text":"

Main function of the Galaxy.

Iterate over all the rulable users to promote them to citizens. A citizen is a user who has a corresponding star in the Galaxy. Also build up the lanes, which are the links between the different citizen.

Users who can be ruled are defined with the picture_count_threshold: all users who are identified in a strictly lower number of pictures won't be promoted to citizens. This does very effectively limit the quantity of computing to do and only includes users who have had a minimum of activity.

This method still remains very expensive, so think thoroughly before you call it, especially in production.

:param picture_count_threshold: the minimum number of picture to have to be included in the galaxy

Source code in galaxy/models.py
def rule(self, picture_count_threshold=10) -> None:\n    \"\"\"Main function of the Galaxy.\n\n    Iterate over all the rulable users to promote them to citizens.\n    A citizen is a user who has a corresponding star in the Galaxy.\n    Also build up the lanes, which are the links between the different citizen.\n\n    Users who can be ruled are defined with the `picture_count_threshold`:\n    all users who are identified in a strictly lower number of pictures\n    won't be promoted to citizens.\n    This does very effectively limit the quantity of computing to do\n    and only includes users who have had a minimum of activity.\n\n    This method still remains very expensive, so think thoroughly before\n    you call it, especially in production.\n\n    :param picture_count_threshold: the minimum number of picture to have to be\n                                    included in the galaxy\n    \"\"\"\n    total_time = time.time()\n    self.logger.info(\"Listing rulable citizen.\")\n    rulable_users = (\n        User.objects.filter(subscriptions__isnull=False)\n        .annotate(pictures_count=Count(\"pictures\"))\n        .filter(pictures_count__gt=picture_count_threshold)\n        .distinct()\n    )\n\n    # force fetch of the whole query to make sure there won't\n    # be any more db hits\n    # this is memory expensive but prevents a lot of db hits, therefore\n    # is far more time efficient\n\n    rulable_users = list(rulable_users)\n    rulable_users_count = len(rulable_users)\n    user1_count = 0\n    self.logger.info(\n        f\"{rulable_users_count} citizen have been listed. Starting to rule.\"\n    )\n\n    stars = []\n    self.logger.info(\"Creating stars for all citizen\")\n    for user in rulable_users:\n        star = GalaxyStar(\n            owner=user, galaxy=self, mass=self.compute_user_score(user)\n        )\n        stars.append(star)\n    GalaxyStar.objects.bulk_create(stars)\n\n    stars = {}\n    for star in GalaxyStar.objects.filter(galaxy=self):\n        stars[star.owner.id] = star\n\n    self.logger.info(\"Creating lanes between stars\")\n    # Display current speed every $speed_count_frequency users\n    speed_count_frequency = max(rulable_users_count // 10, 1)  # ten time at most\n    global_avg_speed_accumulator = 0\n    global_avg_speed_count = 0\n    t_global_start = time.time()\n    while len(rulable_users) > 0:\n        user1 = rulable_users.pop()\n        user1_count += 1\n        rulable_users_count2 = len(rulable_users)\n\n        star1 = stars[user1.id]\n\n        user_avg_speed = 0\n        user_avg_speed_count = 0\n\n        tstart = time.time()\n        lanes = []\n        for user2_count, user2 in enumerate(rulable_users, start=1):\n            self.logger.debug(\"\")\n            self.logger.debug(\n                f\"\\t> Examining '{user1}' ({user1_count}/{rulable_users_count}) with '{user2}' ({user2_count}/{rulable_users_count2})\"\n            )\n\n            star2 = stars[user2.id]\n\n            score = Galaxy.compute_users_score(user1, user2)\n            distance = self.scale_distance(sum(score))\n            if distance < 30:  # TODO: this needs tuning with real-world data\n                lanes.append(\n                    GalaxyLane(\n                        star1=star1,\n                        star2=star2,\n                        distance=distance,\n                        family=score.family,\n                        pictures=score.pictures,\n                        clubs=score.clubs,\n                    )\n                )\n\n            if user2_count % speed_count_frequency == 0:\n                tend = time.time()\n                delta = tend - tstart\n                speed = float(speed_count_frequency) / delta\n                user_avg_speed += speed\n                user_avg_speed_count += 1\n                self.logger.debug(\n                    f\"\\tSpeed: {speed:.2f} users per second (time for last {speed_count_frequency} citizens: {delta:.2f} second)\"\n                )\n                tstart = time.time()\n\n        GalaxyLane.objects.bulk_create(lanes)\n\n        self.logger.info(\"\")\n\n        t_global_end = time.time()\n        global_delta = t_global_end - t_global_start\n        speed = 1.0 / global_delta\n        global_avg_speed_accumulator += speed\n        global_avg_speed_count += 1\n        global_avg_speed = global_avg_speed_accumulator / global_avg_speed_count\n\n        self.logger.info(f\" Ruling of {self} \".center(60, \"#\"))\n        self.logger.info(\n            f\"Progression: {user1_count}/{rulable_users_count} citizen -- {rulable_users_count - user1_count} remaining\"\n        )\n        self.logger.info(f\"Speed: {60.0*global_avg_speed:.2f} citizen per minute\")\n\n        # We can divide the computed ETA by 2 because each loop, there is one citizen less to check, and maths tell\n        # us that this averages to a division by two\n        eta = rulable_users_count2 / global_avg_speed / 2\n        eta_hours = int(eta // 3600)\n        eta_minutes = int(eta // 60 % 60)\n        self.logger.info(\n            f\"ETA: {eta_hours} hours {eta_minutes} minutes ({eta / 3600 / 24:.2f} days)\"\n        )\n        self.logger.info(\"#\" * 60)\n        t_global_start = time.time()\n\n    # Here, we get the IDs of the old galaxies that we'll need to delete. In normal operation, only one galaxy\n    # should be returned, and we can't delete it yet, as it's the one still displayed by the Sith.\n    old_galaxies_pks = list(\n        Galaxy.objects.filter(state__isnull=False).values_list(\"pk\", flat=True)\n    )\n    self.logger.info(\n        f\"These old galaxies will be deleted once the new one is ready: {old_galaxies_pks}\"\n    )\n\n    # Making the state sets this new galaxy as being ready. From now on, the Sith will show us to the world.\n    self.make_state()\n\n    # Avoid accident if there is nothing to delete\n    if len(old_galaxies_pks) > 0:\n        # Former galaxies can now be deleted.\n        Galaxy.objects.filter(pk__in=old_galaxies_pks).delete()\n\n    total_time = time.time() - total_time\n    total_time_hours = int(total_time // 3600)\n    total_time_minutes = int(total_time // 60 % 60)\n    total_time_seconds = int(total_time % 60)\n    self.logger.info(\n        f\"{self} ruled in {total_time:.2f} seconds ({total_time_hours} hours, {total_time_minutes} minutes, {total_time_seconds} seconds)\"\n    )\n
"},{"location":"reference/galaxy/views/#galaxy.views.Galaxy.make_state","title":"make_state()","text":"

Compute JSON structure to send to 3d-force-graph: https://github.com/vasturiano/3d-force-graph/.

Source code in galaxy/models.py
def make_state(self) -> None:\n    \"\"\"Compute JSON structure to send to 3d-force-graph: https://github.com/vasturiano/3d-force-graph/.\"\"\"\n    self.logger.info(\n        \"Caching current Galaxy state for a quicker display of the Empire's power.\"\n    )\n\n    without_nickname = Concat(\n        F(\"owner__first_name\"), Value(\" \"), F(\"owner__last_name\")\n    )\n    with_nickname = Concat(\n        F(\"owner__first_name\"),\n        Value(\" \"),\n        F(\"owner__last_name\"),\n        Value(\" (\"),\n        F(\"owner__nick_name\"),\n        Value(\")\"),\n    )\n    stars = (\n        GalaxyStar.objects.filter(galaxy=self)\n        .order_by(\n            \"owner\"\n        )  # This helps determinism for the tests and doesn't cost much\n        .annotate(\n            owner_name=Case(\n                When(owner__nick_name=None, then=without_nickname),\n                default=with_nickname,\n            )\n        )\n    )\n    lanes = (\n        GalaxyLane.objects.filter(star1__galaxy=self)\n        .order_by(\n            \"star1\"\n        )  # This helps determinism for the tests and doesn't cost much\n        .annotate(\n            star1_owner=F(\"star1__owner__id\"),\n            star2_owner=F(\"star2__owner__id\"),\n        )\n    )\n    json = GalaxyDict(\n        nodes=[\n            StarDict(\n                id=star.owner_id,\n                name=star.owner_name,\n                mass=star.mass,\n            )\n            for star in stars\n        ],\n        links=[],\n    )\n    for path in lanes:\n        json[\"links\"].append(\n            {\n                \"source\": path.star1_owner,\n                \"target\": path.star2_owner,\n                \"value\": path.distance,\n            }\n        )\n    self.state = json\n    self.save()\n    self.logger.info(f\"{self} is now ready!\")\n
"},{"location":"reference/galaxy/views/#galaxy.views.GalaxyLane","title":"GalaxyLane","text":"

Bases: Model

Define a lane (edge -> link between galaxy citizen) in the galaxy map.

Store a reference to both its ends and the distance it covers. Score details between citizen owning the stars is also stored here.

"},{"location":"reference/galaxy/views/#galaxy.views.GalaxyUserView","title":"GalaxyUserView","text":"

Bases: CanViewMixin, UserTabsMixin, DetailView

"},{"location":"reference/galaxy/views/#galaxy.views.GalaxyDataView","title":"GalaxyDataView","text":"

Bases: FormerSubscriberMixin, View

"},{"location":"reference/launderette/models/","title":"Models","text":""},{"location":"reference/launderette/models/#launderette.models.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/launderette/models/#launderette.models.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/launderette/models/#launderette.models.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/launderette/models/#launderette.models.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/launderette/models/#launderette.models.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/launderette/models/#launderette.models.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/launderette/models/#launderette.models.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/launderette/models/#launderette.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/launderette/models/#launderette.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/launderette/models/#launderette.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/launderette/models/#launderette.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/launderette/models/#launderette.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/launderette/models/#launderette.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/launderette/models/#launderette.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/launderette/models/#launderette.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/launderette/models/#launderette.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/launderette/models/#launderette.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/launderette/models/#launderette.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/launderette/models/#launderette.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/launderette/models/#launderette.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/launderette/models/#launderette.models.Counter","title":"Counter","text":"

Bases: Model

"},{"location":"reference/launderette/models/#launderette.models.Counter.gen_token","title":"gen_token()","text":"

Generate a new token for this counter.

Source code in counter/models.py
def gen_token(self) -> None:\n    \"\"\"Generate a new token for this counter.\"\"\"\n    self.token = \"\".join(\n        random.choice(string.ascii_letters + string.digits) for _ in range(30)\n    )\n    self.save()\n
"},{"location":"reference/launderette/models/#launderette.models.Counter.barmen_list","title":"barmen_list()","text":"

Returns the barman list as list of User.

Source code in counter/models.py
@cached_property\ndef barmen_list(self) -> list[User]:\n    \"\"\"Returns the barman list as list of User.\"\"\"\n    return [\n        p.user for p in self.permanencies.filter(end=None).select_related(\"user\")\n    ]\n
"},{"location":"reference/launderette/models/#launderette.models.Counter.get_random_barman","title":"get_random_barman()","text":"

Return a random user being currently a barman.

Source code in counter/models.py
def get_random_barman(self) -> User:\n    \"\"\"Return a random user being currently a barman.\"\"\"\n    return random.choice(self.barmen_list)\n
"},{"location":"reference/launderette/models/#launderette.models.Counter.update_activity","title":"update_activity()","text":"

Update the barman activity to prevent timeout.

Source code in counter/models.py
def update_activity(self) -> None:\n    \"\"\"Update the barman activity to prevent timeout.\"\"\"\n    self.permanencies.filter(end=None).update(activity=timezone.now())\n
"},{"location":"reference/launderette/models/#launderette.models.Counter.can_refill","title":"can_refill()","text":"

Show if the counter authorize the refilling with physic money.

Source code in counter/models.py
def can_refill(self) -> bool:\n    \"\"\"Show if the counter authorize the refilling with physic money.\"\"\"\n    if self.type != \"BAR\":\n        return False\n    # at least one of the barmen is in the AE board\n    ae = Club.objects.get(unix_name=SITH_MAIN_CLUB[\"unix_name\"])\n    return any(ae.get_membership_for(barman) for barman in self.barmen_list)\n
"},{"location":"reference/launderette/models/#launderette.models.Counter.get_top_barmen","title":"get_top_barmen()","text":"

Return a QuerySet querying the office hours stats of all the barmen of all time of this counter, ordered by descending number of hours.

Each element of the QuerySet corresponds to a barman and has the following data Source code in counter/models.py
def get_top_barmen(self) -> QuerySet:\n    \"\"\"Return a QuerySet querying the office hours stats of all the barmen of all time\n    of this counter, ordered by descending number of hours.\n\n    Each element of the QuerySet corresponds to a barman and has the following data :\n        - the full name (first name + last name) of the barman\n        - the nickname of the barman\n        - the promo of the barman\n        - the total number of office hours the barman did attend\n    \"\"\"\n    return (\n        self.permanencies.exclude(end=None)\n        .annotate(\n            name=Concat(F(\"user__first_name\"), Value(\" \"), F(\"user__last_name\"))\n        )\n        .annotate(nickname=F(\"user__nick_name\"))\n        .annotate(promo=F(\"user__promo\"))\n        .values(\"user\", \"name\", \"nickname\", \"promo\")\n        .annotate(perm_sum=Sum(F(\"end\") - F(\"start\")))\n        .exclude(perm_sum=None)\n        .order_by(\"-perm_sum\")\n    )\n
"},{"location":"reference/launderette/models/#launderette.models.Counter.get_top_customers","title":"get_top_customers(since=None)","text":"

Return a QuerySet querying the money spent by customers of this counter since the specified date, ordered by descending amount of money spent.

Each element of the QuerySet corresponds to a customer and has the following data :

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None Source code in counter/models.py
def get_top_customers(self, since: datetime | date | None = None) -> QuerySet:\n    \"\"\"Return a QuerySet querying the money spent by customers of this counter\n    since the specified date, ordered by descending amount of money spent.\n\n    Each element of the QuerySet corresponds to a customer and has the following data :\n\n    - the full name (first name + last name) of the customer\n    - the nickname of the customer\n    - the amount of money spent by the customer\n\n    Args:\n        since: timestamp from which to perform the calculation\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return (\n        self.sellings.filter(date__gte=since)\n        .annotate(\n            name=Concat(\n                F(\"customer__user__first_name\"),\n                Value(\" \"),\n                F(\"customer__user__last_name\"),\n            )\n        )\n        .annotate(nickname=F(\"customer__user__nick_name\"))\n        .annotate(promo=F(\"customer__user__promo\"))\n        .annotate(user=F(\"customer__user\"))\n        .values(\"user\", \"promo\", \"name\", \"nickname\")\n        .annotate(\n            selling_sum=Sum(\n                F(\"unit_price\") * F(\"quantity\"), output_field=CurrencyField()\n            )\n        )\n        .filter(selling_sum__gt=0)\n        .order_by(\"-selling_sum\")\n    )\n
"},{"location":"reference/launderette/models/#launderette.models.Counter.get_total_sales","title":"get_total_sales(since=None)","text":"

Compute and return the total turnover of this counter since the given date.

By default, the date is the start of the current semester.

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None

Returns:

Type Description CurrencyField

Total revenue earned at this counter.

Source code in counter/models.py
def get_total_sales(self, since: datetime | date | None = None) -> CurrencyField:\n    \"\"\"Compute and return the total turnover of this counter since the given date.\n\n    By default, the date is the start of the current semester.\n\n    Args:\n        since: timestamp from which to perform the calculation\n\n    Returns:\n        Total revenue earned at this counter.\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return self.sellings.filter(date__gte=since).aggregate(\n        total=Sum(\n            F(\"quantity\") * F(\"unit_price\"),\n            default=0,\n            output_field=CurrencyField(),\n        )\n    )[\"total\"]\n
"},{"location":"reference/launderette/models/#launderette.models.Counter.customer_is_barman","title":"customer_is_barman(customer)","text":"

Check if this counter is a bar and if the customer is currently logged in. This is useful to compute special prices.

Source code in counter/models.py
def customer_is_barman(self, customer: Customer | User) -> bool:\n    \"\"\"Check if this counter is a `bar` and if the customer is currently logged in.\n    This is useful to compute special prices.\"\"\"\n\n    # Customer and User are two different tables,\n    # but they share the same primary key\n    return self.type == \"BAR\" and any(b.pk == customer.pk for b in self.barmen_list)\n
"},{"location":"reference/launderette/models/#launderette.models.Counter.get_products_for","title":"get_products_for(customer)","text":"

Get all allowed products for the provided customer on this counter Prices will be annotated

Source code in counter/models.py
def get_products_for(self, customer: Customer) -> list[Product]:\n    \"\"\"\n    Get all allowed products for the provided customer on this counter\n    Prices will be annotated\n    \"\"\"\n\n    products = self.products.select_related(\"product_type\").prefetch_related(\n        \"buying_groups\"\n    )\n\n    # Only include age appropriate products\n    age = customer.user.age\n    if customer.user.is_banned_alcohol:\n        age = min(age, 17)\n    products = products.filter(limit_age__lte=age)\n\n    # Compute special price for customer if he is a barmen on that bar\n    if self.customer_is_barman(customer):\n        products = products.annotate(price=F(\"special_selling_price\"))\n    else:\n        products = products.annotate(price=F(\"selling_price\"))\n\n    return [\n        product\n        for product in products.all()\n        if product.can_be_sold_to(customer.user)\n    ]\n
"},{"location":"reference/launderette/models/#launderette.models.Launderette","title":"Launderette","text":"

Bases: Model

"},{"location":"reference/launderette/models/#launderette.models.Launderette.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in launderette/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    launderette_club = Club.objects.filter(\n        unix_name=settings.SITH_LAUNDERETTE_MANAGER[\"unix_name\"]\n    ).first()\n    m = launderette_club.get_membership_for(user)\n    return bool(m and m.role >= 9)\n
"},{"location":"reference/launderette/models/#launderette.models.Machine","title":"Machine","text":"

Bases: Model

"},{"location":"reference/launderette/models/#launderette.models.Machine.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in launderette/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    launderette_club = Club.objects.filter(\n        unix_name=settings.SITH_LAUNDERETTE_MANAGER[\"unix_name\"]\n    ).first()\n    m = launderette_club.get_membership_for(user)\n    return bool(m and m.role >= 9)\n
"},{"location":"reference/launderette/models/#launderette.models.Token","title":"Token","text":"

Bases: Model

"},{"location":"reference/launderette/models/#launderette.models.Token.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in launderette/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    launderette_club = Club.objects.filter(\n        unix_name=settings.SITH_LAUNDERETTE_MANAGER[\"unix_name\"]\n    ).first()\n    m = launderette_club.get_membership_for(user)\n    return bool(m and m.role >= 9)\n
"},{"location":"reference/launderette/models/#launderette.models.Slot","title":"Slot","text":"

Bases: Model

"},{"location":"reference/launderette/views/","title":"Views","text":""},{"location":"reference/launderette/views/#launderette.views.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/launderette/views/#launderette.views.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/launderette/views/#launderette.views.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/launderette/views/#launderette.views.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/launderette/views/#launderette.views.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/launderette/views/#launderette.views.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/launderette/views/#launderette.views.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/launderette/views/#launderette.views.CanEditMixin","title":"CanEditMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to edit this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/launderette/views/#launderette.views.CanEditPropMixin","title":"CanEditPropMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has owner permissions on the child view object.

In other word, you can make a view with this view as parent, and it will be retricted to the users that are in the object's owner_group or that pass the obj.can_be_viewed_by test.

Raises:

Type Description PermissionDenied

If the user cannot see the object

"},{"location":"reference/launderette/views/#launderette.views.CanViewMixin","title":"CanViewMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to view this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/launderette/views/#launderette.views.Page","title":"Page","text":"

Bases: Model

The page class to build a Wiki Each page may have a parent and it's URL is of the form my.site/page/// It has an ID field, but don't use it, since it's only there for DB part, and because compound primary key is awkward! Prefere querying pages with Page.get_page_by_full_name().

Be careful with the _full_name attribute: this field may not be valid until you call save(). It's made for fast query, but don't rely on it when playing with a Page object, use get_full_name() instead!

"},{"location":"reference/launderette/views/#launderette.views.Page.save","title":"save(*args, **kwargs)","text":"

Performs some needed actions before and after saving a page in database.

Source code in core/models.py
def save(self, *args, **kwargs):\n    \"\"\"Performs some needed actions before and after saving a page in database.\"\"\"\n    locked = kwargs.pop(\"force_lock\", False)\n    if not locked:\n        locked = self.is_locked()\n    if not locked:\n        raise NotLocked(\"The page is not locked and thus can not be saved\")\n    self.full_clean()\n    if not self.id:\n        super().save(\n            *args, **kwargs\n        )  # Save a first time to correctly set _full_name\n    # This reset the _full_name just before saving to maintain a coherent field quicker for queries than the\n    # recursive method\n    # It also update all the children to maintain correct names\n    self._full_name = self.get_full_name()\n    for c in self.children.all():\n        c.save()\n    super().save(*args, **kwargs)\n    self.unset_lock()\n
"},{"location":"reference/launderette/views/#launderette.views.Page.get_page_by_full_name","title":"get_page_by_full_name(name) staticmethod","text":"

Quicker to get a page with that method rather than building the request every time.

Source code in core/models.py
@staticmethod\ndef get_page_by_full_name(name):\n    \"\"\"Quicker to get a page with that method rather than building the request every time.\"\"\"\n    return Page.objects.filter(_full_name=name).first()\n
"},{"location":"reference/launderette/views/#launderette.views.Page.clean","title":"clean()","text":"

Cleans up only the name for the moment, but this can be used to make any treatment before saving the object.

Source code in core/models.py
def clean(self):\n    \"\"\"Cleans up only the name for the moment, but this can be used to make any treatment before saving the object.\"\"\"\n    if \"/\" in self.name:\n        self.name = self.name.split(\"/\")[-1]\n    if (\n        Page.objects.exclude(pk=self.pk)\n        .filter(_full_name=self.get_full_name())\n        .exists()\n    ):\n        raise ValidationError(_(\"Duplicate page\"), code=\"duplicate\")\n    super().clean()\n    if self.parent is not None and self in self.get_parent_list():\n        raise ValidationError(_(\"Loop in page tree\"), code=\"loop\")\n
"},{"location":"reference/launderette/views/#launderette.views.Page.is_locked","title":"is_locked()","text":"

Is True if the page is locked, False otherwise.

This is where the timeout is handled, so a locked page for which the timeout is reach will be unlocked and this function will return False.

Source code in core/models.py
def is_locked(self):\n    \"\"\"Is True if the page is locked, False otherwise.\n\n    This is where the timeout is handled,\n    so a locked page for which the timeout is reach will be unlocked and this\n    function will return False.\n    \"\"\"\n    if self.lock_timeout and (\n        timezone.now() - self.lock_timeout > timedelta(minutes=5)\n    ):\n        self.unset_lock()\n    return (\n        self.lock_user\n        and self.lock_timeout\n        and (timezone.now() - self.lock_timeout < timedelta(minutes=5))\n    )\n
"},{"location":"reference/launderette/views/#launderette.views.Page.set_lock","title":"set_lock(user)","text":"

Sets a lock on the current page or raise an AlreadyLocked exception.

Source code in core/models.py
def set_lock(self, user):\n    \"\"\"Sets a lock on the current page or raise an AlreadyLocked exception.\"\"\"\n    if self.is_locked() and self.get_lock() != user:\n        raise AlreadyLocked(\"The page is already locked by someone else\")\n    self.lock_user = user\n    self.lock_timeout = timezone.now()\n    super().save()\n
"},{"location":"reference/launderette/views/#launderette.views.Page.set_lock_recursive","title":"set_lock_recursive(user)","text":"

Locks recursively all the child pages for editing properties.

Source code in core/models.py
def set_lock_recursive(self, user):\n    \"\"\"Locks recursively all the child pages for editing properties.\"\"\"\n    for p in self.children.all():\n        p.set_lock_recursive(user)\n    self.set_lock(user)\n
"},{"location":"reference/launderette/views/#launderette.views.Page.unset_lock_recursive","title":"unset_lock_recursive()","text":"

Unlocks recursively all the child pages.

Source code in core/models.py
def unset_lock_recursive(self):\n    \"\"\"Unlocks recursively all the child pages.\"\"\"\n    for p in self.children.all():\n        p.unset_lock_recursive()\n    self.unset_lock()\n
"},{"location":"reference/launderette/views/#launderette.views.Page.unset_lock","title":"unset_lock()","text":"

Always try to unlock, even if there is no lock.

Source code in core/models.py
def unset_lock(self):\n    \"\"\"Always try to unlock, even if there is no lock.\"\"\"\n    self.lock_user = None\n    self.lock_timeout = None\n    super().save()\n
"},{"location":"reference/launderette/views/#launderette.views.Page.get_lock","title":"get_lock()","text":"

Returns the page's mutex containing the time and the user in a dict.

Source code in core/models.py
def get_lock(self):\n    \"\"\"Returns the page's mutex containing the time and the user in a dict.\"\"\"\n    if self.lock_user:\n        return self.lock_user\n    raise NotLocked(\"The page is not locked and thus can not return its user\")\n
"},{"location":"reference/launderette/views/#launderette.views.Page.get_full_name","title":"get_full_name()","text":"

Computes the real full_name of the page based on its name and its parent's name You can and must rely on this function when working on a page object that is not freshly fetched from the DB (For example when treating a Page object coming from a form).

Source code in core/models.py
def get_full_name(self):\n    \"\"\"Computes the real full_name of the page based on its name and its parent's name\n    You can and must rely on this function when working on a page object that is not freshly fetched from the DB\n    (For example when treating a Page object coming from a form).\n    \"\"\"\n    if self.parent is None:\n        return self.name\n    return f\"{self.parent.get_full_name()}/{self.name}\"\n
"},{"location":"reference/launderette/views/#launderette.views.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/launderette/views/#launderette.views.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/launderette/views/#launderette.views.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/launderette/views/#launderette.views.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/launderette/views/#launderette.views.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/launderette/views/#launderette.views.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/launderette/views/#launderette.views.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/launderette/views/#launderette.views.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/launderette/views/#launderette.views.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/launderette/views/#launderette.views.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/launderette/views/#launderette.views.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/launderette/views/#launderette.views.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/launderette/views/#launderette.views.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/launderette/views/#launderette.views.GetUserForm","title":"GetUserForm","text":"

Bases: Form

The Form class aims at providing a valid user_id field in its cleaned data, in order to pass it to some view, reverse function, or any other use.

The Form implements a nice JS widget allowing the user to type a customer account id, or search the database with some nickname, first name, or last name (TODO)

"},{"location":"reference/launderette/views/#launderette.views.Counter","title":"Counter","text":"

Bases: Model

"},{"location":"reference/launderette/views/#launderette.views.Counter.gen_token","title":"gen_token()","text":"

Generate a new token for this counter.

Source code in counter/models.py
def gen_token(self) -> None:\n    \"\"\"Generate a new token for this counter.\"\"\"\n    self.token = \"\".join(\n        random.choice(string.ascii_letters + string.digits) for _ in range(30)\n    )\n    self.save()\n
"},{"location":"reference/launderette/views/#launderette.views.Counter.barmen_list","title":"barmen_list()","text":"

Returns the barman list as list of User.

Source code in counter/models.py
@cached_property\ndef barmen_list(self) -> list[User]:\n    \"\"\"Returns the barman list as list of User.\"\"\"\n    return [\n        p.user for p in self.permanencies.filter(end=None).select_related(\"user\")\n    ]\n
"},{"location":"reference/launderette/views/#launderette.views.Counter.get_random_barman","title":"get_random_barman()","text":"

Return a random user being currently a barman.

Source code in counter/models.py
def get_random_barman(self) -> User:\n    \"\"\"Return a random user being currently a barman.\"\"\"\n    return random.choice(self.barmen_list)\n
"},{"location":"reference/launderette/views/#launderette.views.Counter.update_activity","title":"update_activity()","text":"

Update the barman activity to prevent timeout.

Source code in counter/models.py
def update_activity(self) -> None:\n    \"\"\"Update the barman activity to prevent timeout.\"\"\"\n    self.permanencies.filter(end=None).update(activity=timezone.now())\n
"},{"location":"reference/launderette/views/#launderette.views.Counter.can_refill","title":"can_refill()","text":"

Show if the counter authorize the refilling with physic money.

Source code in counter/models.py
def can_refill(self) -> bool:\n    \"\"\"Show if the counter authorize the refilling with physic money.\"\"\"\n    if self.type != \"BAR\":\n        return False\n    # at least one of the barmen is in the AE board\n    ae = Club.objects.get(unix_name=SITH_MAIN_CLUB[\"unix_name\"])\n    return any(ae.get_membership_for(barman) for barman in self.barmen_list)\n
"},{"location":"reference/launderette/views/#launderette.views.Counter.get_top_barmen","title":"get_top_barmen()","text":"

Return a QuerySet querying the office hours stats of all the barmen of all time of this counter, ordered by descending number of hours.

Each element of the QuerySet corresponds to a barman and has the following data Source code in counter/models.py
def get_top_barmen(self) -> QuerySet:\n    \"\"\"Return a QuerySet querying the office hours stats of all the barmen of all time\n    of this counter, ordered by descending number of hours.\n\n    Each element of the QuerySet corresponds to a barman and has the following data :\n        - the full name (first name + last name) of the barman\n        - the nickname of the barman\n        - the promo of the barman\n        - the total number of office hours the barman did attend\n    \"\"\"\n    return (\n        self.permanencies.exclude(end=None)\n        .annotate(\n            name=Concat(F(\"user__first_name\"), Value(\" \"), F(\"user__last_name\"))\n        )\n        .annotate(nickname=F(\"user__nick_name\"))\n        .annotate(promo=F(\"user__promo\"))\n        .values(\"user\", \"name\", \"nickname\", \"promo\")\n        .annotate(perm_sum=Sum(F(\"end\") - F(\"start\")))\n        .exclude(perm_sum=None)\n        .order_by(\"-perm_sum\")\n    )\n
"},{"location":"reference/launderette/views/#launderette.views.Counter.get_top_customers","title":"get_top_customers(since=None)","text":"

Return a QuerySet querying the money spent by customers of this counter since the specified date, ordered by descending amount of money spent.

Each element of the QuerySet corresponds to a customer and has the following data :

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None Source code in counter/models.py
def get_top_customers(self, since: datetime | date | None = None) -> QuerySet:\n    \"\"\"Return a QuerySet querying the money spent by customers of this counter\n    since the specified date, ordered by descending amount of money spent.\n\n    Each element of the QuerySet corresponds to a customer and has the following data :\n\n    - the full name (first name + last name) of the customer\n    - the nickname of the customer\n    - the amount of money spent by the customer\n\n    Args:\n        since: timestamp from which to perform the calculation\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return (\n        self.sellings.filter(date__gte=since)\n        .annotate(\n            name=Concat(\n                F(\"customer__user__first_name\"),\n                Value(\" \"),\n                F(\"customer__user__last_name\"),\n            )\n        )\n        .annotate(nickname=F(\"customer__user__nick_name\"))\n        .annotate(promo=F(\"customer__user__promo\"))\n        .annotate(user=F(\"customer__user\"))\n        .values(\"user\", \"promo\", \"name\", \"nickname\")\n        .annotate(\n            selling_sum=Sum(\n                F(\"unit_price\") * F(\"quantity\"), output_field=CurrencyField()\n            )\n        )\n        .filter(selling_sum__gt=0)\n        .order_by(\"-selling_sum\")\n    )\n
"},{"location":"reference/launderette/views/#launderette.views.Counter.get_total_sales","title":"get_total_sales(since=None)","text":"

Compute and return the total turnover of this counter since the given date.

By default, the date is the start of the current semester.

Parameters:

Name Type Description Default since datetime | date | None

timestamp from which to perform the calculation

None

Returns:

Type Description CurrencyField

Total revenue earned at this counter.

Source code in counter/models.py
def get_total_sales(self, since: datetime | date | None = None) -> CurrencyField:\n    \"\"\"Compute and return the total turnover of this counter since the given date.\n\n    By default, the date is the start of the current semester.\n\n    Args:\n        since: timestamp from which to perform the calculation\n\n    Returns:\n        Total revenue earned at this counter.\n    \"\"\"\n    if since is None:\n        since = get_start_of_semester()\n    if isinstance(since, date):\n        since = datetime(since.year, since.month, since.day, tzinfo=tz.utc)\n    return self.sellings.filter(date__gte=since).aggregate(\n        total=Sum(\n            F(\"quantity\") * F(\"unit_price\"),\n            default=0,\n            output_field=CurrencyField(),\n        )\n    )[\"total\"]\n
"},{"location":"reference/launderette/views/#launderette.views.Counter.customer_is_barman","title":"customer_is_barman(customer)","text":"

Check if this counter is a bar and if the customer is currently logged in. This is useful to compute special prices.

Source code in counter/models.py
def customer_is_barman(self, customer: Customer | User) -> bool:\n    \"\"\"Check if this counter is a `bar` and if the customer is currently logged in.\n    This is useful to compute special prices.\"\"\"\n\n    # Customer and User are two different tables,\n    # but they share the same primary key\n    return self.type == \"BAR\" and any(b.pk == customer.pk for b in self.barmen_list)\n
"},{"location":"reference/launderette/views/#launderette.views.Counter.get_products_for","title":"get_products_for(customer)","text":"

Get all allowed products for the provided customer on this counter Prices will be annotated

Source code in counter/models.py
def get_products_for(self, customer: Customer) -> list[Product]:\n    \"\"\"\n    Get all allowed products for the provided customer on this counter\n    Prices will be annotated\n    \"\"\"\n\n    products = self.products.select_related(\"product_type\").prefetch_related(\n        \"buying_groups\"\n    )\n\n    # Only include age appropriate products\n    age = customer.user.age\n    if customer.user.is_banned_alcohol:\n        age = min(age, 17)\n    products = products.filter(limit_age__lte=age)\n\n    # Compute special price for customer if he is a barmen on that bar\n    if self.customer_is_barman(customer):\n        products = products.annotate(price=F(\"special_selling_price\"))\n    else:\n        products = products.annotate(price=F(\"selling_price\"))\n\n    return [\n        product\n        for product in products.all()\n        if product.can_be_sold_to(customer.user)\n    ]\n
"},{"location":"reference/launderette/views/#launderette.views.Customer","title":"Customer","text":"

Bases: Model

Customer data of a User.

It adds some basic customers' information, such as the account ID, and is used by other accounting classes as reference to the customer, rather than using User.

"},{"location":"reference/launderette/views/#launderette.views.Customer.can_buy","title":"can_buy property","text":"

Check if whether this customer has the right to purchase any item.

This must be not confused with the Product.can_be_sold_to(user) method as the present method returns an information about a customer whereas the other tells something about the relation between a User (not a Customer, don't mix them) and a Product.

"},{"location":"reference/launderette/views/#launderette.views.Customer.save","title":"save(*args, allow_negative=False, is_selling=False, **kwargs)","text":"

is_selling : tell if the current action is a selling allow_negative : ignored if not a selling. Allow a selling to put the account in negative Those two parameters avoid blocking the save method of a customer if his account is negative.

Source code in counter/models.py
def save(self, *args, allow_negative=False, is_selling=False, **kwargs):\n    \"\"\"is_selling : tell if the current action is a selling\n    allow_negative : ignored if not a selling. Allow a selling to put the account in negative\n    Those two parameters avoid blocking the save method of a customer if his account is negative.\n    \"\"\"\n    if self.amount < 0 and (is_selling and not allow_negative):\n        raise ValidationError(_(\"Not enough money\"))\n    super().save(*args, **kwargs)\n
"},{"location":"reference/launderette/views/#launderette.views.Customer.get_or_create","title":"get_or_create(user) classmethod","text":"

Work in pretty much the same way as the usual get_or_create method, but with the default field replaced by some under the hood.

If the user has an account, return it as is. Else create a new account with no money on it and a new unique account id

Example : ::

user = User.objects.get(pk=1)\naccount, created = Customer.get_or_create(user)\nif created:\n    print(f\"created a new account with id {account.id}\")\nelse:\n    print(f\"user has already an account, with {account.id} \u20ac on it\"\n
Source code in counter/models.py
@classmethod\ndef get_or_create(cls, user: User) -> tuple[Customer, bool]:\n    \"\"\"Work in pretty much the same way as the usual get_or_create method,\n    but with the default field replaced by some under the hood.\n\n    If the user has an account, return it as is.\n    Else create a new account with no money on it and a new unique account id\n\n    Example : ::\n\n        user = User.objects.get(pk=1)\n        account, created = Customer.get_or_create(user)\n        if created:\n            print(f\"created a new account with id {account.id}\")\n        else:\n            print(f\"user has already an account, with {account.id} \u20ac on it\"\n    \"\"\"\n    if hasattr(user, \"customer\"):\n        return user.customer, False\n\n    # account_id are always a number with a letter appended\n    account_id = (\n        Customer.objects.order_by(Length(\"account_id\"), \"account_id\")\n        .values(\"account_id\")\n        .last()\n    )\n    if account_id is None:\n        # legacy from the old site\n        account = cls.objects.create(user=user, account_id=\"1504a\")\n        return account, True\n\n    account_id = account_id[\"account_id\"]\n    account_num = int(account_id[:-1])\n    while Customer.objects.filter(account_id=account_id).exists():\n        # when entering the first iteration, we are using an already existing account id\n        # so the loop should always execute at least one time\n        account_num += 1\n        account_id = f\"{account_num}{random.choice(string.ascii_lowercase)}\"\n\n    account = cls.objects.create(user=user, account_id=account_id)\n    return account, True\n
"},{"location":"reference/launderette/views/#launderette.views.Selling","title":"Selling","text":"

Bases: Model

Handle the sellings.

"},{"location":"reference/launderette/views/#launderette.views.Selling.save","title":"save(*args, allow_negative=False, **kwargs)","text":"

allow_negative : Allow this selling to use more money than available for this user.

Source code in counter/models.py
def save(self, *args, allow_negative=False, **kwargs):\n    \"\"\"allow_negative : Allow this selling to use more money than available for this user.\"\"\"\n    if not self.date:\n        self.date = timezone.now()\n    self.full_clean()\n    if not self.is_validated:\n        self.customer.amount -= self.quantity * self.unit_price\n        self.customer.save(allow_negative=allow_negative, is_selling=True)\n        self.is_validated = True\n    user = self.customer.user\n    if user.was_subscribed:\n        if (\n            self.product\n            and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_ONE_SEMESTER\n        ):\n            sub = Subscription(\n                member=user,\n                subscription_type=\"un-semestre\",\n                payment_method=\"EBOUTIC\",\n                location=\"EBOUTIC\",\n            )\n            sub.subscription_start = Subscription.compute_start()\n            sub.subscription_start = Subscription.compute_start(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ]\n            )\n            sub.subscription_end = Subscription.compute_end(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ],\n                start=sub.subscription_start,\n            )\n            sub.save()\n        elif (\n            self.product\n            and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_TWO_SEMESTERS\n        ):\n            sub = Subscription(\n                member=user,\n                subscription_type=\"deux-semestres\",\n                payment_method=\"EBOUTIC\",\n                location=\"EBOUTIC\",\n            )\n            sub.subscription_start = Subscription.compute_start()\n            sub.subscription_start = Subscription.compute_start(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ]\n            )\n            sub.subscription_end = Subscription.compute_end(\n                duration=settings.SITH_SUBSCRIPTIONS[sub.subscription_type][\n                    \"duration\"\n                ],\n                start=sub.subscription_start,\n            )\n            sub.save()\n    if user.preferences.notify_on_click:\n        Notification(\n            user=user,\n            url=reverse(\n                \"core:user_account_detail\",\n                kwargs={\n                    \"user_id\": user.id,\n                    \"year\": self.date.year,\n                    \"month\": self.date.month,\n                },\n            ),\n            param=\"%d x %s\" % (self.quantity, self.label),\n            type=\"SELLING\",\n        ).save()\n    super().save(*args, **kwargs)\n    if hasattr(self.product, \"eticket\"):\n        self.send_mail_customer()\n
"},{"location":"reference/launderette/views/#launderette.views.Launderette","title":"Launderette","text":"

Bases: Model

"},{"location":"reference/launderette/views/#launderette.views.Launderette.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in launderette/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    launderette_club = Club.objects.filter(\n        unix_name=settings.SITH_LAUNDERETTE_MANAGER[\"unix_name\"]\n    ).first()\n    m = launderette_club.get_membership_for(user)\n    return bool(m and m.role >= 9)\n
"},{"location":"reference/launderette/views/#launderette.views.Machine","title":"Machine","text":"

Bases: Model

"},{"location":"reference/launderette/views/#launderette.views.Machine.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in launderette/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    launderette_club = Club.objects.filter(\n        unix_name=settings.SITH_LAUNDERETTE_MANAGER[\"unix_name\"]\n    ).first()\n    m = launderette_club.get_membership_for(user)\n    return bool(m and m.role >= 9)\n
"},{"location":"reference/launderette/views/#launderette.views.Slot","title":"Slot","text":"

Bases: Model

"},{"location":"reference/launderette/views/#launderette.views.Token","title":"Token","text":"

Bases: Model

"},{"location":"reference/launderette/views/#launderette.views.Token.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in launderette/models.py
def is_owned_by(self, user):\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    launderette_club = Club.objects.filter(\n        unix_name=settings.SITH_LAUNDERETTE_MANAGER[\"unix_name\"]\n    ).first()\n    m = launderette_club.get_membership_for(user)\n    return bool(m and m.role >= 9)\n
"},{"location":"reference/launderette/views/#launderette.views.LaunderetteMainView","title":"LaunderetteMainView","text":"

Bases: TemplateView

Main presentation view.

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteMainView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add page to the context.

Source code in launderette/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add page to the context.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"page\"] = Page.objects.filter(name=\"launderette\").first()\n    return kwargs\n
"},{"location":"reference/launderette/views/#launderette.views.LaunderetteBookMainView","title":"LaunderetteBookMainView","text":"

Bases: CanViewMixin, ListView

Choose which launderette to book.

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteBookView","title":"LaunderetteBookView","text":"

Bases: CanViewMixin, DetailView

Display the launderette schedule.

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteBookView.get_context_data","title":"get_context_data(**kwargs)","text":"

Add page to the context.

Source code in launderette/views.py
def get_context_data(self, **kwargs):\n    \"\"\"Add page to the context.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"planning\"] = OrderedDict()\n    kwargs[\"slot_type\"] = self.slot_type\n    start_date = datetime.now().replace(\n        hour=0, minute=0, second=0, microsecond=0, tzinfo=tz.utc\n    )\n    for date in LaunderetteBookView.date_iterator(\n        start_date, start_date + timedelta(days=6), timedelta(days=1)\n    ):\n        kwargs[\"planning\"][date] = []\n        for h in LaunderetteBookView.date_iterator(\n            date, date + timedelta(days=1), timedelta(hours=1)\n        ):\n            free = False\n            if (\n                (\n                    self.slot_type == \"BOTH\"\n                    and self.check_slot(\"WASHING\", h)\n                    and self.check_slot(\"DRYING\", h + timedelta(hours=1))\n                )\n                or (self.slot_type == \"WASHING\" and self.check_slot(\"WASHING\", h))\n                or (self.slot_type == \"DRYING\" and self.check_slot(\"DRYING\", h))\n            ):\n                free = True\n            if free and datetime.now().replace(tzinfo=tz.utc) < h:\n                kwargs[\"planning\"][date].append(h)\n            else:\n                kwargs[\"planning\"][date].append(None)\n    return kwargs\n
"},{"location":"reference/launderette/views/#launderette.views.SlotDeleteView","title":"SlotDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

Delete a slot.

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteListView","title":"LaunderetteListView","text":"

Bases: CanEditPropMixin, ListView

Choose which launderette to administer.

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteEditView","title":"LaunderetteEditView","text":"

Bases: CanEditPropMixin, UpdateView

Edit a launderette.

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteCreateView","title":"LaunderetteCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

Create a new launderette.

"},{"location":"reference/launderette/views/#launderette.views.ManageTokenForm","title":"ManageTokenForm","text":"

Bases: Form

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteAdminView","title":"LaunderetteAdminView","text":"

Bases: CanEditPropMixin, BaseFormView, DetailView

The admin page of the launderette.

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteAdminView.form_valid","title":"form_valid(form)","text":"

We handle here the redirection, passing the user id of the asked customer.

Source code in launderette/views.py
def form_valid(self, form):\n    \"\"\"We handle here the redirection, passing the user id of the asked customer.\"\"\"\n    form.process(self.object)\n    if form.is_valid():\n        return super().form_valid(form)\n    else:\n        return super().form_invalid(form)\n
"},{"location":"reference/launderette/views/#launderette.views.LaunderetteAdminView.get_context_data","title":"get_context_data(**kwargs)","text":"

We handle here the login form for the barman.

Source code in launderette/views.py
def get_context_data(self, **kwargs):\n    \"\"\"We handle here the login form for the barman.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    if self.request.method == \"GET\":\n        kwargs[\"form\"] = self.get_form()\n    return kwargs\n
"},{"location":"reference/launderette/views/#launderette.views.GetLaunderetteUserForm","title":"GetLaunderetteUserForm","text":"

Bases: GetUserForm

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteMainClickView","title":"LaunderetteMainClickView","text":"

Bases: CanEditMixin, BaseFormView, DetailView

The click page of the launderette.

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteMainClickView.form_valid","title":"form_valid(form)","text":"

We handle here the redirection, passing the user id of the asked customer.

Source code in launderette/views.py
def form_valid(self, form):\n    \"\"\"We handle here the redirection, passing the user id of the asked customer.\"\"\"\n    self.kwargs[\"user_id\"] = form.cleaned_data[\"user_id\"]\n    return super().form_valid(form)\n
"},{"location":"reference/launderette/views/#launderette.views.LaunderetteMainClickView.get_context_data","title":"get_context_data(**kwargs)","text":"

We handle here the login form for the barman.

Source code in launderette/views.py
def get_context_data(self, **kwargs):\n    \"\"\"We handle here the login form for the barman.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"counter\"] = self.object.counter\n    kwargs[\"form\"] = self.get_form()\n    kwargs[\"barmen\"] = [self.request.user]\n    if \"last_basket\" in self.request.session:\n        kwargs[\"last_basket\"] = self.request.session.pop(\"last_basket\", None)\n        kwargs[\"last_customer\"] = self.request.session.pop(\"last_customer\", None)\n        kwargs[\"last_total\"] = self.request.session.pop(\"last_total\", None)\n        kwargs[\"new_customer_amount\"] = self.request.session.pop(\n            \"new_customer_amount\", None\n        )\n    return kwargs\n
"},{"location":"reference/launderette/views/#launderette.views.ClickTokenForm","title":"ClickTokenForm","text":"

Bases: BaseForm

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteClickView","title":"LaunderetteClickView","text":"

Bases: CanEditMixin, DetailView, BaseFormView

The click page of the launderette.

"},{"location":"reference/launderette/views/#launderette.views.LaunderetteClickView.get","title":"get(request, *args, **kwargs)","text":"

Simple get view.

Source code in launderette/views.py
def get(self, request, *args, **kwargs):\n    \"\"\"Simple get view.\"\"\"\n    self.customer = Customer.objects.filter(user__id=self.kwargs[\"user_id\"]).first()\n    self.subscriber = self.customer.user\n    self.operator = request.user\n    return super().get(request, *args, **kwargs)\n
"},{"location":"reference/launderette/views/#launderette.views.LaunderetteClickView.post","title":"post(request, *args, **kwargs)","text":"

Handle the many possibilities of the post request.

Source code in launderette/views.py
def post(self, request, *args, **kwargs):\n    \"\"\"Handle the many possibilities of the post request.\"\"\"\n    self.object = self.get_object()\n    self.customer = Customer.objects.filter(user__id=self.kwargs[\"user_id\"]).first()\n    self.subscriber = self.customer.user\n    self.operator = request.user\n    return super().post(request, *args, **kwargs)\n
"},{"location":"reference/launderette/views/#launderette.views.LaunderetteClickView.form_valid","title":"form_valid(form)","text":"

We handle here the redirection, passing the user id of the asked customer.

Source code in launderette/views.py
def form_valid(self, form):\n    \"\"\"We handle here the redirection, passing the user id of the asked customer.\"\"\"\n    self.request.session.update(form.last_basket)\n    return super().form_valid(form)\n
"},{"location":"reference/launderette/views/#launderette.views.LaunderetteClickView.get_context_data","title":"get_context_data(**kwargs)","text":"

We handle here the login form for the barman.

Source code in launderette/views.py
def get_context_data(self, **kwargs):\n    \"\"\"We handle here the login form for the barman.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    if \"form\" not in kwargs:\n        kwargs[\"form\"] = self.get_form()\n    kwargs[\"counter\"] = self.object.counter\n    kwargs[\"customer\"] = self.customer\n    return kwargs\n
"},{"location":"reference/launderette/views/#launderette.views.MachineEditView","title":"MachineEditView","text":"

Bases: CanEditPropMixin, UpdateView

Edit a machine.

"},{"location":"reference/launderette/views/#launderette.views.MachineDeleteView","title":"MachineDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

Edit a machine.

"},{"location":"reference/launderette/views/#launderette.views.MachineCreateView","title":"MachineCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

Create a new machine.

"},{"location":"reference/matmat/models/","title":"Models","text":""},{"location":"reference/matmat/views/","title":"Views","text":""},{"location":"reference/matmat/views/#matmat.views.FormerSubscriberMixin","title":"FormerSubscriberMixin","text":"

Bases: AccessMixin

Check if the user was at least an old subscriber.

Raises:

Type Description PermissionDenied

if the user never subscribed.

"},{"location":"reference/matmat/views/#matmat.views.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/matmat/views/#matmat.views.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/matmat/views/#matmat.views.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/matmat/views/#matmat.views.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/matmat/views/#matmat.views.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/matmat/views/#matmat.views.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/matmat/views/#matmat.views.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/matmat/views/#matmat.views.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/matmat/views/#matmat.views.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/matmat/views/#matmat.views.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/matmat/views/#matmat.views.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/matmat/views/#matmat.views.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/matmat/views/#matmat.views.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/matmat/views/#matmat.views.SearchType","title":"SearchType","text":"

Bases: Enum

"},{"location":"reference/matmat/views/#matmat.views.SearchForm","title":"SearchForm(*args, **kwargs)","text":"

Bases: ModelForm

Source code in matmat/views.py
def __init__(self, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n    for key in self.fields:\n        self.fields[key].required = False\n
"},{"location":"reference/matmat/views/#matmat.views.SearchFormListView","title":"SearchFormListView","text":"

Bases: FormerSubscriberMixin, SingleObjectMixin, ListView

"},{"location":"reference/matmat/views/#matmat.views.SearchFormView","title":"SearchFormView","text":"

Bases: FormerSubscriberMixin, FormView

Allows users to search inside the user list.

"},{"location":"reference/matmat/views/#matmat.views.SearchNormalFormView","title":"SearchNormalFormView","text":"

Bases: SearchFormView

"},{"location":"reference/matmat/views/#matmat.views.SearchReverseFormView","title":"SearchReverseFormView","text":"

Bases: SearchFormView

"},{"location":"reference/matmat/views/#matmat.views.SearchQuickFormView","title":"SearchQuickFormView","text":"

Bases: SearchFormView

"},{"location":"reference/matmat/views/#matmat.views.SearchClearFormView","title":"SearchClearFormView","text":"

Bases: FormerSubscriberMixin, View

Clear SearchFormView and redirect to it.

"},{"location":"reference/matmat/views/#matmat.views.search_user","title":"search_user(query)","text":"Source code in core/views/site.py
def search_user(query):\n    try:\n        # slugify turns everything into ascii and every whitespace into -\n        # it ends by removing duplicate - (so ' - ' will turn into '-')\n        # replace('-', ' ') because search is whitespace based\n        query = slugify(query).replace(\"-\", \" \")\n        # TODO: is this necessary?\n        query = html.escape(query)\n        res = (\n            SearchQuerySet()\n            .models(User)\n            .autocomplete(auto=query)\n            .order_by(\"-last_login\")\n            .load_all()[:20]\n        )\n        return [r.object for r in res]\n    except TypeError:\n        return []\n
"},{"location":"reference/pedagogy/models/","title":"Models","text":""},{"location":"reference/pedagogy/models/#pedagogy.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/pedagogy/models/#pedagogy.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/pedagogy/models/#pedagogy.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.UV","title":"UV","text":"

Bases: Model

Contains infos about an UV (course).

"},{"location":"reference/pedagogy/models/#pedagogy.models.UV.is_owned_by","title":"is_owned_by(user)","text":"

Can be created by superuser, root or pedagogy admin user.

Source code in pedagogy/models.py
def is_owned_by(self, user):\n    \"\"\"Can be created by superuser, root or pedagogy admin user.\"\"\"\n    return user.is_in_group(pk=settings.SITH_GROUP_PEDAGOGY_ADMIN_ID)\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.UV.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Only visible by subscribers.

Source code in pedagogy/models.py
def can_be_viewed_by(self, user):\n    \"\"\"Only visible by subscribers.\"\"\"\n    return user.is_subscribed\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.UV.has_user_already_commented","title":"has_user_already_commented(user)","text":"

Help prevent multiples comments from the same user.

This function checks that no other comment has been posted by a specified user.

Returns:

Type Description bool

True if the user has already posted a comment on this UV, else False.

Source code in pedagogy/models.py
def has_user_already_commented(self, user: User) -> bool:\n    \"\"\"Help prevent multiples comments from the same user.\n\n    This function checks that no other comment has been posted by a specified user.\n\n    Returns:\n        True if the user has already posted a comment on this UV, else False.\n    \"\"\"\n    return self.comments.filter(author=user).exists()\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.UVComment","title":"UVComment","text":"

Bases: Model

A comment about an UV.

"},{"location":"reference/pedagogy/models/#pedagogy.models.UVComment.is_owned_by","title":"is_owned_by(user)","text":"

Is owned by a pedagogy admin, a superuser or the author himself.

Source code in pedagogy/models.py
def is_owned_by(self, user):\n    \"\"\"Is owned by a pedagogy admin, a superuser or the author himself.\"\"\"\n    return self.author == user or user.is_owner(self.uv)\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.UVComment.is_reported","title":"is_reported()","text":"

Return True if someone reported this UV.

Source code in pedagogy/models.py
@cached_property\ndef is_reported(self):\n    \"\"\"Return True if someone reported this UV.\"\"\"\n    return self.reports.exists()\n
"},{"location":"reference/pedagogy/models/#pedagogy.models.UVResult","title":"UVResult","text":"

Bases: Model

Results got to an UV.

Views will be implemented after the first release Will list every UV done by an user Linked to user uv Contains a grade settings.SITH_PEDAGOGY_UV_RESULT_GRADE a semester (P/A)20xx.

"},{"location":"reference/pedagogy/models/#pedagogy.models.UVCommentReport","title":"UVCommentReport","text":"

Bases: Model

Report an inapropriate comment.

"},{"location":"reference/pedagogy/models/#pedagogy.models.UVCommentReport.is_owned_by","title":"is_owned_by(user)","text":"

Can be created by a pedagogy admin, a superuser or a subscriber.

Source code in pedagogy/models.py
def is_owned_by(self, user):\n    \"\"\"Can be created by a pedagogy admin, a superuser or a subscriber.\"\"\"\n    return user.is_subscribed or user.is_owner(self.comment.uv)\n
"},{"location":"reference/pedagogy/schemas/","title":"Schemas","text":""},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.ShortUvList","title":"ShortUvList = TypeAdapter(list[UtbmShortUvSchema]) module-attribute","text":""},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UV","title":"UV","text":"

Bases: Model

Contains infos about an UV (course).

"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UV.is_owned_by","title":"is_owned_by(user)","text":"

Can be created by superuser, root or pedagogy admin user.

Source code in pedagogy/models.py
def is_owned_by(self, user):\n    \"\"\"Can be created by superuser, root or pedagogy admin user.\"\"\"\n    return user.is_in_group(pk=settings.SITH_GROUP_PEDAGOGY_ADMIN_ID)\n
"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UV.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Only visible by subscribers.

Source code in pedagogy/models.py
def can_be_viewed_by(self, user):\n    \"\"\"Only visible by subscribers.\"\"\"\n    return user.is_subscribed\n
"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UV.has_user_already_commented","title":"has_user_already_commented(user)","text":"

Help prevent multiples comments from the same user.

This function checks that no other comment has been posted by a specified user.

Returns:

Type Description bool

True if the user has already posted a comment on this UV, else False.

Source code in pedagogy/models.py
def has_user_already_commented(self, user: User) -> bool:\n    \"\"\"Help prevent multiples comments from the same user.\n\n    This function checks that no other comment has been posted by a specified user.\n\n    Returns:\n        True if the user has already posted a comment on this UV, else False.\n    \"\"\"\n    return self.comments.filter(author=user).exists()\n
"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UtbmShortUvSchema","title":"UtbmShortUvSchema","text":"

Bases: Schema

Short representation of an UV in the UTBM API.

Notes

This schema holds only the fields we actually need. The UTBM API returns more data than that.

"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.WorkloadSchema","title":"WorkloadSchema","text":"

Bases: Schema

"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.SemesterUvState","title":"SemesterUvState","text":"

Bases: Schema

The state of the UV during either autumn or spring semester

"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UtbmFullUvSchema","title":"UtbmFullUvSchema","text":"

Bases: Schema

Long representation of an UV in the UTBM API.

"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.SimpleUvSchema","title":"SimpleUvSchema","text":"

Bases: ModelSchema

Our minimal representation of an UV.

"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UvSchema","title":"UvSchema","text":"

Bases: ModelSchema

Our complete representation of an UV

"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UvFilterSchema","title":"UvFilterSchema","text":"

Bases: FilterSchema

"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UvFilterSchema.filter_search","title":"filter_search(value)","text":"

Special filter for the search text.

It does a full text search if available.

Source code in pedagogy/schemas.py
def filter_search(self, value: str | None) -> Q:\n    \"\"\"Special filter for the search text.\n\n    It does a full text search if available.\n    \"\"\"\n    if not value:\n        return Q()\n\n    if len(value) < 3 or (len(value) < 5 and any(c.isdigit() for c in value)):\n        # Likely to be an UV code\n        return Q(code__istartswith=value)\n\n    qs = list(\n        SearchQuerySet()\n        .models(UV)\n        .autocomplete(auto=html.escape(value))\n        .values_list(\"pk\", flat=True)\n    )\n\n    return Q(id__in=qs)\n
"},{"location":"reference/pedagogy/schemas/#pedagogy.schemas.UvFilterSchema.filter_semester","title":"filter_semester(value)","text":"

Special filter for the semester.

If either \"SPRING\" or \"AUTUMN\" is given, UV that are available during \"AUTUMN_AND_SPRING\" will be filtered.

Source code in pedagogy/schemas.py
def filter_semester(self, value: set[str] | None) -> Q:\n    \"\"\"Special filter for the semester.\n\n    If either \"SPRING\" or \"AUTUMN\" is given, UV that are available\n    during \"AUTUMN_AND_SPRING\" will be filtered.\n    \"\"\"\n    if not value:\n        return Q()\n    value.add(\"AUTUMN_AND_SPRING\")\n    return Q(semester__in=value)\n
"},{"location":"reference/pedagogy/views/","title":"Views","text":""},{"location":"reference/pedagogy/views/#pedagogy.views.CanEditPropMixin","title":"CanEditPropMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has owner permissions on the child view object.

In other word, you can make a view with this view as parent, and it will be retricted to the users that are in the object's owner_group or that pass the obj.can_be_viewed_by test.

Raises:

Type Description PermissionDenied

If the user cannot see the object

"},{"location":"reference/pedagogy/views/#pedagogy.views.CanViewMixin","title":"CanViewMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to view this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/pedagogy/views/#pedagogy.views.FormerSubscriberMixin","title":"FormerSubscriberMixin","text":"

Bases: AccessMixin

Check if the user was at least an old subscriber.

Raises:

Type Description PermissionDenied

if the user never subscribed.

"},{"location":"reference/pedagogy/views/#pedagogy.views.Notification","title":"Notification","text":"

Bases: Model

"},{"location":"reference/pedagogy/views/#pedagogy.views.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/pedagogy/views/#pedagogy.views.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/pedagogy/views/#pedagogy.views.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.DetailFormView","title":"DetailFormView","text":"

Bases: SingleObjectMixin, FormView

Class that allow both a detail view and a form view.

"},{"location":"reference/pedagogy/views/#pedagogy.views.DetailFormView.get_object","title":"get_object()","text":"

Get current group from id in url.

Source code in core/views/__init__.py
def get_object(self):\n    \"\"\"Get current group from id in url.\"\"\"\n    return self.cached_object\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.DetailFormView.cached_object","title":"cached_object()","text":"

Optimisation on group retrieval.

Source code in core/views/__init__.py
@cached_property\ndef cached_object(self):\n    \"\"\"Optimisation on group retrieval.\"\"\"\n    return super().get_object()\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UVCommentForm","title":"UVCommentForm(author_id, uv_id, is_creation, *args, **kwargs)","text":"

Bases: ModelForm

Form handeling creation and edit of an UVComment.

Source code in pedagogy/forms.py
def __init__(self, author_id, uv_id, is_creation, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n    self.fields[\"author\"].queryset = User.objects.filter(id=author_id).all()\n    self.fields[\"author\"].initial = author_id\n    self.fields[\"uv\"].queryset = UV.objects.filter(id=uv_id).all()\n    self.fields[\"uv\"].initial = uv_id\n    self.is_creation = is_creation\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UVCommentModerationForm","title":"UVCommentModerationForm","text":"

Bases: Form

Form handeling bulk comment deletion.

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVCommentReportForm","title":"UVCommentReportForm(reporter_id, comment_id, *args, **kwargs)","text":"

Bases: ModelForm

Form handeling creation and edit of an UVReport.

Source code in pedagogy/forms.py
def __init__(self, reporter_id, comment_id, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n    self.fields[\"reporter\"].queryset = User.objects.filter(id=reporter_id).all()\n    self.fields[\"reporter\"].initial = reporter_id\n    self.fields[\"comment\"].queryset = UVComment.objects.filter(id=comment_id).all()\n    self.fields[\"comment\"].initial = comment_id\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UVForm","title":"UVForm(author_id, *args, **kwargs)","text":"

Bases: ModelForm

Form handeling creation and edit of an UV.

Source code in pedagogy/forms.py
def __init__(self, author_id, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n    self.fields[\"author\"].queryset = User.objects.filter(id=author_id).all()\n    self.fields[\"author\"].initial = author_id\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UV","title":"UV","text":"

Bases: Model

Contains infos about an UV (course).

"},{"location":"reference/pedagogy/views/#pedagogy.views.UV.is_owned_by","title":"is_owned_by(user)","text":"

Can be created by superuser, root or pedagogy admin user.

Source code in pedagogy/models.py
def is_owned_by(self, user):\n    \"\"\"Can be created by superuser, root or pedagogy admin user.\"\"\"\n    return user.is_in_group(pk=settings.SITH_GROUP_PEDAGOGY_ADMIN_ID)\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UV.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Only visible by subscribers.

Source code in pedagogy/models.py
def can_be_viewed_by(self, user):\n    \"\"\"Only visible by subscribers.\"\"\"\n    return user.is_subscribed\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UV.has_user_already_commented","title":"has_user_already_commented(user)","text":"

Help prevent multiples comments from the same user.

This function checks that no other comment has been posted by a specified user.

Returns:

Type Description bool

True if the user has already posted a comment on this UV, else False.

Source code in pedagogy/models.py
def has_user_already_commented(self, user: User) -> bool:\n    \"\"\"Help prevent multiples comments from the same user.\n\n    This function checks that no other comment has been posted by a specified user.\n\n    Returns:\n        True if the user has already posted a comment on this UV, else False.\n    \"\"\"\n    return self.comments.filter(author=user).exists()\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UVComment","title":"UVComment","text":"

Bases: Model

A comment about an UV.

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVComment.is_owned_by","title":"is_owned_by(user)","text":"

Is owned by a pedagogy admin, a superuser or the author himself.

Source code in pedagogy/models.py
def is_owned_by(self, user):\n    \"\"\"Is owned by a pedagogy admin, a superuser or the author himself.\"\"\"\n    return self.author == user or user.is_owner(self.uv)\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UVComment.is_reported","title":"is_reported()","text":"

Return True if someone reported this UV.

Source code in pedagogy/models.py
@cached_property\ndef is_reported(self):\n    \"\"\"Return True if someone reported this UV.\"\"\"\n    return self.reports.exists()\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UVCommentReport","title":"UVCommentReport","text":"

Bases: Model

Report an inapropriate comment.

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVCommentReport.is_owned_by","title":"is_owned_by(user)","text":"

Can be created by a pedagogy admin, a superuser or a subscriber.

Source code in pedagogy/models.py
def is_owned_by(self, user):\n    \"\"\"Can be created by a pedagogy admin, a superuser or a subscriber.\"\"\"\n    return user.is_subscribed or user.is_owner(self.comment.uv)\n
"},{"location":"reference/pedagogy/views/#pedagogy.views.UVDetailFormView","title":"UVDetailFormView","text":"

Bases: CanViewMixin, DetailFormView

Display every comment of an UV and detailed infos about it.

Allow to comment the UV.

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVCommentUpdateView","title":"UVCommentUpdateView","text":"

Bases: CanEditPropMixin, UpdateView

Allow edit of a given comment.

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVCommentDeleteView","title":"UVCommentDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

Allow delete of a given comment.

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVGuideView","title":"UVGuideView","text":"

Bases: LoginRequiredMixin, FormerSubscriberMixin, TemplateView

UV guide main page.

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVCommentReportCreateView","title":"UVCommentReportCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

Create a new report for an inapropriate comment.

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVModerationFormView","title":"UVModerationFormView","text":"

Bases: FormView

Moderation interface (Privileged).

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVCreateView","title":"UVCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

Add a new UV (Privileged).

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVDeleteView","title":"UVDeleteView","text":"

Bases: CanEditPropMixin, DeleteView

Allow to delete an UV (Privileged).

"},{"location":"reference/pedagogy/views/#pedagogy.views.UVUpdateView","title":"UVUpdateView","text":"

Bases: CanEditPropMixin, UpdateView

Allow to edit an UV (Privilegied).

"},{"location":"reference/rootplace/forms/","title":"Forms","text":""},{"location":"reference/rootplace/forms/#rootplace.forms.MergeForm","title":"MergeForm","text":"

Bases: Form

"},{"location":"reference/rootplace/forms/#rootplace.forms.SelectUserForm","title":"SelectUserForm","text":"

Bases: Form

"},{"location":"reference/rootplace/forms/#rootplace.forms.BanForm","title":"BanForm","text":"

Bases: ModelForm

Form to ban a user.

"},{"location":"reference/rootplace/models/","title":"Models","text":""},{"location":"reference/rootplace/views/","title":"Views","text":""},{"location":"reference/rootplace/views/#rootplace.views.MergeUsersView","title":"MergeUsersView","text":"

Bases: FormView

"},{"location":"reference/rootplace/views/#rootplace.views.DeleteAllForumUserMessagesView","title":"DeleteAllForumUserMessagesView","text":"

Bases: FormView

Delete all forum messages from an user.

Messages are soft deleted and are still visible from admins GUI frontend to the dedicated command.

"},{"location":"reference/rootplace/views/#rootplace.views.OperationLogListView","title":"OperationLogListView","text":"

Bases: ListView, CanEditPropMixin

List all logs.

"},{"location":"reference/rootplace/views/#rootplace.views.BanView","title":"BanView","text":"

Bases: PermissionRequiredMixin, ListView

UserBan management view.

Displays :

"},{"location":"reference/rootplace/views/#rootplace.views.BanCreateView","title":"BanCreateView","text":"

Bases: PermissionRequiredMixin, CreateView

UserBan creation view.

"},{"location":"reference/rootplace/views/#rootplace.views.BanDeleteView","title":"BanDeleteView","text":"

Bases: PermissionRequiredMixin, DeleteView

UserBan deletion view.

"},{"location":"reference/rootplace/views/#rootplace.views.merge_users","title":"merge_users(u1, u2)","text":"

Merge u2 into u1.

This means that u1 shall receive everything that belonged to u2 :

- pictures\n- refills of the sith account\n- purchases of any item bought on the eboutic or the counters\n- subscriptions\n- godfathers\n- godchildren\n

If u1 had no account id, he shall receive the one of u2. If u1 and u2 were both in the middle of a subscription, the remaining durations stack If u1 had no profile picture, he shall receive the one of u2

Source code in rootplace/views.py
def merge_users(u1: User, u2: User) -> User:\n    \"\"\"Merge u2 into u1.\n\n    This means that u1 shall receive everything that belonged to u2 :\n\n        - pictures\n        - refills of the sith account\n        - purchases of any item bought on the eboutic or the counters\n        - subscriptions\n        - godfathers\n        - godchildren\n\n    If u1 had no account id, he shall receive the one of u2.\n    If u1 and u2 were both in the middle of a subscription, the remaining\n    durations stack\n    If u1 had no profile picture, he shall receive the one of u2\n    \"\"\"\n    for field in u1._meta.fields:\n        if not field.is_relation and not u1.__dict__[field.name]:\n            u1.__dict__[field.name] = u2.__dict__[field.name]\n    for group in u2.groups.all():\n        u1.groups.add(group.id)\n    for godfather in u2.godfathers.exclude(id=u1.id):\n        u1.godfathers.add(godfather)\n    for godchild in u2.godchildren.exclude(id=u1.id):\n        u1.godchildren.add(godchild)\n    __merge_subscriptions(u1, u2)\n    __merge_pictures(u1, u2)\n    u2.invoices.all().update(user=u1)\n    c_src = Customer.objects.filter(user=u2).first()\n    if c_src is not None:\n        c_dest, created = Customer.get_or_create(u1)\n        c_src.refillings.update(customer=c_dest)\n        c_src.buyings.update(customer=c_dest)\n        Customer.objects.filter(pk=c_dest.pk).update_amount()\n        if created:\n            # swap the account numbers, so that the user keep\n            # the id he is accustomed to\n            tmp_id = c_src.account_id\n            # delete beforehand in order not to have a unique constraint violation\n            c_src.delete()\n            c_dest.account_id = tmp_id\n    u1.save()\n    u2.delete()  # everything remaining in u2 gets deleted thanks to on_delete=CASCADE\n    return u1\n
"},{"location":"reference/rootplace/views/#rootplace.views.delete_all_forum_user_messages","title":"delete_all_forum_user_messages(user, moderator, *, verbose=False)","text":"

Soft delete all messages of a user.

Parameters:

Name Type Description Default user User

core.models.User the user to delete messages from

required moderator User

core.models.User the one marked as the moderator.

required verbose bool

bool if True, print the deleted messages

False Source code in rootplace/views.py
def delete_all_forum_user_messages(\n    user: User, moderator: User, *, verbose: bool = False\n):\n    \"\"\"Soft delete all messages of a user.\n\n    Args:\n        user: core.models.User the user to delete messages from\n        moderator: core.models.User the one marked as the moderator.\n        verbose: bool if True, print the deleted messages\n    \"\"\"\n    for message in user.forum_messages.all():\n        if message.is_deleted():\n            continue\n\n        if verbose:\n            logging.getLogger(\"django\").info(message)\n        ForumMessageMeta(message=message, user=moderator, action=\"DELETE\").save()\n
"},{"location":"reference/sas/models/","title":"Models","text":""},{"location":"reference/sas/models/#sas.models.SithFile","title":"SithFile","text":"

Bases: Model

"},{"location":"reference/sas/models/#sas.models.SithFile.clean","title":"clean()","text":"

Cleans up the file.

Source code in core/models.py
def clean(self):\n    \"\"\"Cleans up the file.\"\"\"\n    super().clean()\n    if \"/\" in self.name:\n        raise ValidationError(_(\"Character '/' not authorized in name\"))\n    if self == self.parent:\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self == self.parent or (\n        self.parent is not None and self in self.get_parent_list()\n    ):\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self.parent and self.parent.is_file:\n        raise ValidationError(\n            _(\"You can not make a file be a children of a non folder file\")\n        )\n    if (\n        self.parent is None\n        and SithFile.objects.exclude(id=self.id)\n        .filter(parent=None, name=self.name)\n        .exists()\n    ) or (\n        self.parent\n        and self.parent.children.exclude(id=self.id).filter(name=self.name).exists()\n    ):\n        raise ValidationError(_(\"Duplicate file\"), code=\"duplicate\")\n    if self.is_folder:\n        if self.file:\n            try:\n                Image.open(BytesIO(self.file.read()))\n            except Image.UnidentifiedImageError as e:\n                raise ValidationError(\n                    _(\"This is not a valid folder thumbnail\")\n                ) from e\n        self.mime_type = \"inode/directory\"\n    if self.is_file and (self.file is None or self.file == \"\"):\n        raise ValidationError(_(\"You must provide a file\"))\n
"},{"location":"reference/sas/models/#sas.models.SithFile.apply_rights_recursively","title":"apply_rights_recursively(*, only_folders=False)","text":"

Apply the rights of this file to all children recursively.

Parameters:

Name Type Description Default only_folders bool

If True, only apply the rights to SithFiles that are folders.

False Source code in core/models.py
def apply_rights_recursively(self, *, only_folders: bool = False) -> None:\n    \"\"\"Apply the rights of this file to all children recursively.\n\n    Args:\n        only_folders: If True, only apply the rights to SithFiles that are folders.\n    \"\"\"\n    file_ids = []\n    explored_ids = [self.id]\n    while len(explored_ids) > 0:  # find all children recursively\n        file_ids.extend(explored_ids)\n        next_level = SithFile.objects.filter(parent_id__in=explored_ids)\n        if only_folders:\n            next_level = next_level.filter(is_folder=True)\n        explored_ids = list(next_level.values_list(\"id\", flat=True))\n    for through in (SithFile.view_groups.through, SithFile.edit_groups.through):\n        # force evaluation. Without this, the iterator yields nothing\n        groups = list(\n            through.objects.filter(sithfile_id=self.id).values_list(\n                \"group_id\", flat=True\n            )\n        )\n        # delete previous rights\n        through.objects.filter(sithfile_id__in=file_ids).delete()\n        through.objects.bulk_create(  # create new rights\n            [through(sithfile_id=f, group_id=g) for f in file_ids for g in groups]\n        )\n
"},{"location":"reference/sas/models/#sas.models.SithFile.copy_rights","title":"copy_rights()","text":"

Copy, if possible, the rights of the parent folder.

Source code in core/models.py
def copy_rights(self):\n    \"\"\"Copy, if possible, the rights of the parent folder.\"\"\"\n    if self.parent is not None:\n        self.edit_groups.set(self.parent.edit_groups.all())\n        self.view_groups.set(self.parent.view_groups.all())\n
"},{"location":"reference/sas/models/#sas.models.SithFile.move_to","title":"move_to(parent)","text":"

Move a file to a new parent. parent must be a SithFile with the is_folder=True property. Otherwise, this function doesn't change anything. This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify SithFiles recursively, so it stays efficient even with top-level folders.

Source code in core/models.py
def move_to(self, parent):\n    \"\"\"Move a file to a new parent.\n    `parent` must be a SithFile with the `is_folder=True` property. Otherwise, this function doesn't change\n    anything.\n    This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify\n    SithFiles recursively, so it stays efficient even with top-level folders.\n    \"\"\"\n    if not parent.is_folder:\n        return\n    self.parent = parent\n    self.clean()\n    self.save()\n
"},{"location":"reference/sas/models/#sas.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/sas/models/#sas.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/sas/models/#sas.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/sas/models/#sas.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/sas/models/#sas.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/sas/models/#sas.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/sas/models/#sas.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/sas/models/#sas.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/sas/models/#sas.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/sas/models/#sas.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/sas/models/#sas.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/sas/models/#sas.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/sas/models/#sas.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/sas/models/#sas.models.SasFile","title":"SasFile","text":"

Bases: SithFile

Proxy model for any file in the SAS.

May be used to have logic that should be shared by both Picture and Album.

"},{"location":"reference/sas/models/#sas.models.PictureQuerySet","title":"PictureQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/sas/models/#sas.models.PictureQuerySet.viewable_by","title":"viewable_by(user)","text":"

Filter the pictures that this user can view.

Warning

Calling this queryset method may add several additional requests.

Source code in sas/models.py
def viewable_by(self, user: User) -> Self:\n    \"\"\"Filter the pictures that this user can view.\n\n    Warning:\n        Calling this queryset method may add several additional requests.\n    \"\"\"\n    if user.is_root or user.is_in_group(pk=settings.SITH_GROUP_SAS_ADMIN_ID):\n        return self.all()\n    if user.was_subscribed:\n        return self.filter(is_moderated=True)\n    return self.filter(people__user_id=user.id, is_moderated=True)\n
"},{"location":"reference/sas/models/#sas.models.SASPictureManager","title":"SASPictureManager","text":"

Bases: Manager

"},{"location":"reference/sas/models/#sas.models.Picture","title":"Picture","text":"

Bases: SasFile

"},{"location":"reference/sas/models/#sas.models.AlbumQuerySet","title":"AlbumQuerySet","text":"

Bases: QuerySet

"},{"location":"reference/sas/models/#sas.models.AlbumQuerySet.viewable_by","title":"viewable_by(user)","text":"

Filter the albums that this user can view.

Warning

Calling this queryset method may add several additional requests.

Source code in sas/models.py
def viewable_by(self, user: User) -> Self:\n    \"\"\"Filter the albums that this user can view.\n\n    Warning:\n        Calling this queryset method may add several additional requests.\n    \"\"\"\n    if user.is_root or user.is_in_group(pk=settings.SITH_GROUP_SAS_ADMIN_ID):\n        return self.all()\n    if user.was_subscribed:\n        return self.filter(is_moderated=True)\n    # known bug : if all children of an album are also albums\n    # then this album is excluded, even if one of the sub-albums should be visible.\n    # The fs-like navigation is likely to be half-broken for non-subscribers,\n    # but that's ok, since non-subscribers are expected to see only the albums\n    # containing pictures on which they have been identified (hence, very few).\n    # Most, if not all, of their albums will be displayed on the\n    # `latest albums` section of the SAS.\n    # Moreover, they will still see all of their picture in their profile.\n    return self.filter(\n        Exists(Picture.objects.filter(parent_id=OuterRef(\"pk\")).viewable_by(user))\n    )\n
"},{"location":"reference/sas/models/#sas.models.SASAlbumManager","title":"SASAlbumManager","text":"

Bases: Manager

"},{"location":"reference/sas/models/#sas.models.Album","title":"Album","text":"

Bases: SasFile

"},{"location":"reference/sas/models/#sas.models.Album.NAME_MAX_LENGTH","title":"NAME_MAX_LENGTH = 50 class-attribute","text":"

Maximum length of an album's name.

SithFile have a maximum length of 256 characters. However, this limit is too high for albums. Names longer than 50 characters are harder to read and harder to display on the SAS page.

It is to be noted, though, that this does not add or modify any db behaviour. It's just a constant to be used in views and forms.

"},{"location":"reference/sas/models/#sas.models.PeoplePictureRelation","title":"PeoplePictureRelation","text":"

Bases: Model

The PeoplePictureRelation class makes the connection between User and Picture.

"},{"location":"reference/sas/models/#sas.models.PictureModerationRequest","title":"PictureModerationRequest","text":"

Bases: Model

A request to remove a Picture from the SAS.

"},{"location":"reference/sas/models/#sas.models.exif_auto_rotate","title":"exif_auto_rotate(image)","text":"Source code in core/utils.py
def exif_auto_rotate(image):\n    for orientation in ExifTags.TAGS:\n        if ExifTags.TAGS[orientation] == \"Orientation\":\n            break\n    exif = dict(image._getexif().items())\n\n    if exif[orientation] == 3:\n        image = image.rotate(180, expand=True)\n    elif exif[orientation] == 6:\n        image = image.rotate(270, expand=True)\n    elif exif[orientation] == 8:\n        image = image.rotate(90, expand=True)\n\n    return image\n
"},{"location":"reference/sas/models/#sas.models.resize_image","title":"resize_image(im, edge, img_format, *, optimize=True)","text":"

Resize an image to fit the given edge length and format.

Parameters:

Name Type Description Default im Image

the image to resize

required edge int

the length that the greater side of the resized image should have

required img_format str

the target format of the image (\"JPEG\", \"PNG\", \"WEBP\"...)

required optimize bool

Should the resized image be optimized ?

True Source code in core/utils.py
def resize_image(\n    im: Image, edge: int, img_format: str, *, optimize: bool = True\n) -> ContentFile:\n    \"\"\"Resize an image to fit the given edge length and format.\n\n    Args:\n        im: the image to resize\n        edge: the length that the greater side of the resized image should have\n        img_format: the target format of the image (\"JPEG\", \"PNG\", \"WEBP\"...)\n        optimize: Should the resized image be optimized ?\n    \"\"\"\n    (w, h) = im.size\n    ratio = edge / max(w, h)\n    (width, height) = int(w * ratio), int(h * ratio)\n    return resize_image_explicit(im, (width, height), img_format, optimize=optimize)\n
"},{"location":"reference/sas/models/#sas.models.sas_notification_callback","title":"sas_notification_callback(notif)","text":"Source code in sas/models.py
def sas_notification_callback(notif):\n    count = Picture.objects.filter(is_moderated=False).count()\n    if count:\n        notif.viewed = False\n    else:\n        notif.viewed = True\n    notif.param = \"%s\" % count\n    notif.date = timezone.now()\n
"},{"location":"reference/sas/schemas/","title":"Schemas","text":""},{"location":"reference/sas/schemas/#sas.schemas.SimpleUserSchema","title":"SimpleUserSchema","text":"

Bases: ModelSchema

A schema with the minimum amount of information to represent a user.

"},{"location":"reference/sas/schemas/#sas.schemas.UserProfileSchema","title":"UserProfileSchema","text":"

Bases: ModelSchema

The necessary information to show a user profile

"},{"location":"reference/sas/schemas/#sas.schemas.Album","title":"Album","text":"

Bases: SasFile

"},{"location":"reference/sas/schemas/#sas.schemas.Album.NAME_MAX_LENGTH","title":"NAME_MAX_LENGTH = 50 class-attribute","text":"

Maximum length of an album's name.

SithFile have a maximum length of 256 characters. However, this limit is too high for albums. Names longer than 50 characters are harder to read and harder to display on the SAS page.

It is to be noted, though, that this does not add or modify any db behaviour. It's just a constant to be used in views and forms.

"},{"location":"reference/sas/schemas/#sas.schemas.Picture","title":"Picture","text":"

Bases: SasFile

"},{"location":"reference/sas/schemas/#sas.schemas.PictureModerationRequest","title":"PictureModerationRequest","text":"

Bases: Model

A request to remove a Picture from the SAS.

"},{"location":"reference/sas/schemas/#sas.schemas.AlbumSchema","title":"AlbumSchema","text":"

Bases: ModelSchema

"},{"location":"reference/sas/schemas/#sas.schemas.PictureFilterSchema","title":"PictureFilterSchema","text":"

Bases: FilterSchema

"},{"location":"reference/sas/schemas/#sas.schemas.PictureSchema","title":"PictureSchema","text":"

Bases: ModelSchema

"},{"location":"reference/sas/schemas/#sas.schemas.PictureRelationCreationSchema","title":"PictureRelationCreationSchema","text":"

Bases: Schema

"},{"location":"reference/sas/schemas/#sas.schemas.IdentifiedUserSchema","title":"IdentifiedUserSchema","text":"

Bases: Schema

"},{"location":"reference/sas/schemas/#sas.schemas.ModerationRequestSchema","title":"ModerationRequestSchema","text":"

Bases: ModelSchema

"},{"location":"reference/sas/views/","title":"Views","text":""},{"location":"reference/sas/views/#sas.views.CanEditMixin","title":"CanEditMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to edit this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/sas/views/#sas.views.CanViewMixin","title":"CanViewMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to view this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/sas/views/#sas.views.SithFile","title":"SithFile","text":"

Bases: Model

"},{"location":"reference/sas/views/#sas.views.SithFile.clean","title":"clean()","text":"

Cleans up the file.

Source code in core/models.py
def clean(self):\n    \"\"\"Cleans up the file.\"\"\"\n    super().clean()\n    if \"/\" in self.name:\n        raise ValidationError(_(\"Character '/' not authorized in name\"))\n    if self == self.parent:\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self == self.parent or (\n        self.parent is not None and self in self.get_parent_list()\n    ):\n        raise ValidationError(_(\"Loop in folder tree\"), code=\"loop\")\n    if self.parent and self.parent.is_file:\n        raise ValidationError(\n            _(\"You can not make a file be a children of a non folder file\")\n        )\n    if (\n        self.parent is None\n        and SithFile.objects.exclude(id=self.id)\n        .filter(parent=None, name=self.name)\n        .exists()\n    ) or (\n        self.parent\n        and self.parent.children.exclude(id=self.id).filter(name=self.name).exists()\n    ):\n        raise ValidationError(_(\"Duplicate file\"), code=\"duplicate\")\n    if self.is_folder:\n        if self.file:\n            try:\n                Image.open(BytesIO(self.file.read()))\n            except Image.UnidentifiedImageError as e:\n                raise ValidationError(\n                    _(\"This is not a valid folder thumbnail\")\n                ) from e\n        self.mime_type = \"inode/directory\"\n    if self.is_file and (self.file is None or self.file == \"\"):\n        raise ValidationError(_(\"You must provide a file\"))\n
"},{"location":"reference/sas/views/#sas.views.SithFile.apply_rights_recursively","title":"apply_rights_recursively(*, only_folders=False)","text":"

Apply the rights of this file to all children recursively.

Parameters:

Name Type Description Default only_folders bool

If True, only apply the rights to SithFiles that are folders.

False Source code in core/models.py
def apply_rights_recursively(self, *, only_folders: bool = False) -> None:\n    \"\"\"Apply the rights of this file to all children recursively.\n\n    Args:\n        only_folders: If True, only apply the rights to SithFiles that are folders.\n    \"\"\"\n    file_ids = []\n    explored_ids = [self.id]\n    while len(explored_ids) > 0:  # find all children recursively\n        file_ids.extend(explored_ids)\n        next_level = SithFile.objects.filter(parent_id__in=explored_ids)\n        if only_folders:\n            next_level = next_level.filter(is_folder=True)\n        explored_ids = list(next_level.values_list(\"id\", flat=True))\n    for through in (SithFile.view_groups.through, SithFile.edit_groups.through):\n        # force evaluation. Without this, the iterator yields nothing\n        groups = list(\n            through.objects.filter(sithfile_id=self.id).values_list(\n                \"group_id\", flat=True\n            )\n        )\n        # delete previous rights\n        through.objects.filter(sithfile_id__in=file_ids).delete()\n        through.objects.bulk_create(  # create new rights\n            [through(sithfile_id=f, group_id=g) for f in file_ids for g in groups]\n        )\n
"},{"location":"reference/sas/views/#sas.views.SithFile.copy_rights","title":"copy_rights()","text":"

Copy, if possible, the rights of the parent folder.

Source code in core/models.py
def copy_rights(self):\n    \"\"\"Copy, if possible, the rights of the parent folder.\"\"\"\n    if self.parent is not None:\n        self.edit_groups.set(self.parent.edit_groups.all())\n        self.view_groups.set(self.parent.view_groups.all())\n
"},{"location":"reference/sas/views/#sas.views.SithFile.move_to","title":"move_to(parent)","text":"

Move a file to a new parent. parent must be a SithFile with the is_folder=True property. Otherwise, this function doesn't change anything. This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify SithFiles recursively, so it stays efficient even with top-level folders.

Source code in core/models.py
def move_to(self, parent):\n    \"\"\"Move a file to a new parent.\n    `parent` must be a SithFile with the `is_folder=True` property. Otherwise, this function doesn't change\n    anything.\n    This is done only at the DB level, so that it's very fast for the user. Indeed, this function doesn't modify\n    SithFiles recursively, so it stays efficient even with top-level folders.\n    \"\"\"\n    if not parent.is_folder:\n        return\n    self.parent = parent\n    self.clean()\n    self.save()\n
"},{"location":"reference/sas/views/#sas.views.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/sas/views/#sas.views.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/sas/views/#sas.views.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/sas/views/#sas.views.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/sas/views/#sas.views.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/sas/views/#sas.views.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/sas/views/#sas.views.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/sas/views/#sas.views.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/sas/views/#sas.views.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/sas/views/#sas.views.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/sas/views/#sas.views.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/sas/views/#sas.views.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/sas/views/#sas.views.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/sas/views/#sas.views.FileView","title":"FileView","text":"

Bases: CanViewMixin, DetailView, FormMixin

Handle the upload of new files into a folder.

"},{"location":"reference/sas/views/#sas.views.FileView.handle_clipboard","title":"handle_clipboard(request, obj) staticmethod","text":"

Handle the clipboard in the view.

This method can fail, since it does not catch the exceptions coming from below, allowing proper handling in the calling view. Use this method like this:

FileView.handle_clipboard(request, self.object)\n

request is usually the self.request obj in your view obj is the SithFile object you want to put in the clipboard, or where you want to paste the clipboard

Source code in core/views/files.py
@staticmethod\ndef handle_clipboard(request, obj):\n    \"\"\"Handle the clipboard in the view.\n\n    This method can fail, since it does not catch the exceptions coming from\n    below, allowing proper handling in the calling view.\n    Use this method like this:\n\n        FileView.handle_clipboard(request, self.object)\n\n    `request` is usually the self.request obj in your view\n    `obj` is the SithFile object you want to put in the clipboard, or\n             where you want to paste the clipboard\n    \"\"\"\n    if \"delete\" in request.POST:\n        for f_id in request.POST.getlist(\"file_list\"):\n            file = SithFile.objects.filter(id=f_id).first()\n            if file:\n                file.delete()\n    if \"clear\" in request.POST:\n        request.session[\"clipboard\"] = []\n    if \"cut\" in request.POST:\n        for f_id_str in request.POST.getlist(\"file_list\"):\n            f_id = int(f_id_str)\n            if (\n                f_id in [c.id for c in obj.children.all()]\n                and f_id not in request.session[\"clipboard\"]\n            ):\n                request.session[\"clipboard\"].append(f_id)\n    if \"paste\" in request.POST:\n        for f_id in request.session[\"clipboard\"]:\n            file = SithFile.objects.filter(id=f_id).first()\n            if file:\n                file.move_to(obj)\n        request.session[\"clipboard\"] = []\n    request.session.modified = True\n
"},{"location":"reference/sas/views/#sas.views.AlbumEditForm","title":"AlbumEditForm","text":"

Bases: ModelForm

"},{"location":"reference/sas/views/#sas.views.PictureEditForm","title":"PictureEditForm","text":"

Bases: ModelForm

"},{"location":"reference/sas/views/#sas.views.PictureModerationRequestForm","title":"PictureModerationRequestForm(*args, user, picture, **kwargs)","text":"

Bases: ModelForm

Form to create a PictureModerationRequest.

The form only manages the reason field, because the author and the picture are set in the view.

Source code in sas/forms.py
def __init__(self, *args, user: User, picture: Picture, **kwargs):\n    super().__init__(*args, **kwargs)\n    self.user = user\n    self.picture = picture\n
"},{"location":"reference/sas/views/#sas.views.SASForm","title":"SASForm","text":"

Bases: Form

"},{"location":"reference/sas/views/#sas.views.Album","title":"Album","text":"

Bases: SasFile

"},{"location":"reference/sas/views/#sas.views.Album.NAME_MAX_LENGTH","title":"NAME_MAX_LENGTH = 50 class-attribute","text":"

Maximum length of an album's name.

SithFile have a maximum length of 256 characters. However, this limit is too high for albums. Names longer than 50 characters are harder to read and harder to display on the SAS page.

It is to be noted, though, that this does not add or modify any db behaviour. It's just a constant to be used in views and forms.

"},{"location":"reference/sas/views/#sas.views.Picture","title":"Picture","text":"

Bases: SasFile

"},{"location":"reference/sas/views/#sas.views.SASMainView","title":"SASMainView","text":"

Bases: FormView

"},{"location":"reference/sas/views/#sas.views.PictureView","title":"PictureView","text":"

Bases: CanViewMixin, DetailView

"},{"location":"reference/sas/views/#sas.views.AlbumUploadView","title":"AlbumUploadView","text":"

Bases: CanViewMixin, DetailView, FormMixin

"},{"location":"reference/sas/views/#sas.views.AlbumView","title":"AlbumView","text":"

Bases: CanViewMixin, DetailView, FormMixin

"},{"location":"reference/sas/views/#sas.views.ModerationView","title":"ModerationView","text":"

Bases: TemplateView

"},{"location":"reference/sas/views/#sas.views.PictureEditView","title":"PictureEditView","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/sas/views/#sas.views.PictureAskRemovalView","title":"PictureAskRemovalView","text":"

Bases: CanViewMixin, DetailView, FormView

View to allow users to ask pictures to be removed.

"},{"location":"reference/sas/views/#sas.views.PictureAskRemovalView.get_form_kwargs","title":"get_form_kwargs()","text":"

Add the user and picture to the form kwargs.

Those are required to create the PictureModerationRequest, and aren't part of the form itself (picture is a path parameter, and user is the request user).

Source code in sas/views.py
def get_form_kwargs(self) -> dict[str, Any]:\n    \"\"\"Add the user and picture to the form kwargs.\n\n    Those are required to create the PictureModerationRequest,\n    and aren't part of the form itself\n    (picture is a path parameter, and user is the request user).\n    \"\"\"\n    return super().get_form_kwargs() | {\n        \"user\": self.request.user,\n        \"picture\": self.object,\n    }\n
"},{"location":"reference/sas/views/#sas.views.PictureAskRemovalView.get_success_url","title":"get_success_url()","text":"

Return the URL to the album containing the picture.

Source code in sas/views.py
def get_success_url(self) -> str:\n    \"\"\"Return the URL to the album containing the picture.\"\"\"\n    album = Album.objects.filter(pk=self.object.parent_id).first()\n    if not album:\n        return reverse(\"sas:main\")\n    return album.get_absolute_url()\n
"},{"location":"reference/sas/views/#sas.views.AlbumEditView","title":"AlbumEditView","text":"

Bases: CanEditMixin, UpdateView

"},{"location":"reference/sas/views/#sas.views.send_file","title":"send_file(request, file_id, file_class=SithFile, file_attr='file')","text":"

Send a protected file, if the user can see it.

In prod, the server won't handle the download itself, but set the appropriate headers in the response to make the reverse-proxy deal with it. In debug mode, the server will directly send the file.

Source code in core/views/files.py
def send_file(\n    request: HttpRequest,\n    file_id: int,\n    file_class: type[SithFile] = SithFile,\n    file_attr: str = \"file\",\n) -> HttpResponse:\n    \"\"\"Send a protected file, if the user can see it.\n\n    In prod, the server won't handle the download itself,\n    but set the appropriate headers in the response to make the reverse-proxy\n    deal with it.\n    In debug mode, the server will directly send the file.\n    \"\"\"\n    f = get_object_or_404(file_class, id=file_id)\n    if not can_view(f, request.user) and not is_logged_in_counter(request):\n        raise PermissionDenied\n    name = getattr(f, file_attr).name\n\n    return send_raw_file(settings.MEDIA_ROOT / name)\n
"},{"location":"reference/sas/views/#sas.views.send_album","title":"send_album(request, album_id)","text":"Source code in sas/views.py
def send_album(request, album_id):\n    return send_file(request, album_id, Album)\n
"},{"location":"reference/sas/views/#sas.views.send_pict","title":"send_pict(request, picture_id)","text":"Source code in sas/views.py
def send_pict(request, picture_id):\n    return send_file(request, picture_id, Picture)\n
"},{"location":"reference/sas/views/#sas.views.send_compressed","title":"send_compressed(request, picture_id)","text":"Source code in sas/views.py
def send_compressed(request, picture_id):\n    return send_file(request, picture_id, Picture, \"compressed\")\n
"},{"location":"reference/sas/views/#sas.views.send_thumb","title":"send_thumb(request, picture_id)","text":"Source code in sas/views.py
def send_thumb(request, picture_id):\n    return send_file(request, picture_id, Picture, \"thumbnail\")\n
"},{"location":"reference/staticfiles/apps/","title":"Apps","text":"

Bases: StaticFilesConfig

Application in charge of processing statics files. It replaces the original django staticfiles It integrates scss files and javascript bundling. It makes sure that statics are properly collected and that they are automatically when using the development server.

"},{"location":"reference/staticfiles/apps/#staticfiles.apps.StaticFilesConfig.ignore_patterns","title":"ignore_patterns = IGNORE_PATTERNS class-attribute instance-attribute","text":""},{"location":"reference/staticfiles/apps/#staticfiles.apps.StaticFilesConfig.name","title":"name = 'staticfiles' class-attribute instance-attribute","text":""},{"location":"reference/staticfiles/finders/","title":"Finders","text":""},{"location":"reference/staticfiles/finders/#staticfiles.finders.GENERATED_ROOT","title":"GENERATED_ROOT = Path(__file__).parent.resolve() / 'generated' module-attribute","text":""},{"location":"reference/staticfiles/finders/#staticfiles.finders.IGNORE_PATTERNS_BUNDLED","title":"IGNORE_PATTERNS_BUNDLED = [f'{BUNDLED_FOLDER_NAME}/*'] module-attribute","text":""},{"location":"reference/staticfiles/finders/#staticfiles.finders.GeneratedFilesFinder","title":"GeneratedFilesFinder(app_names=None, *args, **kwargs)","text":"

Bases: FileSystemFinder

Find generated and regular static files

Source code in staticfiles/finders.py
def __init__(self, app_names=None, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n\n    # Add GENERATED_ROOT after adding everything in settings.STATICFILES_DIRS\n    self.locations.append((\"\", GENERATED_ROOT))\n    generated_storage = FileSystemStorage(location=GENERATED_ROOT)\n    generated_storage.prefix = \"\"\n    self.storages[GENERATED_ROOT] = generated_storage\n
"},{"location":"reference/staticfiles/processors/","title":"Processors","text":""},{"location":"reference/staticfiles/processors/#staticfiles.processors.BUNDLED_FOLDER_NAME","title":"BUNDLED_FOLDER_NAME = 'bundled' module-attribute","text":""},{"location":"reference/staticfiles/processors/#staticfiles.processors.BUNDLED_ROOT","title":"BUNDLED_ROOT = GENERATED_ROOT / BUNDLED_FOLDER_NAME module-attribute","text":""},{"location":"reference/staticfiles/processors/#staticfiles.processors.GENERATED_ROOT","title":"GENERATED_ROOT = Path(__file__).parent.resolve() / 'generated' module-attribute","text":""},{"location":"reference/staticfiles/processors/#staticfiles.processors.JsBundlerManifestEntry","title":"JsBundlerManifestEntry(src, out) dataclass","text":""},{"location":"reference/staticfiles/processors/#staticfiles.processors.JSBundlerManifest","title":"JSBundlerManifest(manifest)","text":"Source code in staticfiles/processors.py
def __init__(self, manifest: Path):\n    with open(manifest, \"r\") as f:\n        self._manifest = json.load(f)\n\n    self._files = chain(\n        *[\n            JsBundlerManifestEntry.from_json_entry(value)\n            for value in self._manifest.values()\n            if value.get(\"isEntry\", False)\n        ]\n    )\n    self.mapping = {file.src: file.out for file in self._files}\n
"},{"location":"reference/staticfiles/processors/#staticfiles.processors.JSBundler","title":"JSBundler","text":""},{"location":"reference/staticfiles/processors/#staticfiles.processors.JSBundler.compile","title":"compile() staticmethod","text":"

Bundle js files with the javascript bundler for production.

Source code in staticfiles/processors.py
@staticmethod\ndef compile():\n    \"\"\"Bundle js files with the javascript bundler for production.\"\"\"\n    process = subprocess.Popen([\"npm\", \"run\", \"compile\"])\n    process.wait()\n    if process.returncode:\n        raise RuntimeError(f\"Bundler failed with returncode {process.returncode}\")\n
"},{"location":"reference/staticfiles/processors/#staticfiles.processors.JSBundler.runserver","title":"runserver() staticmethod","text":"

Bundle js files automatically in background when called in debug mode.

Source code in staticfiles/processors.py
@staticmethod\ndef runserver() -> subprocess.Popen:\n    \"\"\"Bundle js files automatically in background when called in debug mode.\"\"\"\n    logging.getLogger(\"django\").info(\"Running javascript bundling server\")\n    return subprocess.Popen([\"npm\", \"run\", \"serve\"])\n
"},{"location":"reference/staticfiles/processors/#staticfiles.processors.Scss","title":"Scss","text":""},{"location":"reference/staticfiles/processors/#staticfiles.processors.Scss.compile","title":"compile(files) staticmethod","text":"

Compile scss files to css files.

Source code in staticfiles/processors.py
@staticmethod\ndef compile(files: CompileArg | Iterable[CompileArg]):\n    \"\"\"Compile scss files to css files.\"\"\"\n    # Generate files inside the generated folder\n    # .css files respects the hierarchy in the static folder it was found\n    # This converts arg.absolute -> generated/{arg.relative}.scss\n    # Example:\n    #   app/static/foo.scss          -> generated/foo.css\n    #   app/static/bar/foo.scss      -> generated/bar/foo.css\n    #   custom/location/bar/foo.scss -> generated/bar/foo.css\n    if isinstance(files, Scss.CompileArg):\n        files = [files]\n\n    base_args = {\"output_style\": \"compressed\", \"precision\": settings.SASS_PRECISION}\n\n    compiled_files = {\n        file.relative.with_suffix(\".css\"): sass.compile(\n            filename=str(file.absolute), **base_args\n        )\n        for file in files\n    }\n    for file, content in compiled_files.items():\n        dest = GENERATED_ROOT / file\n        dest.parent.mkdir(exist_ok=True, parents=True)\n        dest.write_text(content)\n
"},{"location":"reference/staticfiles/processors/#staticfiles.processors.JS","title":"JS","text":""},{"location":"reference/staticfiles/processors/#staticfiles.processors.OpenApi","title":"OpenApi","text":""},{"location":"reference/staticfiles/processors/#staticfiles.processors.OpenApi.compile","title":"compile() classmethod","text":"

Compile a TS client for the sith API. Only generates it if it changed.

Source code in staticfiles/processors.py
@classmethod\ndef compile(cls):\n    \"\"\"Compile a TS client for the sith API. Only generates it if it changed.\"\"\"\n    logging.getLogger(\"django\").info(\"Compiling open api typescript client\")\n    out = cls.OPENAPI_DIR / \"schema.json\"\n    cls.OPENAPI_DIR.mkdir(parents=True, exist_ok=True)\n\n    old_hash = \"\"\n    if out.exists():\n        with open(out, \"rb\") as f:\n            old_hash = sha1(f.read()).hexdigest()\n\n    schema = api.get_openapi_schema()\n    # Remove hash from operationIds\n    # This is done for cache invalidation but this is too aggressive\n    for path in schema[\"paths\"].values():\n        for action, desc in path.items():\n            path[action][\"operationId\"] = \"_\".join(\n                desc[\"operationId\"].split(\"_\")[:-1]\n            )\n    schema = str(schema)\n\n    if old_hash == sha1(schema.encode(\"utf-8\")).hexdigest():\n        logging.getLogger(\"django\").info(\"\u2728 Api did not change, nothing to do \u2728\")\n        return\n\n    with open(out, \"w\") as f:\n        _ = f.write(schema)\n\n    subprocess.run([\"npx\", \"openapi-ts\"], check=True)\n
"},{"location":"reference/staticfiles/storage/","title":"Storage","text":""},{"location":"reference/staticfiles/storage/#staticfiles.storage.JS","title":"JS","text":""},{"location":"reference/staticfiles/storage/#staticfiles.storage.JSBundler","title":"JSBundler","text":""},{"location":"reference/staticfiles/storage/#staticfiles.storage.JSBundler.compile","title":"compile() staticmethod","text":"

Bundle js files with the javascript bundler for production.

Source code in staticfiles/processors.py
@staticmethod\ndef compile():\n    \"\"\"Bundle js files with the javascript bundler for production.\"\"\"\n    process = subprocess.Popen([\"npm\", \"run\", \"compile\"])\n    process.wait()\n    if process.returncode:\n        raise RuntimeError(f\"Bundler failed with returncode {process.returncode}\")\n
"},{"location":"reference/staticfiles/storage/#staticfiles.storage.JSBundler.runserver","title":"runserver() staticmethod","text":"

Bundle js files automatically in background when called in debug mode.

Source code in staticfiles/processors.py
@staticmethod\ndef runserver() -> subprocess.Popen:\n    \"\"\"Bundle js files automatically in background when called in debug mode.\"\"\"\n    logging.getLogger(\"django\").info(\"Running javascript bundling server\")\n    return subprocess.Popen([\"npm\", \"run\", \"serve\"])\n
"},{"location":"reference/staticfiles/storage/#staticfiles.storage.Scss","title":"Scss","text":""},{"location":"reference/staticfiles/storage/#staticfiles.storage.Scss.compile","title":"compile(files) staticmethod","text":"

Compile scss files to css files.

Source code in staticfiles/processors.py
@staticmethod\ndef compile(files: CompileArg | Iterable[CompileArg]):\n    \"\"\"Compile scss files to css files.\"\"\"\n    # Generate files inside the generated folder\n    # .css files respects the hierarchy in the static folder it was found\n    # This converts arg.absolute -> generated/{arg.relative}.scss\n    # Example:\n    #   app/static/foo.scss          -> generated/foo.css\n    #   app/static/bar/foo.scss      -> generated/bar/foo.css\n    #   custom/location/bar/foo.scss -> generated/bar/foo.css\n    if isinstance(files, Scss.CompileArg):\n        files = [files]\n\n    base_args = {\"output_style\": \"compressed\", \"precision\": settings.SASS_PRECISION}\n\n    compiled_files = {\n        file.relative.with_suffix(\".css\"): sass.compile(\n            filename=str(file.absolute), **base_args\n        )\n        for file in files\n    }\n    for file, content in compiled_files.items():\n        dest = GENERATED_ROOT / file\n        dest.parent.mkdir(exist_ok=True, parents=True)\n        dest.write_text(content)\n
"},{"location":"reference/staticfiles/storage/#staticfiles.storage.ManifestPostProcessingStorage","title":"ManifestPostProcessingStorage","text":"

Bases: ManifestStaticFilesStorage

"},{"location":"reference/staticfiles/storage/#staticfiles.storage.ManifestPostProcessingStorage.url","title":"url(name, *, force=False)","text":"

Get the URL for a file, convert .scss calls to .css calls to bundled files to their output ones

Source code in staticfiles/storage.py
def url(self, name: str, *, force: bool = False) -> str:\n    \"\"\"Get the URL for a file, convert .scss calls to .css calls to bundled files to their output ones\"\"\"\n    # This name swap has to be done here\n    # Otherwise, the manifest isn't aware of the file and can't work properly\n    if settings.DEBUG:\n        # In production, the bundler manifest is used at compile time, we don't need to convert anything\n        try:\n            manifest = JSBundler.get_manifest()\n        except FileNotFoundError as e:\n            raise Exception(\n                \"Error loading manifest file, the bundler seems to be busy\"\n            ) from e\n        converted = manifest.mapping.get(name, None)\n        if converted:\n            name = converted\n\n    path = Path(name)\n    if path.suffix == \".scss\":\n        # Compile scss files automatically in debug mode\n        if settings.DEBUG:\n            Scss.compile(\n                [\n                    Scss.CompileArg(absolute=Path(p), relative=Path(name))\n                    for p in find(name, all=True)\n                ]\n            )\n        name = str(path.with_suffix(\".css\"))\n\n    return super().url(name, force=force)\n
"},{"location":"reference/subscription/models/","title":"Models","text":""},{"location":"reference/subscription/models/#subscription.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/subscription/models/#subscription.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/subscription/models/#subscription.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/subscription/models/#subscription.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/subscription/models/#subscription.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/subscription/models/#subscription.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/subscription/models/#subscription.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/subscription/models/#subscription.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/subscription/models/#subscription.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/subscription/models/#subscription.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/subscription/models/#subscription.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/subscription/models/#subscription.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/subscription/models/#subscription.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/subscription/models/#subscription.models.Subscription","title":"Subscription","text":"

Bases: Model

"},{"location":"reference/subscription/models/#subscription.models.Subscription.semester_duration","title":"semester_duration property","text":"

Duration of this subscription, in number of semester.

Notes

The Subscription object doesn't have to actually exist in the database to access this property

Examples:

subscription = Subscription(subscription_type=\"deux-semestres\")\nassert subscription.semester_duration == 2.0\n
"},{"location":"reference/subscription/models/#subscription.models.Subscription.compute_start","title":"compute_start(d=None, duration=1, user=None) staticmethod","text":"

Computes the start date of the subscription.

The computation is done with respect to the given date (default is today) and the start date given in settings.SITH_SEMESTER_START_AUTUMN. It takes the nearest past start date. Exemples: with SITH_SEMESTER_START_AUTUMN = (8, 15) Today -> Start date 2015-03-17 -> 2015-02-15 2015-01-11 -> 2014-08-15.

Source code in subscription/models.py
@staticmethod\ndef compute_start(\n    d: date | None = None, duration: int | float = 1, user: User | None = None\n) -> date:\n    \"\"\"Computes the start date of the subscription.\n\n    The computation is done with respect to the given date (default is today)\n    and the start date given in settings.SITH_SEMESTER_START_AUTUMN.\n    It takes the nearest past start date.\n    Exemples: with SITH_SEMESTER_START_AUTUMN = (8, 15)\n        Today      -> Start date\n        2015-03-17 -> 2015-02-15\n        2015-01-11 -> 2014-08-15.\n    \"\"\"\n    if not d:\n        d = date.today()\n    if user is not None and user.subscriptions.exists():\n        last = user.subscriptions.last()\n        if last.is_valid_now():\n            d = last.subscription_end\n    if duration <= 2:  # Sliding subscriptions for 1 or 2 semesters\n        return d\n    return get_start_of_semester(d)\n
"},{"location":"reference/subscription/models/#subscription.models.Subscription.compute_end","title":"compute_end(duration, start=None, user=None) staticmethod","text":"

Compute the end date of the subscription.

Parameters:

Name Type Description Default duration int | float

the duration of the subscription, in semester (for example, 2 => 2 semesters => 1 year)

required start date | None

The start date of the subscription

None user User | None

the user which is (or will be) subscribed

None Exemples

Start - Duration -> End date 2015-09-18 - 1 -> 2016-03-18 2015-09-18 - 2 -> 2016-09-18 2015-09-18 - 3 -> 2017-03-18 2015-09-18 - 4 -> 2017-09-18.

Source code in subscription/models.py
@staticmethod\ndef compute_end(\n    duration: int | float, start: date | None = None, user: User | None = None\n) -> date:\n    \"\"\"Compute the end date of the subscription.\n\n    Args:\n        duration:\n            the duration of the subscription, in semester\n            (for example, 2 => 2 semesters => 1 year)\n        start: The start date of the subscription\n        user: the user which is (or will be) subscribed\n\n    Exemples:\n        Start - Duration -> End date\n        2015-09-18 - 1 -> 2016-03-18\n        2015-09-18 - 2 -> 2016-09-18\n        2015-09-18 - 3 -> 2017-03-18\n        2015-09-18 - 4 -> 2017-09-18.\n    \"\"\"\n    if start is None:\n        start = Subscription.compute_start(duration=duration, user=user)\n\n    return start + relativedelta(\n        months=round(6 * duration),\n        days=math.ceil((6 * duration - round(6 * duration)) * 30),\n    )\n
"},{"location":"reference/subscription/models/#subscription.models.get_start_of_semester","title":"get_start_of_semester(today=None)","text":"

Return the date of the start of the semester of the given date. If no date is given, return the start date of the current semester.

The current semester is computed as follows:

Parameters:

Name Type Description Default today date | None

the date to use to compute the semester. If None, use today's date.

None

Returns:

Type Description date

the date of the start of the semester

Source code in core/utils.py
def get_start_of_semester(today: date | None = None) -> date:\n    \"\"\"Return the date of the start of the semester of the given date.\n    If no date is given, return the start date of the current semester.\n\n    The current semester is computed as follows:\n\n    - If the date is between 15/08 and 31/12  => Autumn semester.\n    - If the date is between 01/01 and 15/02  => Autumn semester of the previous year.\n    - If the date is between 15/02 and 15/08  => Spring semester\n\n    Args:\n        today: the date to use to compute the semester. If None, use today's date.\n\n    Returns:\n        the date of the start of the semester\n    \"\"\"\n    if today is None:\n        today = localdate()\n\n    autumn = date(today.year, *settings.SITH_SEMESTER_START_AUTUMN)\n    spring = date(today.year, *settings.SITH_SEMESTER_START_SPRING)\n\n    if today >= autumn:  # between 15/08 (included) and 31/12 -> autumn semester\n        return autumn\n    if today >= spring:  # between 15/02 (included) and 15/08 -> spring semester\n        return spring\n    # between 01/01 and 15/02 -> autumn semester of the previous year\n    return autumn.replace(year=autumn.year - 1)\n
"},{"location":"reference/subscription/models/#subscription.models.validate_type","title":"validate_type(value)","text":"Source code in subscription/models.py
def validate_type(value):\n    if value not in settings.SITH_SUBSCRIPTIONS:\n        raise ValidationError(_(\"Bad subscription type\"))\n
"},{"location":"reference/subscription/models/#subscription.models.validate_payment","title":"validate_payment(value)","text":"Source code in subscription/models.py
def validate_payment(value):\n    if value not in settings.SITH_SUBSCRIPTION_PAYMENT_METHOD:\n        raise ValidationError(_(\"Bad payment method\"))\n
"},{"location":"reference/subscription/views/","title":"Views","text":""},{"location":"reference/subscription/views/#subscription.views.PAYMENT_METHOD","title":"PAYMENT_METHOD = [('CHECK', _('Check')), ('CASH', _('Cash')), ('CARD', _('Credit card'))] module-attribute","text":""},{"location":"reference/subscription/views/#subscription.views.SelectionDateForm","title":"SelectionDateForm(*args, **kwargs)","text":"

Bases: Form

Source code in subscription/forms.py
def __init__(self, *args, **kwargs):\n    super().__init__(*args, **kwargs)\n    self.fields[\"start_date\"] = forms.DateTimeField(\n        label=_(\"Start date\"), widget=SelectDateTime, required=True\n    )\n    self.fields[\"end_date\"] = forms.DateTimeField(\n        label=_(\"End date\"), widget=SelectDateTime, required=True\n    )\n
"},{"location":"reference/subscription/views/#subscription.views.SubscriptionExistingUserForm","title":"SubscriptionExistingUserForm(*args, **kwargs)","text":"

Bases: SubscriptionForm

Form to add a subscription to an existing user.

Source code in subscription/forms.py
def __init__(self, *args, **kwargs):\n    initial = kwargs.pop(\"initial\", {})\n    if \"subscription_type\" not in initial:\n        initial[\"subscription_type\"] = \"deux-semestres\"\n    if \"payment_method\" not in initial:\n        initial[\"payment_method\"] = \"CARD\"\n    super().__init__(*args, initial=initial, **kwargs)\n
"},{"location":"reference/subscription/views/#subscription.views.SubscriptionNewUserForm","title":"SubscriptionNewUserForm(*args, **kwargs)","text":"

Bases: SubscriptionForm

Form to create subscriptions with the user they belong to.

Examples:

```py assert not User.objects.filter(email=request.POST.get(\"email\")).exists() form = SubscriptionNewUserForm(request.POST) if form.is_valid(): form.save()

"},{"location":"reference/subscription/views/#subscription.views.SubscriptionNewUserForm--now-the-user-exists-and-is-subscribed","title":"now the user exists and is subscribed","text":"

user = User.objects.get(email=request.POST.get(\"email\")) assert user.is_subscribed

Source code in subscription/forms.py
def __init__(self, *args, **kwargs):\n    initial = kwargs.pop(\"initial\", {})\n    if \"subscription_type\" not in initial:\n        initial[\"subscription_type\"] = \"deux-semestres\"\n    if \"payment_method\" not in initial:\n        initial[\"payment_method\"] = \"CARD\"\n    super().__init__(*args, initial=initial, **kwargs)\n
"},{"location":"reference/subscription/views/#subscription.views.Subscription","title":"Subscription","text":"

Bases: Model

"},{"location":"reference/subscription/views/#subscription.views.Subscription.semester_duration","title":"semester_duration property","text":"

Duration of this subscription, in number of semester.

Notes

The Subscription object doesn't have to actually exist in the database to access this property

Examples:

subscription = Subscription(subscription_type=\"deux-semestres\")\nassert subscription.semester_duration == 2.0\n
"},{"location":"reference/subscription/views/#subscription.views.Subscription.compute_start","title":"compute_start(d=None, duration=1, user=None) staticmethod","text":"

Computes the start date of the subscription.

The computation is done with respect to the given date (default is today) and the start date given in settings.SITH_SEMESTER_START_AUTUMN. It takes the nearest past start date. Exemples: with SITH_SEMESTER_START_AUTUMN = (8, 15) Today -> Start date 2015-03-17 -> 2015-02-15 2015-01-11 -> 2014-08-15.

Source code in subscription/models.py
@staticmethod\ndef compute_start(\n    d: date | None = None, duration: int | float = 1, user: User | None = None\n) -> date:\n    \"\"\"Computes the start date of the subscription.\n\n    The computation is done with respect to the given date (default is today)\n    and the start date given in settings.SITH_SEMESTER_START_AUTUMN.\n    It takes the nearest past start date.\n    Exemples: with SITH_SEMESTER_START_AUTUMN = (8, 15)\n        Today      -> Start date\n        2015-03-17 -> 2015-02-15\n        2015-01-11 -> 2014-08-15.\n    \"\"\"\n    if not d:\n        d = date.today()\n    if user is not None and user.subscriptions.exists():\n        last = user.subscriptions.last()\n        if last.is_valid_now():\n            d = last.subscription_end\n    if duration <= 2:  # Sliding subscriptions for 1 or 2 semesters\n        return d\n    return get_start_of_semester(d)\n
"},{"location":"reference/subscription/views/#subscription.views.Subscription.compute_end","title":"compute_end(duration, start=None, user=None) staticmethod","text":"

Compute the end date of the subscription.

Parameters:

Name Type Description Default duration int | float

the duration of the subscription, in semester (for example, 2 => 2 semesters => 1 year)

required start date | None

The start date of the subscription

None user User | None

the user which is (or will be) subscribed

None Exemples

Start - Duration -> End date 2015-09-18 - 1 -> 2016-03-18 2015-09-18 - 2 -> 2016-09-18 2015-09-18 - 3 -> 2017-03-18 2015-09-18 - 4 -> 2017-09-18.

Source code in subscription/models.py
@staticmethod\ndef compute_end(\n    duration: int | float, start: date | None = None, user: User | None = None\n) -> date:\n    \"\"\"Compute the end date of the subscription.\n\n    Args:\n        duration:\n            the duration of the subscription, in semester\n            (for example, 2 => 2 semesters => 1 year)\n        start: The start date of the subscription\n        user: the user which is (or will be) subscribed\n\n    Exemples:\n        Start - Duration -> End date\n        2015-09-18 - 1 -> 2016-03-18\n        2015-09-18 - 2 -> 2016-09-18\n        2015-09-18 - 3 -> 2017-03-18\n        2015-09-18 - 4 -> 2017-09-18.\n    \"\"\"\n    if start is None:\n        start = Subscription.compute_start(duration=duration, user=user)\n\n    return start + relativedelta(\n        months=round(6 * duration),\n        days=math.ceil((6 * duration - round(6 * duration)) * 30),\n    )\n
"},{"location":"reference/subscription/views/#subscription.views.CanCreateSubscriptionMixin","title":"CanCreateSubscriptionMixin","text":"

Bases: UserPassesTestMixin

"},{"location":"reference/subscription/views/#subscription.views.NewSubscription","title":"NewSubscription","text":"

Bases: CanCreateSubscriptionMixin, TemplateView

"},{"location":"reference/subscription/views/#subscription.views.CreateSubscriptionFragment","title":"CreateSubscriptionFragment","text":"

Bases: CanCreateSubscriptionMixin, CreateView

"},{"location":"reference/subscription/views/#subscription.views.CreateSubscriptionExistingUserFragment","title":"CreateSubscriptionExistingUserFragment","text":"

Bases: CreateSubscriptionFragment

Create a subscription for a user who already exists.

"},{"location":"reference/subscription/views/#subscription.views.CreateSubscriptionNewUserFragment","title":"CreateSubscriptionNewUserFragment","text":"

Bases: CreateSubscriptionFragment

Create a subscription for a user who already exists.

"},{"location":"reference/subscription/views/#subscription.views.SubscriptionCreatedFragment","title":"SubscriptionCreatedFragment","text":"

Bases: CanCreateSubscriptionMixin, DetailView

"},{"location":"reference/subscription/views/#subscription.views.SubscriptionsStatsView","title":"SubscriptionsStatsView","text":"

Bases: FormView

"},{"location":"reference/trombi/models/","title":"Models","text":""},{"location":"reference/trombi/models/#trombi.models.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/trombi/models/#trombi.models.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/trombi/models/#trombi.models.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/trombi/models/#trombi.models.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/trombi/models/#trombi.models.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/trombi/models/#trombi.models.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/trombi/models/#trombi.models.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/trombi/models/#trombi.models.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/trombi/models/#trombi.models.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/trombi/models/#trombi.models.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/trombi/models/#trombi.models.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/trombi/models/#trombi.models.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/trombi/models/#trombi.models.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/trombi/models/#trombi.models.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/trombi/models/#trombi.models.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/trombi/models/#trombi.models.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/trombi/models/#trombi.models.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/trombi/models/#trombi.models.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/trombi/models/#trombi.models.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/trombi/models/#trombi.models.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/trombi/models/#trombi.models.TrombiManager","title":"TrombiManager","text":"

Bases: Manager

"},{"location":"reference/trombi/models/#trombi.models.AvailableTrombiManager","title":"AvailableTrombiManager","text":"

Bases: Manager

"},{"location":"reference/trombi/models/#trombi.models.Trombi","title":"Trombi","text":"

Bases: Model

Main class of the trombi, the Trombi itself.

It contains the deadlines for the users, and the link to the club that makes its Trombi.

"},{"location":"reference/trombi/models/#trombi.models.TrombiUser","title":"TrombiUser","text":"

Bases: Model

Bound between a User and a Trombi.

This class is here to avoid cross-references between the core, club, and trombi modules. It also adds the pictures to the profile without needing all the security like the other SithFiles.

"},{"location":"reference/trombi/models/#trombi.models.TrombiComment","title":"TrombiComment","text":"

Bases: Model

A comment given by someone to someone else in the same Trombi instance.

"},{"location":"reference/trombi/models/#trombi.models.TrombiClubMembership","title":"TrombiClubMembership","text":"

Bases: Model

A membership in a club.

"},{"location":"reference/trombi/models/#trombi.models.get_semester_code","title":"get_semester_code(d=None)","text":"

Return the semester code of the given date. If no date is given, return the semester code of the current semester.

The semester code is an upper letter (A for autumn, P for spring), followed by the last two digits of the year. For example, the autumn semester of 2018 is \"A18\".

Parameters:

Name Type Description Default d date | None

the date to use to compute the semester. If None, use today's date.

None

Returns:

Type Description str

the semester code corresponding to the given date

Source code in core/utils.py
def get_semester_code(d: date | None = None) -> str:\n    \"\"\"Return the semester code of the given date.\n    If no date is given, return the semester code of the current semester.\n\n    The semester code is an upper letter (A for autumn, P for spring),\n    followed by the last two digits of the year.\n    For example, the autumn semester of 2018 is \"A18\".\n\n    Args:\n        d: the date to use to compute the semester. If None, use today's date.\n\n    Returns:\n        the semester code corresponding to the given date\n    \"\"\"\n    if d is None:\n        d = localdate()\n\n    start = get_start_of_semester(d)\n\n    if (start.month, start.day) == settings.SITH_SEMESTER_START_AUTUMN:\n        return \"A\" + str(start.year)[-2:]\n    return \"P\" + str(start.year)[-2:]\n
"},{"location":"reference/trombi/views/","title":"Views","text":""},{"location":"reference/trombi/views/#trombi.views.Club","title":"Club","text":"

Bases: Model

The Club class, made as a tree to allow nice tidy organization.

"},{"location":"reference/trombi/views/#trombi.views.Club.president","title":"president()","text":"

Fetch the membership of the current president of this club.

Source code in club/models.py
@cached_property\ndef president(self) -> Membership | None:\n    \"\"\"Fetch the membership of the current president of this club.\"\"\"\n    return self.members.filter(\n        role=settings.SITH_CLUB_ROLES_ID[\"President\"], end_date=None\n    ).first()\n
"},{"location":"reference/trombi/views/#trombi.views.Club.check_loop","title":"check_loop()","text":"

Raise a validation error when a loop is found within the parent list.

Source code in club/models.py
def check_loop(self):\n    \"\"\"Raise a validation error when a loop is found within the parent list.\"\"\"\n    objs = []\n    cur = self\n    while cur.parent is not None:\n        if cur in objs:\n            raise ValidationError(_(\"You can not make loops in clubs\"))\n        objs.append(cur)\n        cur = cur.parent\n
"},{"location":"reference/trombi/views/#trombi.views.Club.is_owned_by","title":"is_owned_by(user)","text":"

Method to see if that object can be super edited by the given user.

Source code in club/models.py
def is_owned_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be super edited by the given user.\"\"\"\n    if user.is_anonymous:\n        return False\n    return user.is_root or user.is_board_member\n
"},{"location":"reference/trombi/views/#trombi.views.Club.can_be_edited_by","title":"can_be_edited_by(user)","text":"

Method to see if that object can be edited by the given user.

Source code in club/models.py
def can_be_edited_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be edited by the given user.\"\"\"\n    return self.has_rights_in_club(user)\n
"},{"location":"reference/trombi/views/#trombi.views.Club.can_be_viewed_by","title":"can_be_viewed_by(user)","text":"

Method to see if that object can be seen by the given user.

Source code in club/models.py
def can_be_viewed_by(self, user: User) -> bool:\n    \"\"\"Method to see if that object can be seen by the given user.\"\"\"\n    return user.was_subscribed\n
"},{"location":"reference/trombi/views/#trombi.views.Club.get_membership_for","title":"get_membership_for(user)","text":"

Return the current membership the given user.

Note

The result is cached.

Source code in club/models.py
def get_membership_for(self, user: User) -> Membership | None:\n    \"\"\"Return the current membership the given user.\n\n    Note:\n        The result is cached.\n    \"\"\"\n    if user.is_anonymous:\n        return None\n    membership = cache.get(f\"membership_{self.id}_{user.id}\")\n    if membership == \"not_member\":\n        return None\n    if membership is None:\n        membership = self.members.filter(user=user, end_date=None).first()\n        if membership is None:\n            cache.set(f\"membership_{self.id}_{user.id}\", \"not_member\")\n        else:\n            cache.set(f\"membership_{self.id}_{user.id}\", membership)\n    return membership\n
"},{"location":"reference/trombi/views/#trombi.views.CanCreateMixin","title":"CanCreateMixin(*args, **kwargs)","text":"

Bases: View

Protect any child view that would create an object.

Raises:

Type Description PermissionDenied

If the user has not the necessary permission to create the object of the view.

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/trombi/views/#trombi.views.CanEditMixin","title":"CanEditMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to edit this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/trombi/views/#trombi.views.CanEditPropMixin","title":"CanEditPropMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has owner permissions on the child view object.

In other word, you can make a view with this view as parent, and it will be retricted to the users that are in the object's owner_group or that pass the obj.can_be_viewed_by test.

Raises:

Type Description PermissionDenied

If the user cannot see the object

"},{"location":"reference/trombi/views/#trombi.views.CanViewMixin","title":"CanViewMixin","text":"

Bases: GenericContentPermissionMixinBuilder

Ensure the user has permission to view this view's object.

Raises:

Type Description PermissionDenied

if the user cannot edit this view's object.

"},{"location":"reference/trombi/views/#trombi.views.User","title":"User","text":"

Bases: AbstractUser

Defines the base user class, useable in every app.

This is almost the same as the auth module AbstractUser since it inherits from it, but some fields are required, and the username is generated automatically with the name of the user (see generate_username()).

Added field: nick_name, date_of_birth Required fields: email, first_name, last_name, date_of_birth

"},{"location":"reference/trombi/views/#trombi.views.User.cached_groups","title":"cached_groups property","text":"

Get the list of groups this user is in.

The result is cached for the default duration (should be 5 minutes)

Returns: A list of all the groups this user is in.

"},{"location":"reference/trombi/views/#trombi.views.User.is_in_group","title":"is_in_group(*, pk=None, name=None)","text":"

Check if this user is in the given group. Either a group id or a group name must be provided. If both are passed, only the id will be considered.

The group will be fetched using the given parameter. If no group is found, return False. If a group is found, check if this user is in the latter.

Returns:

Type Description bool

True if the user is the group, else False

Source code in core/models.py
def is_in_group(self, *, pk: int | None = None, name: str | None = None) -> bool:\n    \"\"\"Check if this user is in the given group.\n    Either a group id or a group name must be provided.\n    If both are passed, only the id will be considered.\n\n    The group will be fetched using the given parameter.\n    If no group is found, return False.\n    If a group is found, check if this user is in the latter.\n\n    Returns:\n         True if the user is the group, else False\n    \"\"\"\n    if pk is not None:\n        group: Optional[Group] = get_group(pk=pk)\n    elif name is not None:\n        group: Optional[Group] = get_group(name=name)\n    else:\n        raise ValueError(\"You must either provide the id or the name of the group\")\n    if group is None:\n        return False\n    if group.id == settings.SITH_GROUP_SUBSCRIBERS_ID:\n        return self.is_subscribed\n    if group.id == settings.SITH_GROUP_ROOT_ID:\n        return self.is_root\n    return group in self.cached_groups\n
"},{"location":"reference/trombi/views/#trombi.views.User.age","title":"age()","text":"

Return the age this user has the day the method is called. If the user has not filled his age, return 0.

Source code in core/models.py
@cached_property\ndef age(self) -> int:\n    \"\"\"Return the age this user has the day the method is called.\n    If the user has not filled his age, return 0.\n    \"\"\"\n    if self.date_of_birth is None:\n        return 0\n    today = timezone.now()\n    age = today.year - self.date_of_birth.year\n    # remove a year if this year's birthday is yet to come\n    age -= (today.month, today.day) < (\n        self.date_of_birth.month,\n        self.date_of_birth.day,\n    )\n    return age\n
"},{"location":"reference/trombi/views/#trombi.views.User.get_short_name","title":"get_short_name()","text":"

Returns the short name for the user.

Source code in core/models.py
def get_short_name(self):\n    \"\"\"Returns the short name for the user.\"\"\"\n    if self.nick_name:\n        return self.nick_name\n    return self.first_name + \" \" + self.last_name\n
"},{"location":"reference/trombi/views/#trombi.views.User.get_display_name","title":"get_display_name()","text":"

Returns the display name of the user.

A nickname if possible, otherwise, the full name.

Source code in core/models.py
def get_display_name(self) -> str:\n    \"\"\"Returns the display name of the user.\n\n    A nickname if possible, otherwise, the full name.\n    \"\"\"\n    if self.nick_name:\n        return \"%s (%s)\" % (self.get_full_name(), self.nick_name)\n    return self.get_full_name()\n
"},{"location":"reference/trombi/views/#trombi.views.User.get_family","title":"get_family(godfathers_depth=4, godchildren_depth=4)","text":"

Get the family of the user, with the given depth.

Parameters:

Name Type Description Default godfathers_depth NonNegativeInt

The number of generations of godfathers to fetch

4 godchildren_depth NonNegativeInt

The number of generations of godchildren to fetch

4

Returns:

Type Description set[through]

A list of family relationships in this user's family

Source code in core/models.py
def get_family(\n    self,\n    godfathers_depth: NonNegativeInt = 4,\n    godchildren_depth: NonNegativeInt = 4,\n) -> set[User.godfathers.through]:\n    \"\"\"Get the family of the user, with the given depth.\n\n    Args:\n        godfathers_depth: The number of generations of godfathers to fetch\n        godchildren_depth: The number of generations of godchildren to fetch\n\n    Returns:\n        A list of family relationships in this user's family\n    \"\"\"\n    res = []\n    for depth, key, reverse_key in [\n        (godfathers_depth, \"from_user_id\", \"to_user_id\"),\n        (godchildren_depth, \"to_user_id\", \"from_user_id\"),\n    ]:\n        if depth == 0:\n            continue\n        links = list(User.godfathers.through.objects.filter(**{key: self.id}))\n        res.extend(links)\n        for _ in range(1, depth):  # noqa: F402 we don't care about gettext here\n            ids = [getattr(c, reverse_key) for c in links]\n            links = list(\n                User.godfathers.through.objects.filter(\n                    **{f\"{key}__in\": ids}\n                ).exclude(id__in=[r.id for r in res])\n            )\n            if not links:\n                break\n            res.extend(links)\n    return set(res)\n
"},{"location":"reference/trombi/views/#trombi.views.User.email_user","title":"email_user(subject, message, from_email=None, **kwargs)","text":"

Sends an email to this User.

Source code in core/models.py
def email_user(self, subject, message, from_email=None, **kwargs):\n    \"\"\"Sends an email to this User.\"\"\"\n    if from_email is None:\n        from_email = settings.DEFAULT_FROM_EMAIL\n    send_mail(subject, message, from_email, [self.email], **kwargs)\n
"},{"location":"reference/trombi/views/#trombi.views.User.generate_username","title":"generate_username()","text":"

Generates a unique username based on the first and last names.

For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.

Returns:

Type Description str

The generated username.

Source code in core/models.py
def generate_username(self) -> str:\n    \"\"\"Generates a unique username based on the first and last names.\n\n    For example: Guy Carlier gives gcarlier, and gcarlier1 if the first one exists.\n\n    Returns:\n        The generated username.\n    \"\"\"\n\n    def remove_accents(data):\n        return \"\".join(\n            x\n            for x in unicodedata.normalize(\"NFKD\", data)\n            if unicodedata.category(x)[0] == \"L\"\n        ).lower()\n\n    user_name = (\n        remove_accents(self.first_name[0] + self.last_name)\n        .encode(\"ascii\", \"ignore\")\n        .decode(\"utf-8\")\n    )\n    # load all usernames which could conflict with the new one.\n    # we need to actually load them, instead of performing a count,\n    # because we cannot be sure that two usernames refer to the\n    # actual same word (eg. tmore and tmoreau)\n    possible_conflicts: list[str] = list(\n        User.objects.filter(username__startswith=user_name).values_list(\n            \"username\", flat=True\n        )\n    )\n    nb_conflicts = sum(\n        1 for name in possible_conflicts if name.rstrip(string.digits) == user_name\n    )\n    if nb_conflicts > 0:\n        user_name += str(nb_conflicts)  # exemple => exemple1\n    self.username = user_name\n    return user_name\n
"},{"location":"reference/trombi/views/#trombi.views.User.is_owner","title":"is_owner(obj)","text":"

Determine if the object is owned by the user.

Source code in core/models.py
def is_owner(self, obj):\n    \"\"\"Determine if the object is owned by the user.\"\"\"\n    if hasattr(obj, \"is_owned_by\") and obj.is_owned_by(self):\n        return True\n    if hasattr(obj, \"owner_group\") and self.is_in_group(pk=obj.owner_group.id):\n        return True\n    return self.is_root\n
"},{"location":"reference/trombi/views/#trombi.views.User.can_edit","title":"can_edit(obj)","text":"

Determine if the object can be edited by the user.

Source code in core/models.py
def can_edit(self, obj):\n    \"\"\"Determine if the object can be edited by the user.\"\"\"\n    if hasattr(obj, \"can_be_edited_by\") and obj.can_be_edited_by(self):\n        return True\n    if hasattr(obj, \"edit_groups\"):\n        for pk in obj.edit_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    if isinstance(obj, User) and obj == self:\n        return True\n    return self.is_owner(obj)\n
"},{"location":"reference/trombi/views/#trombi.views.User.can_view","title":"can_view(obj)","text":"

Determine if the object can be viewed by the user.

Source code in core/models.py
def can_view(self, obj):\n    \"\"\"Determine if the object can be viewed by the user.\"\"\"\n    if hasattr(obj, \"can_be_viewed_by\") and obj.can_be_viewed_by(self):\n        return True\n    if hasattr(obj, \"view_groups\"):\n        for pk in obj.view_groups.values_list(\"pk\", flat=True):\n            if self.is_in_group(pk=pk):\n                return True\n    return self.can_edit(obj)\n
"},{"location":"reference/trombi/views/#trombi.views.User.clubs_with_rights","title":"clubs_with_rights()","text":"

The list of clubs where the user has rights

Source code in core/models.py
@cached_property\ndef clubs_with_rights(self) -> list[Club]:\n    \"\"\"The list of clubs where the user has rights\"\"\"\n    memberships = self.memberships.ongoing().board().select_related(\"club\")\n    return [m.club for m in memberships]\n
"},{"location":"reference/trombi/views/#trombi.views.QuickNotifMixin","title":"QuickNotifMixin","text":""},{"location":"reference/trombi/views/#trombi.views.QuickNotifMixin.get_context_data","title":"get_context_data(**kwargs)","text":"

Add quick notifications to context.

Source code in core/views/mixins.py
def get_context_data(self, **kwargs):\n    \"\"\"Add quick notifications to context.\"\"\"\n    kwargs = super().get_context_data(**kwargs)\n    kwargs[\"quick_notifs\"] = []\n    for n in self.quick_notif_list:\n        kwargs[\"quick_notifs\"].append(settings.SITH_QUICK_NOTIF[n])\n    for key, val in settings.SITH_QUICK_NOTIF.items():\n        for gk in self.request.GET:\n            if key == gk:\n                kwargs[\"quick_notifs\"].append(val)\n    return kwargs\n
"},{"location":"reference/trombi/views/#trombi.views.TabedViewMixin","title":"TabedViewMixin","text":"

Bases: View

Basic functions for displaying tabs in the template.

"},{"location":"reference/trombi/views/#trombi.views.Trombi","title":"Trombi","text":"

Bases: Model

Main class of the trombi, the Trombi itself.

It contains the deadlines for the users, and the link to the club that makes its Trombi.

"},{"location":"reference/trombi/views/#trombi.views.TrombiClubMembership","title":"TrombiClubMembership","text":"

Bases: Model

A membership in a club.

"},{"location":"reference/trombi/views/#trombi.views.TrombiComment","title":"TrombiComment","text":"

Bases: Model

A comment given by someone to someone else in the same Trombi instance.

"},{"location":"reference/trombi/views/#trombi.views.TrombiUser","title":"TrombiUser","text":"

Bases: Model

Bound between a User and a Trombi.

This class is here to avoid cross-references between the core, club, and trombi modules. It also adds the pictures to the profile without needing all the security like the other SithFiles.

"},{"location":"reference/trombi/views/#trombi.views.TrombiTabsMixin","title":"TrombiTabsMixin","text":"

Bases: TabedViewMixin

"},{"location":"reference/trombi/views/#trombi.views.UserIsInATrombiMixin","title":"UserIsInATrombiMixin","text":"

Bases: View

Check if the requested user has a trombi_user attribute.

"},{"location":"reference/trombi/views/#trombi.views.TrombiForm","title":"TrombiForm","text":"

Bases: ModelForm

"},{"location":"reference/trombi/views/#trombi.views.TrombiCreateView","title":"TrombiCreateView(*args, **kwargs)","text":"

Bases: CanCreateMixin, CreateView

Create a trombi for a club.

Source code in core/auth/mixins.py
def __init__(self, *args, **kwargs):\n    warnings.warn(\n        f\"{self.__class__.__name__} is deprecated and should be replaced \"\n        \"by other permission verification mecanism.\",\n        DeprecationWarning,\n        stacklevel=2,\n    )\n    super().__init__(*args, **kwargs)\n
"},{"location":"reference/trombi/views/#trombi.views.TrombiCreateView.post","title":"post(request, *args, **kwargs)","text":"

Affect club.

Source code in trombi/views.py
def post(self, request, *args, **kwargs):\n    \"\"\"Affect club.\"\"\"\n    form = self.get_form()\n    if form.is_valid():\n        club = get_object_or_404(Club, id=self.kwargs[\"club_id\"])\n        form.instance.club = club\n        ret = self.form_valid(form)\n        return ret\n    else:\n        return self.form_invalid(form)\n
"},{"location":"reference/trombi/views/#trombi.views.TrombiEditView","title":"TrombiEditView","text":"

Bases: CanEditPropMixin, TrombiTabsMixin, UpdateView

"},{"location":"reference/trombi/views/#trombi.views.AddUserForm","title":"AddUserForm","text":"

Bases: Form

"},{"location":"reference/trombi/views/#trombi.views.TrombiDetailView","title":"TrombiDetailView","text":"

Bases: CanEditMixin, QuickNotifMixin, TrombiTabsMixin, DetailView

"},{"location":"reference/trombi/views/#trombi.views.TrombiExportView","title":"TrombiExportView","text":"

Bases: CanEditMixin, TrombiTabsMixin, DetailView

"},{"location":"reference/trombi/views/#trombi.views.TrombiDeleteUserView","title":"TrombiDeleteUserView","text":"

Bases: CanEditPropMixin, TrombiTabsMixin, DeleteView

"},{"location":"reference/trombi/views/#trombi.views.TrombiModerateCommentsView","title":"TrombiModerateCommentsView","text":"

Bases: CanEditPropMixin, QuickNotifMixin, TrombiTabsMixin, DetailView

"},{"location":"reference/trombi/views/#trombi.views.TrombiModerateForm","title":"TrombiModerateForm","text":"

Bases: Form

"},{"location":"reference/trombi/views/#trombi.views.TrombiModerateCommentView","title":"TrombiModerateCommentView","text":"

Bases: DetailView

"},{"location":"reference/trombi/views/#trombi.views.TrombiModelChoiceField","title":"TrombiModelChoiceField","text":"

Bases: ModelChoiceField

"},{"location":"reference/trombi/views/#trombi.views.UserTrombiForm","title":"UserTrombiForm","text":"

Bases: Form

"},{"location":"reference/trombi/views/#trombi.views.UserTrombiToolsView","title":"UserTrombiToolsView","text":"

Bases: LoginRequiredMixin, QuickNotifMixin, TrombiTabsMixin, TemplateView

Display a user's trombi tools.

"},{"location":"reference/trombi/views/#trombi.views.UserTrombiEditPicturesView","title":"UserTrombiEditPicturesView","text":"

Bases: TrombiTabsMixin, UserIsInATrombiMixin, UpdateView

"},{"location":"reference/trombi/views/#trombi.views.UserTrombiEditProfileView","title":"UserTrombiEditProfileView","text":"

Bases: QuickNotifMixin, TrombiTabsMixin, UserIsInATrombiMixin, UpdateView

"},{"location":"reference/trombi/views/#trombi.views.UserTrombiResetClubMembershipsView","title":"UserTrombiResetClubMembershipsView","text":"

Bases: UserIsInATrombiMixin, RedirectView

"},{"location":"reference/trombi/views/#trombi.views.UserTrombiDeleteMembershipView","title":"UserTrombiDeleteMembershipView","text":"

Bases: TrombiTabsMixin, CanEditMixin, DeleteView

"},{"location":"reference/trombi/views/#trombi.views.UserTrombiAddMembershipView","title":"UserTrombiAddMembershipView","text":"

Bases: TrombiTabsMixin, CreateView

"},{"location":"reference/trombi/views/#trombi.views.UserTrombiEditMembershipView","title":"UserTrombiEditMembershipView","text":"

Bases: CanEditMixin, TrombiTabsMixin, UpdateView

"},{"location":"reference/trombi/views/#trombi.views.UserTrombiProfileView","title":"UserTrombiProfileView","text":"

Bases: TrombiTabsMixin, DetailView

"},{"location":"reference/trombi/views/#trombi.views.TrombiCommentFormView","title":"TrombiCommentFormView","text":"

Bases: LoginRequiredMixin, View

Create/edit a trombi comment.

"},{"location":"reference/trombi/views/#trombi.views.TrombiCommentCreateView","title":"TrombiCommentCreateView","text":"

Bases: TrombiCommentFormView, CreateView

"},{"location":"reference/trombi/views/#trombi.views.TrombiCommentEditView","title":"TrombiCommentEditView","text":"

Bases: TrombiCommentFormView, CanViewMixin, UpdateView

"},{"location":"tutorial/devtools/","title":"Configurer son \u00e9diteur","text":"

Le projet n'est en aucun cas li\u00e9 \u00e0 un quelconque environnement de d\u00e9veloppement. Il est possible pour chacun de travailler avec les outils dont il a envie et d'utiliser l'\u00e9diteur de code avec lequel il est le plus \u00e0 l'aise.

Pour donner une id\u00e9e, Skia a \u00e9crit une \u00e9norme partie de projet avec l'\u00e9diteur Vim sur du GNU/Linux alors que Sli a utilis\u00e9 Sublime Text sur MacOS et que Mar\u00e9chal travaille avec PyCharm sur Windows muni de WSL Arch Linux btw.

"},{"location":"tutorial/devtools/#configurer-les-pre-commit-hooks","title":"Configurer les pre-commit hooks","text":"

La proc\u00e9dure habituelle pour contribuer au projet consiste \u00e0 commit des modifications, puis \u00e0 les push sur le d\u00e9p\u00f4t distant et \u00e0 ouvrir une pull request. Cette PR va faire tourner les outils de v\u00e9rification de la qualit\u00e9 de code. Si la v\u00e9rification \u00e9choue, la PR est bloqu\u00e9e, et il faut r\u00e9parer le probl\u00e8me (ce qui implique de push un micro-commit ou de push force sur la branche).

Dans l'id\u00e9al, on aimerait donc qu'il soit impossible d'oublier de faire tourner ces v\u00e9rifications. Pour \u00e7a, il existe un m\u00e9canisme : les pre-commits hooks. Ce sont des actions qui tournent automatiquement lorsque vous effectuez un git commit. Ces derni\u00e8res vont analyser et \u00e9ventuellement modifier le code, avant que Git n'ajoute effectivement le commit sur l'arbre git. Voyez \u00e7a comme une micro-CI qui tourne en local.

Les git hooks sont une fonctionnalit\u00e9 par d\u00e9faut de Git. Cependant, leur configuration peut-\u00eatre un peu emb\u00eatante si vous le faites manuellement. Pour g\u00e9rer \u00e7a plus simplement, nous utilisons le logiciel python pre-commit qui permet de contr\u00f4ler leur installation via un seul fichier de configuration, plac\u00e9 \u00e0 la racine du projet (plus pr\u00e9cis\u00e9ment, il s'agit du fichier .pre-commit-config.yaml).

Note

Les pre-commits sont \u00e9galement utilis\u00e9s dans la CI. Si ces derniers fonctionnent localement, vous avez la garantie que la pipeline ne sera pas fach\u00e9e. ;)

C'est une fonctionnalit\u00e9 de git lui-m\u00eame, mais c'est assez emb\u00eatant \u00e0 g\u00e9rer manuellement. Pour g\u00e9rer \u00e7a plus simplement, nous utilisons le logiciel python pre-commit qui permet de contr\u00f4ller leur installation via un fichier yaml.

Le logiciel est install\u00e9 par d\u00e9faut par uv. Il suffit ensuite de lancer :

uv run pre-commit install\n
Une fois que vous avez fait cette commande, pre-commit tournera automatiquement chaque fois que vous ferez un nouveau commit.

Il est \u00e9galement possible d'appeler soi-m\u00eame les pre-commits :

uv run pre-commit run --all-files\n
"},{"location":"tutorial/devtools/#configurer-ruff-pour-son-editeur","title":"Configurer Ruff pour son \u00e9diteur","text":"

Note

Ruff est inclus dans les d\u00e9pendances du projet. Si vous avez r\u00e9ussi \u00e0 terminer l'installation, vous n'avez donc pas de configuration suppl\u00e9mentaire \u00e0 effectuer.

Pour utiliser Ruff, placez-vous \u00e0 la racine du projet et lancez la commande suivante :

uv run ruff format  # pour formatter le code\nuv run ruff check  # pour linter le code\n

Ruff va alors faire son travail sur l'ensemble du projet puis vous dire si des documents ont \u00e9t\u00e9 reformat\u00e9s (si vous avez fait ruff format) ou bien s'il y a des erreurs \u00e0 r\u00e9parer (si vous avez faire ruff check).

Appeler Ruff en ligne de commandes avant de pousser votre code sur Github est une technique qui marche tr\u00e8s bien. Cependant, vous risquez de souvent l'oublier. Or, lorsque le code ne respecte pas les standards de qualit\u00e9, la pipeline bloque les PR sur les branches prot\u00e9g\u00e9es.

Pour \u00e9viter de vous faire r\u00e9guli\u00e8rement avoir, vous pouvez configurer votre \u00e9diteur pour que Ruff fasse son travail automatiquement \u00e0 chaque \u00e9dition d'un fichier. Nous tenterons de vous faire ici un r\u00e9sum\u00e9 pour deux \u00e9diteurs de textes populaires que sont VsCode et Sublime Text.

VsCodeSublime Text

Installez l'extension Ruff pour VsCode. Ensuite, ajoutez ceci dans votre configuration :

{\n    \"[python]\": {\n        \"editor.formatOnSave\": true,\n        \"editor.defaultFormatter\": \"charliermarsh.ruff\"\n    }\n}\n

Vous devez installer le plugin LSP-ruff. Suivez ensuite les instructions donn\u00e9es dans la description du plugin.

Dans la configuration de votre projet, ajoutez ceci:

{\n    \"settings\": {\n        \"lsp_format_on_save\": true, // Commun \u00e0 ruff et biome\n        \"LSP\": { \n            \"LSP-ruff\": {\n                \"enabled\": true,\n            }\n        }\n    }\n}\n

Si vous utilisez le plugin anaconda, pensez \u00e0 modifier les param\u00e8tres du linter pep8 pour \u00e9viter de recevoir des warnings dans le formatage de ruff comme ceci :

{\n    \"pep8_ignore\": [\n      \"E203\",\n      \"E266\",\n      \"E501\",\n      \"W503\"\n    ]\n}\n
"},{"location":"tutorial/devtools/#configurer-biome-pour-son-editeur","title":"Configurer Biome pour son \u00e9diteur","text":"

Note

Biome est inclus dans les d\u00e9pendances du projet. Si vous avez r\u00e9ussi \u00e0 terminer l'installation, vous n'avez donc pas de configuration suppl\u00e9mentaire \u00e0 effectuer.

Pour utiliser Biome, placez-vous \u00e0 la racine du projet et lancer la commande suivante:

    npx @biomejs/biome check # Pour checker le code avec le linter et le formater\n    npx @biomejs/biome check --write # Pour appliquer les changemnts\n

Biome va alors faire son travail sur l'ensemble du projet puis vous dire si des documents ont \u00e9t\u00e9 reformat\u00e9s (si vous avez fait npx @biomejs/biome format --write) ou bien s'il y a des erreurs \u00e0 r\u00e9parer (si vous avez faire npx @biomejs/biome lint) ou les deux (si vous avez fait npx @biomejs/biome check --write).

Appeler Biome en ligne de commandes avant de pousser votre code sur Github est une technique qui marche tr\u00e8s bien. Cependant, vous risquez de souvent l'oublier. Or, lorsque le code ne respecte pas les Biomes de qualit\u00e9, la pipeline bloque les PR sur les branches prot\u00e9g\u00e9es.

Pour \u00e9viter de vous faire r\u00e9guli\u00e8rement avoir, vous pouvez configurer votre \u00e9diteur pour que Biome fasse son travail automatiquement \u00e0 chaque \u00e9dition d'un fichier. Nous tenterons de vous faire ici un r\u00e9sum\u00e9 pour deux \u00e9diteurs de textes populaires que sont VsCode et Sublime Text.

VsCodeSublime Text

Biome est fourni par le plugin Biome.

Ensuite, ajoutez ceci dans votre configuration :

{\n  \"[javascript]\": {\n    \"editor.formatOnSave\": true,\n    \"editor.defaultFormatter\": \"biomejs.biome\"\n  },\n  \"[typescript]\": {\n    \"editor.formatOnSave\": true,\n    \"editor.defaultFormatter\": \"biomejs.biome\"\n  }\n}\n

Tout comme pour ruff, il suffit d'installer un plugin lsp LSP-biome.

Et enfin, dans la configuration de votre projet, ajouter les lignes suivantes :

{\n    \"settings\": {\n        \"lsp_format_on_save\": true, // Commun \u00e0 ruff et biome\n        \"LSP\": { \n            \"LSP-biome\": {\n                \"enabled\": true,\n            }\n        }\n    }\n}\n
"},{"location":"tutorial/etransaction/","title":"Etransactions","text":"

La boutique en ligne n\u00e9cessite une interaction avec la banque pour son fonctionnement.

Malheureusement, la mani\u00e8re dont cette interaction marche est trop complexe pour \u00eatre r\u00e9sum\u00e9e ici.

Nous ne pouvons donc que vous redirigez vers la doc du cr\u00e9dit agricole : https://www.ca-moncommerce.com/espace-client-mon-commerce/up2pay-e-transactions/ma-documentation/

"},{"location":"tutorial/fragments/","title":"Cr\u00e9er des fragments","text":"

Pour utiliser HTMX, on a besoin de renvoyer des fragments depuis le backend. Le truc, c'est que tout est optimis\u00e9 pour utiliser base.jinja qui est assez gros.

Dans beaucoup de sc\u00e9nario, on veut pouvoir renvoyer soit la vue compl\u00e8te, soit juste le fragment. En particulier quand on utilise l'attribut hx-history de htmx.

Pour rem\u00e9dier \u00e0 cela, il existe le mixin AllowFragment.

Une fois ajout\u00e9 \u00e0 une vue Django, il ajoute le boolean is_fragment dans les templates jinja. Sa valeur est True uniquement si HTMX envoie la requ\u00eate. Il est ensuite tr\u00e8s simple de faire un if/else pour h\u00e9riter de core/base_fragment.jinja au lieu de core/base.jinja dans cette situation.

Exemple d'utilisation d'une vue avec fragment:

from django.views.generic import TemplateView\nfrom core.views import AllowFragment\n\nclass FragmentView(AllowFragment, TemplateView):\n    template_name = \"my_template.jinja\"\n

Exemple de template (my_template.jinja)

{% if is_fragment %}\n  {% extends \"core/base_fragment.jinja\" %}\n{% else %}\n  {% extends \"core/base.jinja\" %}\n{% endif %}\n\n\n{% block title %}\n  {% trans %}My view with a fragment{% endtrans %}\n{% endblock %}\n\n{% block content %}\n  <h3>{% trans %}This will be a fragment when is_fragment is True{% endtrans %}\n{% endblock %}\n

"},{"location":"tutorial/groups/","title":"Gestion des groupes","text":""},{"location":"tutorial/groups/#un-peu-dhistoire","title":"Un peu d'histoire","text":"

Par d\u00e9faut, Django met \u00e0 disposition un mod\u00e8le Group, li\u00e9 par clef \u00e9trang\u00e8re au mod\u00e8le User. Pour cr\u00e9er un syst\u00e8me de gestion des groupes qui semblait plus appropri\u00e9 aux d\u00e9veloppeurs initiaux, un nouveau mod\u00e8le core.models.Group a \u00e9t\u00e9 cr\u00e9e, et la relation de clef \u00e9trang\u00e8re a \u00e9t\u00e9 modifi\u00e9e pour lier core.models.User \u00e0 ce dernier.

L'ancien mod\u00e8le Group \u00e9tait implicitement divis\u00e9 en deux cat\u00e9gories :

Cependant, ce nouveau syst\u00e8me s'\u00e9loignait trop du cadre de Django et a fini par devenir une g\u00eane. La v\u00e9rification des droits lors des op\u00e9rations est devenue une op\u00e9ration complexe et co\u00fbteuse en temps.

La gestion des groupes a donc \u00e9t\u00e9 modifi\u00e9e pour recoller un peu plus au cadre de Django. Toutefois, il n'a pas \u00e9t\u00e9 tent\u00e9 de revenir \u00e0 100% sur l'architecture pr\u00f4n\u00e9e par Django.

D'une part, cela repr\u00e9sentait un risque pour le succ\u00e8s de l'application de la migration sur la base de donn\u00e9es de production.

D'autre part, si une autre architecture a \u00e9t\u00e9 tent\u00e9e au d\u00e9but, ce n'\u00e9tait pas sans raison : ce que nous voulons mod\u00e9liser sur le site AE n'est pas compl\u00e8tement mod\u00e9lisable avec ce qu'offre Django. Il faut donc bien garder une surcouche au-dessus de l'authentification de Django. Tout le d\u00e9fi est de r\u00e9ussir \u00e0 maintenir cette surcouche aussi fine que possible sans limiter ce que nous voulons faire.

"},{"location":"tutorial/groups/#representation-en-base-de-donnees","title":"Repr\u00e9sentation en base de donn\u00e9es","text":"

Le mod\u00e8le core.models.Group a donc \u00e9t\u00e9 l\u00e9g\u00e8rement remani\u00e9 et la distinction entre groupes m\u00e9ta et groupes r\u00e9els a \u00e9t\u00e9 plus ou moins supprim\u00e9e. La liaison de clef \u00e9trang\u00e8re se fait toujours entre core.models.User et core.models.Group.

Cependant, il y a une subtilit\u00e9. Depuis le d\u00e9but, le mod\u00e8le Group de django n'a jamais disparu. En effet, lorsqu'un mod\u00e8le h\u00e9rite d'un mod\u00e8le qui n'est pas abstrait, Django garde les deux tables et les lie par une clef \u00e9trang\u00e8re unique de clef primaire \u00e0 clef primaire (pour plus de d\u00e9tail, lire la doc de django sur l'h\u00e9ritage de mod\u00e8le)

L'organisation r\u00e9elle de notre syst\u00e8me de groupes est donc la suivante :

---\ntitle: Repr\u00e9sentation des groupes\n---\nerDiagram\n    core_user }o..o{ core_group: core_user_groups\n    auth_group }o..o{ auth_permission: auth_group_permissions\n    core_group ||--|| auth_group: \"\"\n    core_user }o..o{ auth_permission :\"core_user_user_permissions\"\n\n    core_user {\n        int id PK\n        string username\n        string email\n        string first_name\n        etc etc\n    }\n    core_group {\n        int group_ptr_id PK,FK\n        string description\n        bool is_manually_manageable\n    }\n    auth_group {\n        int id PK\n        name string\n    }\n    auth_permission {\n        int id PK\n        string name\n    }

Cette organisation, rajoute une certaine complexit\u00e9, mais celle-ci est presque enti\u00e8rement g\u00e9r\u00e9e par django, ce qui fait que la gestion n'est pas tellement plus compliqu\u00e9e du point de vue du d\u00e9veloppeur.

Chaque fois qu'un queryset implique notre Group ou le Group de django, l'autre mod\u00e8le est automatiquement ajout\u00e9 \u00e0 la requ\u00eate par jointure. De cette fa\u00e7on, on peut manipuler l'un ou l'autre, sans m\u00eame se rendre que les tables sont dans des tables s\u00e9par\u00e9es.

Par exemple :

pythonSQL g\u00e9n\u00e9r\u00e9
from core.models import Group\n\nGroup.objects.all()\n
SELECT \"auth_group\".\"id\",\n       \"auth_group\".\"name\",\n       \"core_group\".\"group_ptr_id\",\n       \"core_group\".\"is_manually_manageable\",\n       \"core_group\".\"description\"\nFROM \"core_group\"\n         INNER JOIN \"auth_group\" ON (\"core_group\".\"group_ptr_id\" = \"auth_group\".\"id\")\n

Warning

Django r\u00e9ussit \u00e0 abstraire assez bien la logique relationnelle. Cependant, gardez bien en m\u00e9moire que ce n'est pas quelque chose de magique et que cette mani\u00e8re de faire a des limitations. Par exemple, il devient impossible de bulk_create des groupes.

"},{"location":"tutorial/groups/#la-definition-dun-groupe","title":"La d\u00e9finition d'un groupe","text":"

Un groupe est constitu\u00e9 des informations suivantes :

Si un groupe est g\u00e9rable manuellement, alors les administrateurs du site auront le droit d'assigner des utilisateurs \u00e0 ce groupe depuis l'interface d\u00e9di\u00e9e.

S'il n'est pas g\u00e9rable manuellement, on cache aux utilisateurs du site la gestion des membres de ce groupe. La gestion se fait alors uniquement \"sous le capot\", de mani\u00e8re automatique lors de certains \u00e9v\u00e8nements. Par exemple, lorsqu'un utilisateur rejoint un club, il est automatiquement ajout\u00e9 au groupe des membres du club. Lorsqu'il quitte le club, il est retir\u00e9 du groupe.

"},{"location":"tutorial/groups/#les-groupes-utilises","title":"Les groupes utilis\u00e9s","text":""},{"location":"tutorial/groups/#groupes-principaux","title":"Groupes principaux","text":"

Les groupes les plus notables g\u00e9rables par les administrateurs du site sont :

En plus de ces groupes, on peut noter :

Utilisation du groupe Public

Le groupe Public est un groupe particulier. Tout le monde faisant partie de ce groupe (m\u00eame les utilisateurs non-connect\u00e9s en sont implicitement consid\u00e9r\u00e9s comme membres), il ne doit pas \u00eatre utilis\u00e9 pour r\u00e9soudre les permissions d'une vue.

En revanche, il est utile pour attribuer une ressource \u00e0 tout le monde. Par exemple, un produit avec le groupe de vente Public est consid\u00e9r\u00e9 comme achetable par tous utilisateurs. S'il n'avait eu aucun group de vente, il n'aurait \u00e9t\u00e9 accessible \u00e0 personne.

"},{"location":"tutorial/groups/#groupes-de-club","title":"Groupes de club","text":"

Chaque club est associ\u00e9 \u00e0 deux groupes : le groupe des membres et le groupe du bureau.

Lorsqu'un utilisateur rejoint un club, il est automatiquement ajout\u00e9 au groupe des membres. S'il rejoint le club en tant que membre du bureau, il est \u00e9galement ajout\u00e9 au groupe du bureau.

Lorsqu'un utilisateur quitte le club, il est automatiquement retir\u00e9 des groupes li\u00e9s au club. S'il quitte le bureau, mais reste dans le club, il est retir\u00e9 du groupe du bureau, mais reste dans le groupe des membres.

"},{"location":"tutorial/groups/#groupes-de-ban","title":"Groupes de ban","text":"

Les groupes de ban sont une cat\u00e9gorie de groupes \u00e0 part, qui ne sont pas stock\u00e9s dans la m\u00eame table et qui ne sont pas g\u00e9r\u00e9s sur la m\u00eame interface que les autres groupes.

Les groupes de ban existants sont les suivants :

"},{"location":"tutorial/install-advanced/","title":"Installer le projet (avanc\u00e9)","text":"

Si le projet marche chez vous apr\u00e8s avoir suivi les \u00e9tapes donn\u00e9es dans la page pr\u00e9c\u00e9dente, alors vous pouvez d\u00e9velopper. Ce que nous nous vous avons pr\u00e9sent\u00e9 n'est absolument pas la m\u00eame configuration que celle du site, mais elle n'en est pas moins fonctionnelle.

Cependant, vous pourriez avoir envie de faire en sorte que votre environnement de d\u00e9veloppement soit encore plus proche de celui en production. Voici les \u00e9tapes \u00e0 suivre pour \u00e7a.

Tip

Configurer les d\u00e9pendances du projet peut demander beaucoup d'allers et retours entre votre r\u00e9pertoire projet et divers autres emplacements.

Vous pouvez gagner du temps en d\u00e9clarant un alias :

bash/zshnu
alias cdp=\"cd /repertoire/du/projet\"\n
alias cdp = cd /repertoire/du/projet\n

Chaque fois qu'on vous demandera de retourner au r\u00e9pertoire projet, vous aurez juste \u00e0 faire :

cdp\n
"},{"location":"tutorial/install-advanced/#installer-les-dependances-manquantes","title":"Installer les d\u00e9pendances manquantes","text":"

Pour installer compl\u00e8tement le projet, il va falloir quelques d\u00e9pendances en plus. Commencez par installer les d\u00e9pendances syst\u00e8me :

LinuxmacOS Debian/UbuntuArch Linux
sudo apt install postgresql redis libq-dev nginx\n
sudo pacman -S postgresql redis nginx\n
brew install postgresql redis lipbq nginx\nexport PATH=\"/usr/local/opt/libpq/bin:$PATH\"\nsource ~/.zshrc\n

Puis, installez les d\u00e9pendances n\u00e9cessaires en prod :

uv sync --group prod\n

Info

Certaines d\u00e9pendances peuvent \u00eatre un peu longues \u00e0 installer (notamment psycopg-c). C'est parce que ces d\u00e9pendances compilent certains modules \u00e0 l'installation.

"},{"location":"tutorial/install-advanced/#configurer-redis","title":"Configurer Redis","text":"

Redis est utilis\u00e9 comme cache. Assurez-vous qu'il tourne :

sudo systemctl redis status\n

Et s'il ne tourne pas, d\u00e9marrez-le :

sudo systemctl start redis\nsudo systemctl enable redis  # si vous voulez que redis d\u00e9marre automatiquement au boot\n

Puis ajoutez le code suivant \u00e0 la fin de votre fichier settings_custom.py :

CACHES = {\n    \"default\": {\n        \"BACKEND\": \"django.core.cache.backends.redis.RedisCache\",\n        \"LOCATION\": \"redis://127.0.0.1:6379\",\n    }\n}\n
"},{"location":"tutorial/install-advanced/#configurer-postgresql","title":"Configurer PostgreSQL","text":"

PostgreSQL est utilis\u00e9 comme base de donn\u00e9es.

Passez sur le compte de l'utilisateur postgres et lancez l'invite de commande sql :

sudo su - postgres\npsql\n

Puis configurez la base de donn\u00e9es :

CREATE DATABASE sith;\nCREATE USER sith WITH PASSWORD 'password';\n\nALTER ROLE sith SET client_encoding TO 'utf8';\nALTER ROLE sith SET default_transaction_isolation TO 'read committed';\nALTER ROLE sith SET timezone TO 'UTC';\n\nGRANT ALL PRIVILEGES ON DATABASE sith TO SITH;\n\\q\n

Si vous utilisez une version de PostgreSQL sup\u00e9rieure ou \u00e9gale \u00e0 15, vous devez ex\u00e9cuter une commande en plus, en \u00e9tant connect\u00e9 en tant que postgres :

psql -d sith -c \"GRANT ALL PRIVILEGES ON SCHEMA public to sith\";\n

Puis ajoutez le code suivant \u00e0 la fin de votre settings_custom.py :

DATABASES = {\n    \"default\": {\n        \"ENGINE\": \"django.db.backends.postgresql\",\n        \"NAME\": \"sith\",\n        \"USER\": \"sith\",\n        \"PASSWORD\": \"password\",\n        \"HOST\": \"localhost\",\n        \"PORT\": \"\",  # laissez ce champ vide pour que le choix du port soit automatique\n    }\n}\n

Enfin, cr\u00e9ez vos donn\u00e9es :

uv run ./manage.py populate\n

Note

N'oubliez de quitter la session de l'utilisateur postgres apr\u00e8s avoir configur\u00e9 la db.

"},{"location":"tutorial/install-advanced/#configurer-nginx","title":"Configurer nginx","text":"

Nginx est utilis\u00e9 comme reverse-proxy.

Warning

Nginx ne sert pas les fichiers de la m\u00eame mani\u00e8re que Django. Les fichiers statiques servis seront ceux du dossier /static, tels que g\u00e9n\u00e9r\u00e9s par les commandes collectstatic et compilestatic. Si vous changez du css ou du js sans faire tourner ces commandes, ces changements ne seront pas refl\u00e9t\u00e9s.

De mani\u00e8re g\u00e9n\u00e9rale, utiliser nginx en dev n'est pas tr\u00e8s utile, voire est g\u00eanant si vous travaillez sur le front. Ne vous emb\u00eatez pas avec \u00e7a, sauf par curiosit\u00e9 intellectuelle, ou bien si vous voulez tester sp\u00e9cifiquement des interactions avec le reverse proxy.

Placez-vous dans le r\u00e9pertoire /etc/nginx, et cr\u00e9ez les dossiers et fichiers n\u00e9cessaires :

cd /etc/nginx/\nsudo mkdir sites-enabled sites-available\nsudo touch sites-available/sith.conf\nsudo ln -s /etc/nginx/sites-available/sith.conf sites-enabled/sith.conf\n

Puis ouvrez le fichier sites-available/sith.conf et mettez-y le contenu suivant :

server {\n    listen 8000;\n\n    server_name _;\n\n    location /static/;\n        root /repertoire/du/projet;\n    }\n    location ~ ^/data/(products|com|club_logos)/ {\n        root /repertoire/du/projet;\n    }\n    location ~ ^/data/(SAS|profiles|users|.compressed|.thumbnails)/ {\n        # https://nginx.org/en/docs/http/ngx_http_core_module.html#internal\n        internal;\n        root /repertoire/du/projet;\n    }\n\n    location / {\n        proxy_pass http://127.0.0.1:8001;\n        include uwsgi_params;\n    }\n}\n

Ouvrez le fichier nginx.conf, et ajoutez la configuration suivante :

http {\n    # Toute la configuration\n    # \u00e9ventuellement d\u00e9j\u00e0 l\u00e0\n\n    include /etc/nginx/sites-enabled/sith.conf;\n}\n

V\u00e9rifiez que votre configuration est bonne :

sudo nginx -t\n

Si votre configuration n'est pas bonne, corrigez-la. Puis lancez ou relancez nginx :

sudo systemctl restart nginx\n

Dans votre settings_custom.py, remplacez DEBUG=True par DEBUG=False.

Enfin, d\u00e9marrez le serveur Django :

cd /repertoire/du/projet\nuv run ./manage.py runserver 8001\n

Et c'est bon, votre reverse-proxy est pr\u00eat \u00e0 tourner devant votre serveur. Nginx \u00e9coutera sur le port 8000. Toutes les requ\u00eates vers des fichiers statiques et les medias publiques seront seront servies directement par nginx. Toutes les autres requ\u00eates seront transmises au serveur django.

"},{"location":"tutorial/install-advanced/#mettre-a-jour-la-base-de-donnees-antispam","title":"Mettre \u00e0 jour la base de donn\u00e9es antispam","text":"

L'anti spam n\u00e9cessite d'\u00eatre \u00e0 jour par rapport \u00e0 des bases de donn\u00e9es externes. Il existe une commande pour \u00e7a qu'il faut lancer r\u00e9guli\u00e8rement. Lors de la mise en production, il est judicieux de configurer un cron pour la mettre \u00e0 jour au moins une fois par jour.

python manage.py update_spam_database\n
"},{"location":"tutorial/install/","title":"Installer le projet","text":""},{"location":"tutorial/install/#dependances-du-systeme","title":"D\u00e9pendances du syst\u00e8me","text":"

Certaines d\u00e9pendances sont n\u00e9cessaires niveau syst\u00e8me :

"},{"location":"tutorial/install/#installer-wsl","title":"Installer WSL","text":"

Si vous utilisez Windows, je suis navr\u00e9 de vous annoncer que, certaines d\u00e9pendances \u00e9tant uniquement disponibles sur des syt\u00e8mes UNIX, il n'est pas possible de d\u00e9velopper le site sur ce syst\u00e8me d'exploitation.

Heureusement, il existe une alternative qui ne requiert pas de d\u00e9sinstaller votre OS ni de mettre un dual boot sur votre ordinateur : WSL.

# dans un shell Windows\nwsl --install\n\n# afficher la liste des distribution disponible avec WSL\nwsl -l -o\n\n# installer WSL avec une distro (ubuntu conseill\u00e9)\nwsl --install -d <nom_distro>\n

Une fois WSL install\u00e9, mettez \u00e0 jour votre distribution et installez les d\u00e9pendances (voir la partie installation sous Ubuntu).

Pour acc\u00e9der au contenu d'un r\u00e9pertoire externe \u00e0 WSL, il suffit d'utiliser la commande suivante :

# oui c'est beau, simple et efficace\ncd /mnt/<la_lettre_du_disque>/vos/fichiers/comme/dhab\n

Note

\u00c0 ce stade, si vous avez r\u00e9ussi votre installation de WSL ou bien qu'il \u00e9tait d\u00e9j\u00e0 install\u00e9, vous pouvez effectuer la mise en place du projet en suivant les instructions pour votre distribution.

"},{"location":"tutorial/install/#installer-les-dependances","title":"Installer les d\u00e9pendances","text":"LinuxmacOS Debian/UbuntuArch Linux

Avant toute chose, assurez-vous que votre syst\u00e8me est \u00e0 jour :

sudo apt update\nsudo apt upgrade\n

Installez les d\u00e9pendances :

sudo apt install curl build-essential libssl-dev \\\nlibjpeg-dev zlib1g-dev npm libffi-dev pkg-config \\\ngettext git\ncurl -LsSf https://astral.sh/uv/install.sh | sh\n
sudo pacman -Syu  # on s'assure que les d\u00e9p\u00f4ts et le syst\u00e8me sont \u00e0 jour\n\nsudo pacman -S uv gcc git gettext pkgconf npm\n

Pour installer les d\u00e9pendances, il est fortement recommand\u00e9 d'installer le gestionnaire de paquets homebrew <https://brew.sh/index_fr>_. Il est \u00e9galement n\u00e9cessaire d'avoir install\u00e9 xcode

brew install git uv npm\n\n# Pour bien configurer gettext\nbrew link gettext # (suivez bien les instructions suppl\u00e9mentaires affich\u00e9es)\n

Note

Si vous rencontrez des erreurs lors de votre configuration, n'h\u00e9sitez pas \u00e0 v\u00e9rifier l'\u00e9tat de votre installation homebrew avec :code:brew doctor

Note

Python ne fait pas parti des d\u00e9pendances puisqu'il est automatiquement install\u00e9 par uv.

"},{"location":"tutorial/install/#finaliser-linstallation","title":"Finaliser l'installation","text":"

Clonez le projet (depuis votre console WSL, si vous utilisez WSL) et installez les d\u00e9pendances :

git clone https://github.com/ae-utbm/sith.git\ncd sith\n\n# Cr\u00e9ation de l'environnement et installation des d\u00e9pendances\nuv sync\nnpm install # D\u00e9pendances frontend\nuv run ./manage.py install_xapian\n

Note

La commande install_xapian est longue et affiche beaucoup de texte \u00e0 l'\u00e9cran. C'est normal, il ne faut pas avoir peur.

Maintenant que les d\u00e9pendances sont install\u00e9es, nous allons cr\u00e9er la base de donn\u00e9es, la remplir avec des donn\u00e9es de test, et compiler les traductions. Cependant, avant de faire cela, il est n\u00e9cessaire de modifier la configuration pour signifier que nous sommes en mode d\u00e9veloppement. Pour cela, nous allons cr\u00e9er un fichier sith/settings_custom.py et l'utiliser pour surcharger les settings de base.

echo \"DEBUG=True\" > sith/settings_custom.py\necho 'SITH_URL = \"localhost:8000\"' >> sith/settings_custom.py\n

Enfin, nous pouvons lancer les commandes suivantes :

# Pr\u00e9pare la base de donn\u00e9es\nuv run ./manage.py setup\n\n# Installe les traductions\nuv run ./manage.py compilemessages\n

Note

Pour \u00e9viter d'avoir \u00e0 utiliser la commande uv run syst\u00e9matiquement, il est possible de consulter direnv.

"},{"location":"tutorial/install/#demarrer-le-serveur-de-developpement","title":"D\u00e9marrer le serveur de d\u00e9veloppement","text":"

Il faut toujours avoir pr\u00e9alablement activ\u00e9 l'environnement virtuel comme fait plus haut et se placer \u00e0 la racine du projet. Il suffit ensuite d'utiliser cette commande :

uv run ./manage.py runserver\n

Note

Le serveur est alors accessible \u00e0 l'adresse http://localhost:8000 ou bien http://127.0.0.1:8000/.

Tip

Vous trouverez \u00e9galement, \u00e0 l'adresse http://localhost:8000/api/docs, une interface swagger, avec toutes les routes de l'API.

"},{"location":"tutorial/install/#generer-la-documentation","title":"G\u00e9n\u00e9rer la documentation","text":"

La documentation est automatiquement mise en ligne \u00e0 chaque envoi de code sur GitHub. Pour l'utiliser en local ou globalement pour la modifier, il existe une commande du site qui g\u00e9n\u00e8re la documentation et lance un serveur la rendant accessible \u00e0 l'adresse http://localhost:8080. Cette commande g\u00e9n\u00e8re la documentation \u00e0 chacune de ses modifications, inutile de relancer le serveur \u00e0 chaque fois.

uv run mkdocs serve\n
"},{"location":"tutorial/install/#lancer-les-tests","title":"Lancer les tests","text":"

Pour lancer les tests, il suffit d'utiliser la commande suivante :

# Lancer tous les tests\nuv run pytest\n\n# Lancer les tests de l'application core\nuv run pytest core\n\n# Lancer les tests de la classe UserRegistrationTest de core\nuv run pytest core/tests/tests_core.py::TestUserRegistration\n

Note

Certains tests sont un peu longs \u00e0 tourner. Pour ne faire tourner que les tests les plus rapides, vous pouvez ex\u00e9cutez pytest ainsi :

uv run pytest -m \"not slow\"\n\n# vous pouvez toujours faire comme au-dessus\nuv run pytest core -m \"not slow\"\n

A l'inverse, vous pouvez ne faire tourner que les tests lents en rempla\u00e7ant -m \"not slow\" par -m slow.

De cette mani\u00e8re, votre processus de d\u00e9veloppement devrait \u00eatre un peu plus fluide. Cependant, n'oubliez pas de bien faire tourner tous les tests avant de push un commit.

"},{"location":"tutorial/perms/","title":"Gestion des permissions","text":""},{"location":"tutorial/perms/#objectifs-du-systeme-de-permissions","title":"Objectifs du syst\u00e8me de permissions","text":"

Les permissions attendues sur le site sont relativement sp\u00e9cifiques. L'acc\u00e8s \u00e0 une ressource peut se faire selon un certain nombre de param\u00e8tres diff\u00e9rents :

L'\u00e9tat de la ressource Certaines ressources sont visibles par tous les cotisants (voire tous les utilisateurs), \u00e0 condition qu'elles aient pass\u00e9 une \u00e9tape de mod\u00e9ration. La visibilit\u00e9 des ressources non-mod\u00e9r\u00e9es n\u00e9cessite des permissions suppl\u00e9mentaires. L'appartenance \u00e0 un groupe Les groupes Root, Admin Com, Admin SAS, etc. sont associ\u00e9s \u00e0 des jeux de permissions. Par exemple, les membres du groupe Admin SAS ont tous les droits sur les ressources li\u00e9es au SAS : ils peuvent voir, cr\u00e9er, \u00e9diter, supprimer et \u00e9ventuellement mod\u00e9rer des images, des albums, des identifications de personnes... Il en va de m\u00eame avec les admins Com pour la communication, les admins p\u00e9dagogie pour le guide des UEs et ainsi de suite. Quant aux membres du groupe Root, ils ont tous les droits sur toutes les ressources du site. Le statut de la cotisation Les non-cotisants n'ont presque aucun droit sur les ressources du site (ils peuvent seulement en voir une poign\u00e9e), les anciens cotisants peuvent voir un grand nombre de ressources et les cotisants actuels ont la plupart des droits qui ne sont pas li\u00e9s \u00e0 un club ou \u00e0 l'administration du site. L'appartenance \u00e0 un club \u00catre dans un club donne le droit de voir la plupart des ressources li\u00e9es au club dans lequel ils sont ; \u00eatre dans le bureau du club donne en outre des droits d'\u00e9dition et de cr\u00e9ation sur ces ressources. \u00catre l'auteur ou le possesseur d'une ressource Certaines ressources, comme les nouvelles, enregistrent l'utilisateur qui les a cr\u00e9\u00e9es ; ce dernier a les droits de voir, de modifier et \u00e9ventuellement de supprimer ses ressources, quand bien m\u00eame elles ne seraient pas visibles pour les utilisateurs normaux (par exemple, parce qu'elles ne sont pas encore mod\u00e9r\u00e9es.)

Le syst\u00e8me de permissions inclus par d\u00e9faut dans django permet de mod\u00e9liser ais\u00e9ment l'acc\u00e8s \u00e0 des ressources au niveau de la table. Ainsi, il n'est pas compliqu\u00e9 de g\u00e9rer les permissions li\u00e9es aux groupes d'administration.

Cependant, une surcouche est n\u00e9cessaire d\u00e8s lors que l'on veut g\u00e9rer les droits li\u00e9s \u00e0 une ligne en particulier d'une table de la base de donn\u00e9es.

Nous essayons le plus possible de nous tenir aux fonctionnalit\u00e9s de django, sans pour autant h\u00e9siter \u00e0 nous rabattre sur notre propre surcouche d\u00e8s lors que les permissions attendues deviennent trop sp\u00e9cifiques pour \u00eatre g\u00e9r\u00e9es avec juste django.

Un peu d'histoire

Les permissions du site n'ont pas toujours \u00e9t\u00e9 g\u00e9r\u00e9es avec un m\u00e9lange de fonctionnalit\u00e9s de django et de notre propre code. Pendant tr\u00e8s longtemps, seule la surcouche \u00e9tait utilis\u00e9e, ce qui menait souvent \u00e0 des v\u00e9rifications de droits inefficaces et \u00e0 une gestion complexe de certaines parties qui auraient pu \u00eatre manipul\u00e9es beaucoup plus simplement.

En plus de \u00e7a, les permissions li\u00e9es \u00e0 la plupart des groupes se faisait de mani\u00e8re hardcod\u00e9e : plut\u00f4t que d'associer un groupe \u00e0 un jeu de permission et de faire une jointure en db sur les groupes de l'utilisateur ayant cette permissions, on conservait la clef primaire du groupe dans la config et on v\u00e9rifiait en dur dans le code que l'utilisateur \u00e9tait un des groupes voulus.

Ce syst\u00e8me poss\u00e9dait le triple d\u00e9savantage de prendre \u00e9norm\u00e9ment de temps, d'\u00eatre extr\u00eamement limit\u00e9 (de fait, si tout est hardcod\u00e9, on est oblig\u00e9 d'avoir le moins de groupes possibles pour que \u00e7a reste g\u00e9rable) et d'\u00eatre d\u00e9sesp\u00e9r\u00e9ment dangereux (par exemple : fin novembre 2024, une erreur dans le code a donn\u00e9 les acc\u00e8s \u00e0 la cr\u00e9ation des cotisations \u00e0 tout le monde ; mi-octobre 2019, le calcul des permissions des etickets pouvait faire tomber le site, cf. ce topic du forum)

"},{"location":"tutorial/perms/#acces-a-toutes-les-ressources-dune-table","title":"Acc\u00e8s \u00e0 toutes les ressources d'une table","text":"

G\u00e9rer ce genre d'acc\u00e8s (par exemple : voir toutes les nouvelles ou pouvoir supprimer n'importe quelle photo) est exactement le probl\u00e8me que le syst\u00e8me de permissions de django r\u00e9sout. Nous utilisons donc ce syst\u00e8me dans ce genre de situations.

Note

Nous d\u00e9crivons ci-dessous l'usage que nous faisons du syst\u00e8me de permissions de django, mais la seule source d'information compl\u00e8te et pleinement fiable sur le fonctionnement r\u00e9el de ce syst\u00e8me est la documentation de django.

"},{"location":"tutorial/perms/#permissions-dun-modele","title":"Permissions d'un mod\u00e8le","text":"

Par d\u00e9faut, django cr\u00e9e quatre permissions pour chaque table de la base de donn\u00e9es :

Ces permissions sont cr\u00e9\u00e9es au m\u00eame moment que le mod\u00e8le. Si la table existe en base de donn\u00e9es, ces permissions existent aussi.

Il est \u00e9galement possible de rajouter nos propres permissions, directement dans les options Meta du mod\u00e8le. Par exemple, prenons le mod\u00e8le suivant :

from django.db import models\n\nclass News(models.Model):\n    # ...\n\n    class Meta:\n        permissions = [\n            (\"moderate_news\", \"Can moderate news\"),\n            (\"view_unmoderated_news\", \"Can view non-moderated news\"),\n        ]\n

Ce dernier aura les permissions : view_news, add_news, change_news, delete_news, moderate_news et view_unmoderated_news.

"},{"location":"tutorial/perms/#utilisation-des-permissions-dun-modele","title":"Utilisation des permissions d'un mod\u00e8le","text":"

Pour v\u00e9rifier qu'un utilisateur a une permission, on utilise les fonctions suivantes :

Ces fonctions attendent un string suivant le format : <nom de l'application>.<nom de la permission>. Par exemple, la permission pour v\u00e9rifier qu'un utilisateur peut mod\u00e9rer une nouvelle sera : com.moderate_news.

Ces fonctions sont utilisables aussi bien dans les templates Jinja que dans le code Python :

JinjaPython
{% if user.has_perm(\"com.moderate_news\") %}\n    <form method=\"post\" action=\"{{ url(\"com:news_moderate\", news_id=387) }}\">\n        <input type=\"submit\" value=\"Mod\u00e9rer\" />\n    </form>\n{% endif %}\n
from com.models import News\nfrom core.models import User\n\n\nuser = User.objects.get(username=\"bibou\")\nnews = News.objects.get(id=387)\nif user.has_perm(\"com.moderate_news\"):\n    news.is_moderated = True\n    news.save()\nelse:\n    raise PermissionDenied\n

Pour utiliser ce syst\u00e8me de permissions dans une class-based view (c'est-\u00e0-dire la plus grande partie de nos vues), Django met \u00e0 disposition PermissionRequiredMixin, qui restreint l'acc\u00e8s \u00e0 la vue aux utilisateurs ayant la ou les permissions requises. Pour les vues sous forme de fonction, il y a le d\u00e9corateur permission_required.

Class-Based ViewFunction-based view
from com.models import News\n\nfrom django.contrib.auth.mixins import PermissionRequiredMixin\nfrom django.shortcuts import redirect\nfrom django.urls import reverse\nfrom django.views import View\nfrom django.views.generic.detail import SingleObjectMixin\n\nclass NewsModerateView(PermissionRequiredMixin, SingleObjectMixin, View):\n    model = News\n    pk_url_kwarg = \"news_id\"\n    permission_required = \"com.moderate_news\"\n    # On peut aussi fournir plusieurs permissions, par exemple :\n    # permission_required = [\"com.moderate_news\", \"com.delete_news\"]\n\n    def post(self, request, *args, **kwargs):\n        # Si nous sommes ici, nous pouvons \u00eatre certains que l'utilisateur\n        # a la permission requise\n        obj = self.get_object()\n        obj.is_moderated = True\n        obj.save()\n        return redirect(reverse(\"com:news_list\"))\n
from com.models import News\n\nfrom django.contrib.auth.decorators import permission_required\nfrom django.shortcuts import get_object_or_404, redirect\nfrom django.urls import reverse\nfrom django.views.decorators.http import require_POST\n\n@permission_required(\"com.moderate_news\")\n@require_POST\ndef moderate_news(request, news_id: int):\n    # Si nous sommes ici, nous pouvons \u00eatre certains que l'utilisateur\n    # a la permission requise\n    news = get_object_or_404(News, id=news_id)\n    news.is_moderated = True\n    news.save()\n    return redirect(reverse(\"com:news_list\"))\n
"},{"location":"tutorial/perms/#acces-a-des-elements-en-particulier","title":"Acc\u00e8s \u00e0 des \u00e9l\u00e9ments en particulier","text":""},{"location":"tutorial/perms/#acces-a-lauteur-de-la-ressource","title":"Acc\u00e8s \u00e0 l'auteur de la ressource","text":"

Dans ce genre de cas, on peut identifier trois acteurs possibles :

Dans ce genre de cas, on souhaite donc accorder l'acc\u00e8s aux utilisateurs qui ont la permission globale, selon le syst\u00e8me d\u00e9crit plus haut, ou bien \u00e0 l'auteur de la ressource.

Pour cela, nous avons le mixin PermissionOrAuthorRequired. Ce dernier va effectuer les m\u00eames v\u00e9rifications que PermissionRequiredMixin puis, si l'utilisateur n'a pas la permission requise, v\u00e9rifier s'il est l'auteur de la ressource.

from com.models import News\nfrom core.auth.mixins import PermissionOrAuthorRequiredMixin\n\nfrom django.views.generic import UpdateView\n\nclass NewsUpdateView(PermissionOrAuthorRequiredMixin, UpdateView):\n    model = News\n    pk_url_kwarg = \"news_id\"\n    permission_required = \"com.change_news\"\n    author_field = \"author\"  # (1)!\n
  1. Nom du champ du mod\u00e8le utilis\u00e9 comme clef \u00e9trang\u00e8re vers l'auteur. Par exemple, ici, la permission sera accord\u00e9e si l'utilisateur connect\u00e9 correspond \u00e0 l'utilisateur d\u00e9sign\u00e9 par News.author.
"},{"location":"tutorial/perms/#acces-en-fonction-de-regles-plus-complexes","title":"Acc\u00e8s en fonction de r\u00e8gles plus complexes","text":"

Tout ce que nous avons d\u00e9crit pr\u00e9c\u00e9demment permet de couvrir la plupart des cas simples. Cependant, il arrivera souvent que les permissions attendues soient plus complexes. Dans ce genre de cas, on rentre enti\u00e8rement dans notre surcouche.

"},{"location":"tutorial/perms/#implementation-dans-les-modeles","title":"Impl\u00e9mentation dans les mod\u00e8les","text":"

La gestion de ce type de permissions se fait directement par mod\u00e8le. Il en existe trois niveaux :

Chacune de ces permissions est v\u00e9rifi\u00e9e par une m\u00e9thode d\u00e9di\u00e9e de la classe User :

Ces m\u00e9thodes vont alors r\u00e9soudre les permissions dans cet ordre :

  1. Si l'objet poss\u00e8de une m\u00e9thode can_be_viewed_by(user) (ou can_be_edited_by(user), ou is_owned_by(user)) et que son appel renvoie True, l'utilisateur a la permission requise.
  2. Sinon, si le mod\u00e8le de l'objet poss\u00e8de un attribut view_groups (ou edit_groups, ou owner_group) et que l'utilisateur est dans l'un des groupes indiqu\u00e9s, il a la permission requise.
  3. Sinon, on regarde si l'utilisateur a la permission de niveau sup\u00e9rieur (les droits owner impliquent les droits d'\u00e9dition, et les droits d'\u00e9dition impliquent les droits de vue).
  4. Si aucune des \u00e9tapes si dessus ne permet d'\u00e9tablir que l'utilisateur n'a la permission requise, c'est qu'il ne l'a pas.

Voici un exemple d'impl\u00e9mentation de ce syst\u00e8me :

Avec les m\u00e9thodesAvec les groupes de permission
from django.db import models\nfrom django.utils.translation import gettext_lazy as _\n\nfrom core.models import User, Group\n\nclass Article(models.Model):\n\n    title = models.CharField(_(\"title\"), max_length=100)\n    content = models.TextField(_(\"content\"))\n\n    def is_owned_by(self, user):  # (1)!\n        return user.is_board_member\n\n    def can_be_edited_by(self, user):  # (2)!\n        return user.is_subscribed\n\n    def can_be_viewed_by(self, user):  # (3)!\n        return not user.is_anonymous\n
  1. Donne ou non les droits d'\u00e9dition des propri\u00e9t\u00e9s de l'objet. Ici, un utilisateur dans le bureau AE aura tous les droits sur cet objet
  2. Donne ou non les droits d'\u00e9dition de l'objet Ici, l'objet ne sera modifiable que par un utilisateur cotisant
  3. Donne ou non les droits de vue de l'objet Ici, l'objet n'est visible que par un utilisateur connect\u00e9

Note

Dans cet exemple, nous utilisons des permissions tr\u00e8s simples pour que vous puissiez constater le squelette de ce syst\u00e8me, plut\u00f4t que la logique de validation dans ce cas particulier.

En r\u00e9alit\u00e9, il serait ici beaucoup plus appropri\u00e9 de donner les permissions com.delete_article et com.change_article_properties (en cr\u00e9ant ce dernier s'il n'existe pas encore) au groupe du bureau AE, de donner \u00e9galement la permission com.change_article au groupe Cotisants et enfin de restreindre l'acc\u00e8s aux vues d'acc\u00e8s aux articles avec LoginRequiredMixin.

from django.db import models\nfrom django.conf import settings\nfrom django.utils.translation import gettext_lazy as _\n\nfrom core.models import User, Group\n\nclass Article(models.Model):\n    title = models.CharField(_(\"title\"), max_length=100)\n    content = models.TextField(_(\"content\"))\n\n    # relation one-to-many\n    owner_group = models.ForeignKey(  # (1)!\n        Group, related_name=\"owned_articles\", default=settings.SITH_GROUP_ROOT_ID\n    )\n\n    # relation many-to-many\n    edit_groups = models.ManyToManyField(  # (2)!\n        Group,\n        related_name=\"editable_articles\",\n        verbose_name=_(\"edit groups\"),\n        blank=True,\n    )\n\n    # relation many-to-many\n    view_groups = models.ManyToManyField(  # (3)!\n        Group,\n        related_name=\"viewable_articles\",\n        verbose_name=_(\"view groups\"),\n        blank=True,\n    )\n
  1. Groupe poss\u00e9dant l'objet Donne les droits d'\u00e9dition des propri\u00e9t\u00e9s de l'objet. Il ne peut y avoir qu'un seul groupe owner par objet.
  2. Tous les groupes ayant droit d'\u00e9dition sur l'objet. Il peut y avoir autant de groupes d'\u00e9dition que l'on veut par objet.
  3. Tous les groupes ayant droit de voir l'objet. Il peut y avoir autant de groupes de vue que l'on veut par objet.
"},{"location":"tutorial/perms/#application-dans-les-templates","title":"Application dans les templates","text":"

Il existe trois fonctions de base sur lesquelles reposent les v\u00e9rifications de permission. Elles sont disponibles dans le contexte par d\u00e9faut du moteur de template et peuvent \u00eatre utilis\u00e9es \u00e0 tout moment.

Voici un exemple d'utilisation dans un template :

{# ... #}\n{% if can_edit(club, user) %}\n    <a href=\"{{ url('club:tools', club_id=club.id) }}\">{{ club }}</a>\n{% endif %}\n
"},{"location":"tutorial/perms/#application-dans-les-vues","title":"Application dans les vues","text":"

G\u00e9n\u00e9ralement, les v\u00e9rifications de droits dans les templates se limitent aux urls \u00e0 afficher puisqu'il ne faut normalement pas mettre de logique autre que d'affichage \u00e0 l'int\u00e9rieur (en r\u00e9alit\u00e9, c'est un principe qu'on a beaucoup viol\u00e9, mais promis on le fera plus). C'est donc habituellement au niveau des vues que cela a lieu.

Pour cela, nous avons rajout\u00e9 des mixins \u00e0 h\u00e9riter lors de la cr\u00e9ation d'une vue bas\u00e9e sur une classe. Ces mixins ne sont compatibles qu'avec les classes r\u00e9cup\u00e9rant un objet ou une liste d'objet. Dans le cas d'un seul objet, une permission refus\u00e9e est lev\u00e9e lorsque l'utilisateur n'a pas le droit de visionner la page. Dans le cas d'une liste d'objet, le mixin filtre les objets non autoris\u00e9s et si aucun ne l'est, l'utilisateur recevra une liste vide d'objet.

Voici un exemple d'utilisation en reprenant l'objet Article cr\u00e9e pr\u00e9c\u00e9demment :

from django.views.generic import CreateView, DetailView\n\nfrom core.auth.mixins import CanViewMixin, CanCreateMixin\n\nfrom com.models import WeekmailArticle\n\n\n# Il est important de mettre le mixin avant la classe h\u00e9rit\u00e9e de Django\n# L'h\u00e9ritage multiple se fait de droite \u00e0 gauche et les mixins ont besoin\n# d'une classe de base pour fonctionner correctement.\nclass ArticlesDetailView(CanViewMixin, DetailView):\n    model = WeekmailArticle\n\n\n# M\u00eame chose pour une vue de cr\u00e9ation de l'objet Article\nclass ArticlesCreateView(CanCreateMixin, CreateView):\n    model = WeekmailArticle\n

Les mixins suivants sont impl\u00e9ment\u00e9s :

CanCreateMixin

L'usage de CanCreateMixin est dangereux et ne doit en aucun cas \u00eatre \u00e9tendu. La fa\u00e7on dont ce mixin marche est qu'il valide le formulaire de cr\u00e9ation et cr\u00e9e l'objet sans le persister en base de donn\u00e9es, puis v\u00e9rifie les droits sur cet objet non-persist\u00e9. Le danger de ce syst\u00e8me vient de multiples raisons :

Performance

Ce syst\u00e8me maison de permissions fonctionne et r\u00e9pond aux attentes de l'\u00e9poque de sa conception. Mais d'un point de vue performance, il est souvent plus que probl\u00e9matique. En effet, toutes les permissions sont dynamiquement calcul\u00e9es et n\u00e9cessitent plusieurs appels en base de donn\u00e9es qui ne se r\u00e9sument pas \u00e0 une \u00ab\u00a0simple\u00a0\u00bb jointure mais \u00e0 plusieurs requ\u00eates diff\u00e9rentes et difficiles \u00e0 optimiser. De plus, \u00e0 chaque calcul de permission, il est n\u00e9cessaire de recommencer tous les calculs depuis le d\u00e9but. La solution \u00e0 \u00e7a est de mettre du cache de session sur les tests effectu\u00e9s r\u00e9cemment, mais cela engendre son autre lot de probl\u00e8mes.

Sur une vue o\u00f9 on manipule un seul objet, passe encore. Mais sur les ListView, on peut arriver \u00e0 des temps de r\u00e9ponse extr\u00eamement \u00e9lev\u00e9s.

"},{"location":"tutorial/perms/#filtrage-des-querysets","title":"Filtrage des querysets","text":"

R\u00e9cup\u00e9rer tous les objets d'un queryset et v\u00e9rifier pour chacun que l'utilisateur a le droit de les voir peut-\u00eatre excessivement co\u00fbteux en ressources (cf. l'encart ci-dessus).

Lorsqu'il est n\u00e9cessaire de r\u00e9cup\u00e9rer un certain nombre d'objets depuis la base de donn\u00e9es, il est donc pr\u00e9f\u00e9rable de filtrer directement depuis le queryset.

Pour cela, certains mod\u00e8les, tels que Picture peuvent \u00eatre filtr\u00e9s avec la m\u00e9thode de queryset viewable_by. Cette derni\u00e8re s'utilise comme n'importe quelle autre m\u00e9thode de queryset :

from sas.models import Picture\nfrom core.models import User\n\nuser = User.objects.get(username=\"bibou\")\npictures = Picture.objects.viewable_by(user)\n

Le r\u00e9sultat de la requ\u00eate contiendra uniquement des \u00e9l\u00e9ments que l'utilisateur s\u00e9lectionn\u00e9 a effectivement le droit de voir.

Si vous d\u00e9sirez utiliser cette m\u00e9thode sur un mod\u00e8le qui ne la poss\u00e8de pas, il est relativement facile de l'\u00e9crire :

from typing import Self\n\nfrom django.db import models\n\nfrom core.models import User\n\n\nclass NewsQuerySet(models.QuerySet):  # (1)!\n    def viewable_by(self, user: User) -> Self:\n        if user.has_perm(\"com.view_unmoderated_news\"):\n            # si l'utilisateur peut tout voir, on retourne tout\n            return self\n        # sinon, on retourne les nouvelles mod\u00e9r\u00e9es ou dont l'utilisateur\n        # est l'auteur\n        return self.filter(\n            models.Q(is_moderated=True)\n            | models.Q(author=user)\n        )\n\n\nclass News(models.Model):\n    is_moderated = models.BooleanField(default=False)\n    author = models.ForeignKey(User, on_delete=models.PROTECT)\n    # ...\n\n    objects = NewsQuerySet.as_manager()  # (2)!\n\n    class Meta:\n        permissions = [(\"view_unmoderated_news\", \"Can view non moderated news\")]\n
  1. On cr\u00e9e un QuerySet maison, dans lequel on d\u00e9finit la m\u00e9thode viewable_by
  2. Puis, on attache ce QuerySet \u00e0 notre mod\u00e8le

Note

Pour plus d'informations sur la cr\u00e9ation de QuerySet personnalis\u00e9s, voir la documentation de django

"},{"location":"tutorial/perms/#api","title":"API","text":"

L'API utilise son propre syst\u00e8me de permissions. Ce n'est pas encore un autre syst\u00e8me en parall\u00e8le, mais un wrapper autour de notre syst\u00e8me de permissions, afin de l'adapter aux besoins de l'API.

En effet, l'interface attendue pour manipuler le plus ais\u00e9ment possible les permissions des routes d'API avec la librairie que nous utilisons est diff\u00e9rente de notre syst\u00e8me, tout en restant adaptable. (Pour plus de d\u00e9tail, voir la doc de la lib).

Si vous avez bien suivi ce qui a \u00e9t\u00e9 dit plus haut, vous ne devriez pas \u00eatre perdu, \u00e9tant donn\u00e9 que le syst\u00e8me de permissions de l'API utilise des noms assez similaires : IsInGroup, IsRoot, IsSubscriber... Vous pouvez trouver des exemples d'utilisation de ce syst\u00e8me dans cette partie.

"},{"location":"tutorial/structure/","title":"Structure du projet","text":""},{"location":"tutorial/structure/#la-structure-dun-projet-django","title":"La structure d'un projet Django","text":"

Un projet Django est structur\u00e9 en applications. Une application est un package Python contenant un ensemble de vues, de mod\u00e8les, de templates, etc. S\u00e9mantiquement, une application repr\u00e9sente un ensemble de fonctionnalit\u00e9s coh\u00e9rentes. Par exemple, dans notre cas, nous avons une application charg\u00e9e de la gestion des comptoirs, une autre de la gestion des clubs, une autre de la gestion du SAS, etc.

On trouve g\u00e9n\u00e9ralement dans un projet Django une application principale qui contient les fichiers de configuration du projet, les urls et \u00e9ventuellement des commandes d'administration.

"},{"location":"tutorial/structure/#arborescence-du-projet","title":"Arborescence du projet","text":"

Le code source du projet est organis\u00e9 comme suit :

sith/\n\u251c\u2500\u2500 .github/\n\u2502   \u251c\u2500\u2500 actions/ (1)\n\u2502   \u2514\u2500\u2500 workflows/ (2)\n\u251c\u2500\u2500 accounting/ (3)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 club/ (4)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 com/ (5)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 core/ (6)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 counter/ (7)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 docs/ (8)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 eboutic/ (9)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 election/ (10)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 forum/ (11)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 galaxy/ (12)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 launderette/ (13)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 locale/ (14)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 matmat/ (15)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 pedagogy/ (16)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 rootplace/ (17)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 sas/ (18)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 sith/ (19)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 subscription/ (20)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 trombi/ (21)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 antispam/ (22)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 staticfiles/ (23)\n\u2502   \u2514\u2500\u2500 ...\n\u2502\n\u251c\u2500\u2500 .coveragerc (24)\n\u251c\u2500\u2500 .envrc (25)\n\u251c\u2500\u2500 .gitattributes\n\u251c\u2500\u2500 .gitignore\n\u251c\u2500\u2500 .mailmap\n\u251c\u2500\u2500 manage.py (26)\n\u251c\u2500\u2500 mkdocs.yml (27)\n\u251c\u2500\u2500 uv.lock\n\u251c\u2500\u2500 pyproject.toml (28)\n\u251c\u2500\u2500 .venv/ (29)\n\u251c\u2500\u2500 .python-version (30)\n\u2514\u2500\u2500 README.md\n
  1. Dossier contenant certaines actions r\u00e9utilisables dans des workflows Github. Par exemple, l'action setup-project installe uv puis appelle configure l'environnement de d\u00e9veloppement
  2. Dossier contenant les fichiers de configuration des workflows Github. Par exemple, le workflow docs.yml compile et publie la documentation \u00e0 chaque push sur la branche master.
  3. Application de gestion de la comptabilit\u00e9.
  4. Application de gestion des clubs et de leurs membres.
  5. Application contenant les fonctionnalit\u00e9s destin\u00e9es aux responsables communication de l'AE.
  6. Application contenant la mod\u00e9lisation centrale du site. On en reparle plus loin sur cette page.
  7. Application de gestion des comptoirs, des permanences sur ces comptoirs et des transactions qui y sont effectu\u00e9es.
  8. Dossier contenant la documentation.
  9. Application de gestion de la boutique en ligne.
  10. Application de gestion des \u00e9lections.
  11. Application de gestion du forum
  12. Application de gestion de la galaxie ; la galaxie est un graphe des niveaux de proximit\u00e9 entre les diff\u00e9rents \u00e9tudiants.
  13. Gestion des machines \u00e0 laver de l'AE
  14. Dossier contenant les fichiers de traduction.
  15. Fonctionnalit\u00e9s de recherche d'utilisateurs.
  16. Le guide des UEs du site, sur lequel les utilisateurs peuvent \u00e9galement laisser leurs avis.
  17. Fonctionnalit\u00e9s utiles aux utilisateurs root.
  18. Le SAS, o\u00f9 l'on trouve toutes les photos de l'AE.
  19. Application principale du projet, contenant sa configuration.
  20. Gestion des cotisations des utilisateurs du site.
  21. Outil pour faciliter la fabrication des trombinoscopes de promo.
  22. Fonctionnalit\u00e9s pour g\u00e9rer le spam.
  23. Gestion des statics du site. Override le syst\u00e8me de statics de Django. Ajoute l'int\u00e9gration du scss et du bundler js de mani\u00e8re transparente pour l'utilisateur.
  24. Fichier de configuration de coverage.
  25. Fichier de configuration de direnv.
  26. Fichier g\u00e9n\u00e9r\u00e9 automatiquement par Django. C'est lui qui permet d'appeler des commandes de gestion du projet avec la syntaxe python ./manage.py <nom de la commande>
  27. Le fichier de configuration de la documentation, avec ses plugins et sa table des mati\u00e8res.
  28. Le fichier o\u00f9 sont d\u00e9clar\u00e9s les d\u00e9pendances et la configuration de certaines d'entre elles.
  29. Dossier d'environnement virtuel g\u00e9n\u00e9r\u00e9 par uv
  30. Fichier qui contr\u00f4le quel version de python utiliser pour le projet
"},{"location":"tutorial/structure/#lapplication-principale","title":"L'application principale","text":"

L'application principale du projet est le package sith. Ce package contient les fichiers de configuration du projet, la racine des urls.

Il est organis\u00e9 comme suit :

sith/\n\u251c\u2500\u2500 settings.py (1)\n\u251c\u2500\u2500 settings_custom.py (2)\n\u251c\u2500\u2500 toolbar_debug.py (3)\n\u251c\u2500\u2500 urls.py (4)\n\u2514\u2500\u2500 wsgi.py (5)\n
  1. Fichier de configuration du projet. Ce fichier contient les param\u00e8tres de configuration du projet. Par exemple, il contient la liste des applications install\u00e9es dans le projet.
  2. Configuration maison pour votre environnement. Toute variable que vous d\u00e9finissez dans ce fichier sera prioritaire sur la configuration donn\u00e9e dans settings.py.
  3. Configuration de la barre de debug. C'est inutilis\u00e9 en prod, mais c'est tr\u00e8s pratique en d\u00e9veloppement.
  4. Fichier de configuration des urls du projet.
  5. Fichier de configuration pour le serveur WSGI. WSGI est un protocole de communication entre le serveur et les applications. Ce fichier ne vous servira sans doute pas sur un environnement de d\u00e9veloppement, mais il est n\u00e9cessaire en production.
"},{"location":"tutorial/structure/#les-applications","title":"Les applications","text":"

Les applications sont des packages Python. Dans ce projet, les applications sont g\u00e9n\u00e9ralement organis\u00e9es comme suit :

.\n\u251c\u2500\u2500 migrations/ (1)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 templates/ (2)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 static/ (3)\n\u2502   \u2514\u2500\u2500 bundled/ (4)\n\u2502   \u2514\u2500\u2500 ...\n\u251c\u2500\u2500 api.py (5)\n\u251c\u2500\u2500 admin.py (6)\n\u251c\u2500\u2500 models.py (7)\n\u251c\u2500\u2500 tests.py (8)\n\u251c\u2500\u2500 schemas.py (9)\n\u251c\u2500\u2500 urls.py (10)\n\u2514\u2500\u2500 views.py (11)\n
  1. Dossier contenant les migrations de la base de donn\u00e9es. Les migrations sont des fichiers Python qui permettent de mettre \u00e0 jour la base de donn\u00e9es. cf. Gestion des migrations
  2. Dossier contenant les templates jinja utilis\u00e9s par cette application.
  3. Dossier contenant les fichiers statics (js, css, scss) qui sont r\u00e9cp\u00e9r\u00e9e par Django.
  4. Dossier contenant du js qui sera process avec le bundler javascript. Le contenu sera automatiquement process et accessible comme si \u00e7a avait \u00e9t\u00e9 plac\u00e9 dans le dossier static/bundled.
  5. Fichier contenant les routes d'API li\u00e9es \u00e0 cette application
  6. Fichier de configuration de l'interface d'administration. Ce fichier permet de d\u00e9clarer les mod\u00e8les de l'application dans l'interface d'administration.
  7. Fichier contenant les mod\u00e8les de l'application. Les mod\u00e8les sont des classes Python qui repr\u00e9sentent les tables de la base de donn\u00e9es.
  8. Fichier contenant les tests de l'application.
  9. Sch\u00e9mas de validation de donn\u00e9es utilis\u00e9s principalement dans l'API.
  10. Configuration des urls de l'application.
  11. Fichier contenant les vues de l'application. Dans les plus grosses applications, ce fichier peut \u00eatre remplac\u00e9 par un package views dans lequel les vues sont r\u00e9parties entre plusieurs fichiers.

L'organisation peut \u00e9ventuellement \u00eatre un peu diff\u00e9rente pour certaines applications, mais le principe g\u00e9n\u00e9ral est le m\u00eame.

"}]}