mirror of
				https://gitlab.com/animath/si/plateforme-corres2math.git
				synced 2025-10-25 18:33:10 +02:00 
			
		
		
		
	Compare commits
	
		
			217 Commits
		
	
	
		
			f341764d0b
			...
			improvemen
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 35a197be07 | ||
|  | d55671efd1 | ||
|  | bcdbc445c6 | ||
|  | 95a69f35d2 | ||
|  | eec941c816 | ||
|  | 64d471936f | ||
|  | 02c977264d | ||
|  | 8f86ea15c8 | ||
|  | 7628387158 | ||
|  | bf2feb9c35 | ||
|  | 969e53b712 | ||
|  | 84e149e8c9 | ||
|  | 9e6a3eb1ca | ||
|  | 1f9f60d880 | ||
|  | 205760f2e9 | ||
|  | 8f742b8e14 | ||
|  | 996d00c7f0 | ||
|  | 8f09ca5553 | ||
|  | b761670133 | ||
|  | f2ca4b7446 | ||
|  | 3faf1294d8 | ||
|  | dbcf15c4f3 | ||
|  | 10115a0419 | ||
|  | 833f9147ce | ||
|  | 55a727612c | ||
|  | de7f51d8da | ||
|  | c658a9767d | ||
|  | 982b61fe03 | ||
|  | ace1dbdc75 | ||
|  | 731f8711ff | ||
|  | d3e18a8fbb | ||
|  | ece1e800ab | ||
|  | 2e0028c063 | ||
|  | d304c3565c | ||
|  | 0fe1f9c348 | ||
|  | 9867e5e2a8 | ||
|  | 48b34e4362 | ||
|  | 10932d1cc5 | ||
|  | 02ca1d1efe | ||
|  | b0a3a22f83 | ||
|  | 4238e49c11 | ||
|  | c3c2c55aca | ||
|  | cba3e56fb8 | ||
|  | f146ae2dd2 | ||
|  | 04dd02b88a | ||
|  | 1ddf39f296 | ||
|  | fa368a399a | ||
|  | c35fb4e996 | ||
|  | b6cefc1519 | ||
|  | e73bb2d18b | ||
|  | f422212aea | ||
|  | 52763cb75a | ||
|  | aed9f457c3 | ||
|  | 6afa1ea40b | ||
|  | e98540a2a8 | ||
|  | 7353ecfd5f | ||
|  | 0a3fffe21e | ||
|  | bf32c34d4c | ||
|  | c62aa3ebd1 | ||
|  | 4181742133 | ||
|  | 25756fb2ef | ||
|  | 4c25ae2928 | ||
|  | 7ae2b152c6 | ||
|  | 682ef05110 | ||
|  | 4043d04826 | ||
|  | dee2152616 | ||
|  | be17d11581 | ||
|  | c80355f2bc | ||
|  | 62b883467c | ||
|  | 5fc46e74d2 | ||
|  | 61719cae1c | ||
|  | 463adc90d6 | ||
|  | a707219a9e | ||
|  | dd1238d44d | ||
|  | 7a8d05c58a | ||
|  | 4975d4ddb7 | ||
|  | c8029fdd7a | ||
|  | 82f8d61eb7 | ||
|  | 1392b50db1 | ||
|  | bcca41bfb6 | ||
|  | 4f1f4b0783 | ||
|  | 2f6a1f6e56 | ||
|  | d0007ee9b2 | ||
|  | 0d9c293443 | ||
|  | 2a282e366e | ||
|  | c11a40e054 | ||
|  | 2895fb88db | ||
|  | 4b14d3562f | ||
|  | 0ea51f47b3 | ||
|  | 0e9afc780c | ||
|  | 52c8a836f1 | ||
|  | 2273cbfbc6 | ||
|  | 95e1c4f821 | ||
|  | e9c56104df | ||
|  | e0f06179a0 | ||
|  | 244084a1fd | ||
|  | aa617a4bb6 | ||
|  | fac239111a | ||
|  | 9bb638f48d | ||
|  | d8ece66b23 | ||
|  | 6d4cd217b2 | ||
|  | 8236a9fe14 | ||
|  | 971169fe2c | ||
|  | 79836f0d94 | ||
|  | 3265c1fc1b | ||
|  | 09a2947095 | ||
|  | 316e4eefc1 | ||
|  | 238333a175 | ||
|  | db30b481a3 | ||
|  | 6647a284f4 | ||
|  | b33723efb3 | ||
|  | c9be07e45a | ||
|  | 642ac94b7a | ||
|  | 2ead8cdad5 | ||
|  | d73a92c7f7 | ||
|  | b80f98ef43 | ||
|  | 8c8ab4c0b7 | ||
|  | f2704b3f13 | ||
|  | 9159e7323d | ||
|  | ab315a69ed | ||
|  | 1d0246af8f | ||
|  | 9571cfc19d | ||
|  | 072ba3b2a3 | ||
|  | 42e1abd9aa | ||
|  | 2d467ef3af | ||
|  | 168bfc83c9 | ||
|  | d1f2402373 | ||
|  | bbeada74c7 | ||
|  | 95fd8a066f | ||
|  | 28e2fa10c3 | ||
|  | d940acb226 | ||
|  | 07451d41d3 | ||
|  | 2a9e0f2692 | ||
|  | 144577bd89 | ||
|  | 7fb811b87f | ||
|  | f05d2cca0b | ||
|  | df74f499ec | ||
|  | 9e097c99d0 | ||
|  | a2fd82ee96 | ||
|  | a7d455eeb1 | ||
|  | 19e25b5d1c | ||
|  | f1e5b63b75 | ||
|  | 980b2c9d8e | ||
|  | 11ebc6848b | ||
|  | f2cc5fb6cd | ||
|  | bbcfab1dcb | ||
|  | 6c59bf11be | ||
|  | b397d00011 | ||
|  | d420177011 | ||
|  | e9b70f351b | ||
|  | 3c8615b487 | ||
|  | 15d096a328 | ||
|  | acb718cdc0 | ||
|  | 691d3c1660 | ||
|  | b3db01c02e | ||
|  | a5e7325a5b | ||
|  | 1b1651c294 | ||
|  | 6424b4658a | ||
|  | fe29c61003 | ||
|  | 427786769f | ||
|  | 552ea17f7d | ||
|  | 6e8b6fbc66 | ||
|  | d62cb40acd | ||
|  | 6429c3cb92 | ||
|  | 37f4f7a721 | ||
|  | d9bd3688a7 | ||
|  | c29ed171b4 | ||
|  | 98abe243a7 | ||
|  | 92e66d75ab | ||
|  | 798c8db781 | ||
|  | 5c76e5f0db | ||
|  | 84a7ec6fb5 | ||
|  | 3b73edd6b0 | ||
|  | 366f0c40a0 | ||
|  | 4b8ed14180 | ||
|  | 0dece5abef | ||
|  | 88c4a6b218 | ||
|  | 56193dbecf | ||
|  | ed88c7cf6f | ||
|  | b2f2dfde5f | ||
|  | 2d62bec690 | ||
|  | 972902eb23 | ||
|  | c6c091b741 | ||
|  | 9ef35217d3 | ||
|  | c84d4151bb | ||
|  | 698fc46072 | ||
|  | 468295f648 | ||
|  | e1ca18085a | ||
|  | 471befb1f8 | ||
|  | 413525c6d2 | ||
|  | cd53d34443 | ||
|  | a2ce495a6b | ||
|  | 1979d33314 | ||
|  | cedb693c5b | ||
|  | 05528cd58c | ||
|  | ad2f1391a4 | ||
|  | b7bc0fe499 | ||
|  | fa0e4cd2f9 | ||
|  | 7a02355c2d | ||
|  | b68057d1a0 | ||
|  | a68a7b681d | ||
|  | f0459838d4 | ||
|  | 4975044046 | ||
|  | 8c70ef3835 | ||
|  | cce3a0a7df | ||
|  | 369c9e2fa2 | ||
|  | 1a56725795 | ||
|  | c7f74b68be | ||
|  | 016398dd8d | ||
|  | ae56203970 | ||
|  | 3741557200 | ||
|  | 0eef01ba9f | ||
|  | 21a71f37f8 | ||
|  | 7ae31f8a61 | ||
|  | ca4b0729e7 | ||
|  | f0eb6abe85 | ||
|  | c882d1ac7a | 
							
								
								
									
										6
									
								
								.bashrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.bashrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| PS1='\[\033[01;31m\]\u@\h\[\033[00m\]:\[\033[01;34m\]\w\[\033[00m\]\$ ' | ||||
|  | ||||
| alias ls='ls --color=auto' | ||||
| alias ll='ls -l' | ||||
| alias la='ls -A' | ||||
| alias l='ls -lACF' | ||||
							
								
								
									
										45
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										45
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1 +1,44 @@ | ||||
| .idea/ | ||||
| # Byte-compiled / optimized / DLL files | ||||
| dist | ||||
| build | ||||
| __pycache__ | ||||
| *.py[cod] | ||||
| *$py.class | ||||
| *.swp | ||||
| *.egg-info | ||||
| _build | ||||
| .tox | ||||
| .coverage | ||||
| coverage | ||||
|  | ||||
| # Translations | ||||
| *.mo | ||||
| *.pot | ||||
|  | ||||
| # Jupyter Notebook | ||||
| .ipynb_checkpoints | ||||
|  | ||||
| # Spyder project settings | ||||
| .spyderproject | ||||
| .spyproject | ||||
|  | ||||
| # Rope project settings | ||||
| .ropeproject | ||||
|  | ||||
| # PyCharm project settings | ||||
| .idea | ||||
|  | ||||
| # VSCode project settings | ||||
| .vscode | ||||
|  | ||||
| # Local data | ||||
| secrets.py | ||||
| *.log | ||||
| media/ | ||||
| # Virtualenv | ||||
| env/ | ||||
| venv/ | ||||
| db.sqlite3 | ||||
|  | ||||
| # Don't git index | ||||
| whoosh_index/ | ||||
|   | ||||
							
								
								
									
										27
									
								
								.gitlab-ci.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								.gitlab-ci.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| stages: | ||||
|   - test | ||||
|   - quality-assurance | ||||
|  | ||||
| py38: | ||||
|   stage: test | ||||
|   image: python:3.8-alpine | ||||
|   before_script: | ||||
|     - apk add --no-cache libmagic | ||||
|     - pip install tox --no-cache-dir | ||||
|   script: tox -e py38 | ||||
|  | ||||
| py39: | ||||
|   stage: test | ||||
|   image: python:3.9-alpine | ||||
|   before_script: | ||||
|     - apk add --no-cache gcc libmagic | ||||
|     - pip install tox --no-cache-dir | ||||
|   script: tox -e py39 | ||||
|  | ||||
| linters: | ||||
|   stage: quality-assurance | ||||
|   image: python:3-alpine | ||||
|   before_script: | ||||
|     - pip install tox --no-cache-dir | ||||
|   script: tox -e linters | ||||
|   allow_failure: true | ||||
| @@ -1,4 +0,0 @@ | ||||
| Options +FollowSymlinks | ||||
| Options -Indexes | ||||
| RewriteEngine On | ||||
| RewriteRule ^(.*)$ dispatcher.php?path=$1 [QSA,L] | ||||
							
								
								
									
										56
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										56
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,39 +1,35 @@ | ||||
| FROM php:7.3-apache as plateforme-builder | ||||
| FROM python:3.8-alpine | ||||
|  | ||||
| # Enabling apache rewrite mod | ||||
| RUN a2enmod rewrite | ||||
| ENV PYTHONUNBUFFERED 1 | ||||
| ENV DJANGO_ALLOW_ASYNC_UNSAFE 1 | ||||
|  | ||||
| RUN apt clean && apt update && apt upgrade -y | ||||
| # Install LaTeX requirements | ||||
| RUN apk add --no-cache gettext nginx gcc libc-dev libffi-dev libxml2-dev libxslt-dev postgresql-dev libmagic | ||||
|  | ||||
| # Install MySQL drivers | ||||
| RUN docker-php-ext-install pdo_mysql \ | ||||
|     &&  docker-php-ext-enable pdo_mysql | ||||
| RUN apk add --no-cache bash | ||||
|  | ||||
| # Install zip utilities | ||||
| RUN apt install -y libzip-dev zip \ | ||||
|     && docker-php-ext-configure zip --with-libzip \ | ||||
|     && docker-php-ext-install zip \ | ||||
|     &&  docker-php-ext-enable zip | ||||
| RUN mkdir /code | ||||
| WORKDIR /code | ||||
| COPY requirements.txt /code/requirements.txt | ||||
| RUN pip install -r requirements.txt --no-cache-dir | ||||
|  | ||||
| # Setup locales | ||||
| RUN apt install locales locales-all -y && locale-gen fr_FR.UTF-8 | ||||
| ENV LANG fr_FR.UTF-8 | ||||
| ENV LANGUAGE fr_FR:fr | ||||
| ENV LC_ALL fr_FR.UTF-8 | ||||
| COPY . /code/ | ||||
|  | ||||
| # Setup timezone | ||||
| RUN echo Europe/Paris > /etc/timezone \ | ||||
|     && ln -sf /usr/share/zoneinfo/Europe/Paris /etc/localtime \ | ||||
|     && dpkg-reconfigure -f noninteractive tzdata | ||||
| RUN python manage.py collectstatic --noinput && \ | ||||
|         python manage.py compilemessages | ||||
|  | ||||
| # Setup mailing | ||||
| RUN apt install -yq msmtp ca-certificates | ||||
| COPY setup/msmtprc /etc/msmtprc | ||||
| RUN echo "sendmail_path=msmtp -t" >> /usr/local/etc/php/conf.d/php-sendmail.ini | ||||
| # Configure nginx | ||||
| RUN mkdir /run/nginx | ||||
| RUN ln -sf /dev/stdout /var/log/nginx/access.log && ln -sf /dev/stderr /var/log/nginx/error.log | ||||
| RUN ln -sf /code/nginx_corres2math.conf /etc/nginx/conf.d/corres2math.conf | ||||
| RUN rm /etc/nginx/conf.d/default.conf | ||||
|  | ||||
| # Setting environment | ||||
| ENV CORRES2MATH_LOCAL_PATH /var/www/html | ||||
| ENV CORRES2MATH_MAIL_DOMAIN correspondances-maths.fr | ||||
| ENV CORRES2MATH_URL_BASE https://inscription.correspondances-maths.fr | ||||
| RUN crontab /code/corres2math.cron | ||||
|  | ||||
| RUN chmod 0777 /var/www/html | ||||
| # With a bashrc, the shell is better | ||||
| RUN ln -s /code/.bashrc /root/.bashrc | ||||
|  | ||||
| ENTRYPOINT ["/code/entrypoint.sh"] | ||||
| EXPOSE 80 | ||||
|  | ||||
| CMD ["./manage.py", "shell_plus", "--ptpython"] | ||||
|   | ||||
							
								
								
									
										674
									
								
								LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										674
									
								
								LICENSE
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,674 @@ | ||||
|                     GNU GENERAL PUBLIC LICENSE | ||||
|                        Version 3, 29 June 2007 | ||||
|  | ||||
|  Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/> | ||||
|  Everyone is permitted to copy and distribute verbatim copies | ||||
|  of this license document, but changing it is not allowed. | ||||
|  | ||||
|                             Preamble | ||||
|  | ||||
|   The GNU General Public License is a free, copyleft license for | ||||
| software and other kinds of works. | ||||
|  | ||||
|   The licenses for most software and other practical works are designed | ||||
| to take away your freedom to share and change the works.  By contrast, | ||||
| the GNU General Public License is intended to guarantee your freedom to | ||||
| share and change all versions of a program--to make sure it remains free | ||||
| software for all its users.  We, the Free Software Foundation, use the | ||||
| GNU General Public License for most of our software; it applies also to | ||||
| any other work released this way by its authors.  You can apply it to | ||||
| your programs, too. | ||||
|  | ||||
|   When we speak of free software, we are referring to freedom, not | ||||
| price.  Our General Public Licenses are designed to make sure that you | ||||
| have the freedom to distribute copies of free software (and charge for | ||||
| them if you wish), that you receive source code or can get it if you | ||||
| want it, that you can change the software or use pieces of it in new | ||||
| free programs, and that you know you can do these things. | ||||
|  | ||||
|   To protect your rights, we need to prevent others from denying you | ||||
| these rights or asking you to surrender the rights.  Therefore, you have | ||||
| certain responsibilities if you distribute copies of the software, or if | ||||
| you modify it: responsibilities to respect the freedom of others. | ||||
|  | ||||
|   For example, if you distribute copies of such a program, whether | ||||
| gratis or for a fee, you must pass on to the recipients the same | ||||
| freedoms that you received.  You must make sure that they, too, receive | ||||
| or can get the source code.  And you must show them these terms so they | ||||
| know their rights. | ||||
|  | ||||
|   Developers that use the GNU GPL protect your rights with two steps: | ||||
| (1) assert copyright on the software, and (2) offer you this License | ||||
| giving you legal permission to copy, distribute and/or modify it. | ||||
|  | ||||
|   For the developers' and authors' protection, the GPL clearly explains | ||||
| that there is no warranty for this free software.  For both users' and | ||||
| authors' sake, the GPL requires that modified versions be marked as | ||||
| changed, so that their problems will not be attributed erroneously to | ||||
| authors of previous versions. | ||||
|  | ||||
|   Some devices are designed to deny users access to install or run | ||||
| modified versions of the software inside them, although the manufacturer | ||||
| can do so.  This is fundamentally incompatible with the aim of | ||||
| protecting users' freedom to change the software.  The systematic | ||||
| pattern of such abuse occurs in the area of products for individuals to | ||||
| use, which is precisely where it is most unacceptable.  Therefore, we | ||||
| have designed this version of the GPL to prohibit the practice for those | ||||
| products.  If such problems arise substantially in other domains, we | ||||
| stand ready to extend this provision to those domains in future versions | ||||
| of the GPL, as needed to protect the freedom of users. | ||||
|  | ||||
|   Finally, every program is threatened constantly by software patents. | ||||
| States should not allow patents to restrict development and use of | ||||
| software on general-purpose computers, but in those that do, we wish to | ||||
| avoid the special danger that patents applied to a free program could | ||||
| make it effectively proprietary.  To prevent this, the GPL assures that | ||||
| patents cannot be used to render the program non-free. | ||||
|  | ||||
|   The precise terms and conditions for copying, distribution and | ||||
| modification follow. | ||||
|  | ||||
|                        TERMS AND CONDITIONS | ||||
|  | ||||
|   0. Definitions. | ||||
|  | ||||
|   "This License" refers to version 3 of the GNU General Public License. | ||||
|  | ||||
|   "Copyright" also means copyright-like laws that apply to other kinds of | ||||
| works, such as semiconductor masks. | ||||
|  | ||||
|   "The Program" refers to any copyrightable work licensed under this | ||||
| License.  Each licensee is addressed as "you".  "Licensees" and | ||||
| "recipients" may be individuals or organizations. | ||||
|  | ||||
|   To "modify" a work means to copy from or adapt all or part of the work | ||||
| in a fashion requiring copyright permission, other than the making of an | ||||
| exact copy.  The resulting work is called a "modified version" of the | ||||
| earlier work or a work "based on" the earlier work. | ||||
|  | ||||
|   A "covered work" means either the unmodified Program or a work based | ||||
| on the Program. | ||||
|  | ||||
|   To "propagate" a work means to do anything with it that, without | ||||
| permission, would make you directly or secondarily liable for | ||||
| infringement under applicable copyright law, except executing it on a | ||||
| computer or modifying a private copy.  Propagation includes copying, | ||||
| distribution (with or without modification), making available to the | ||||
| public, and in some countries other activities as well. | ||||
|  | ||||
|   To "convey" a work means any kind of propagation that enables other | ||||
| parties to make or receive copies.  Mere interaction with a user through | ||||
| a computer network, with no transfer of a copy, is not conveying. | ||||
|  | ||||
|   An interactive user interface displays "Appropriate Legal Notices" | ||||
| to the extent that it includes a convenient and prominently visible | ||||
| feature that (1) displays an appropriate copyright notice, and (2) | ||||
| tells the user that there is no warranty for the work (except to the | ||||
| extent that warranties are provided), that licensees may convey the | ||||
| work under this License, and how to view a copy of this License.  If | ||||
| the interface presents a list of user commands or options, such as a | ||||
| menu, a prominent item in the list meets this criterion. | ||||
|  | ||||
|   1. Source Code. | ||||
|  | ||||
|   The "source code" for a work means the preferred form of the work | ||||
| for making modifications to it.  "Object code" means any non-source | ||||
| form of a work. | ||||
|  | ||||
|   A "Standard Interface" means an interface that either is an official | ||||
| standard defined by a recognized standards body, or, in the case of | ||||
| interfaces specified for a particular programming language, one that | ||||
| is widely used among developers working in that language. | ||||
|  | ||||
|   The "System Libraries" of an executable work include anything, other | ||||
| than the work as a whole, that (a) is included in the normal form of | ||||
| packaging a Major Component, but which is not part of that Major | ||||
| Component, and (b) serves only to enable use of the work with that | ||||
| Major Component, or to implement a Standard Interface for which an | ||||
| implementation is available to the public in source code form.  A | ||||
| "Major Component", in this context, means a major essential component | ||||
| (kernel, window system, and so on) of the specific operating system | ||||
| (if any) on which the executable work runs, or a compiler used to | ||||
| produce the work, or an object code interpreter used to run it. | ||||
|  | ||||
|   The "Corresponding Source" for a work in object code form means all | ||||
| the source code needed to generate, install, and (for an executable | ||||
| work) run the object code and to modify the work, including scripts to | ||||
| control those activities.  However, it does not include the work's | ||||
| System Libraries, or general-purpose tools or generally available free | ||||
| programs which are used unmodified in performing those activities but | ||||
| which are not part of the work.  For example, Corresponding Source | ||||
| includes interface definition files associated with source files for | ||||
| the work, and the source code for shared libraries and dynamically | ||||
| linked subprograms that the work is specifically designed to require, | ||||
| such as by intimate data communication or control flow between those | ||||
| subprograms and other parts of the work. | ||||
|  | ||||
|   The Corresponding Source need not include anything that users | ||||
| can regenerate automatically from other parts of the Corresponding | ||||
| Source. | ||||
|  | ||||
|   The Corresponding Source for a work in source code form is that | ||||
| same work. | ||||
|  | ||||
|   2. Basic Permissions. | ||||
|  | ||||
|   All rights granted under this License are granted for the term of | ||||
| copyright on the Program, and are irrevocable provided the stated | ||||
| conditions are met.  This License explicitly affirms your unlimited | ||||
| permission to run the unmodified Program.  The output from running a | ||||
| covered work is covered by this License only if the output, given its | ||||
| content, constitutes a covered work.  This License acknowledges your | ||||
| rights of fair use or other equivalent, as provided by copyright law. | ||||
|  | ||||
|   You may make, run and propagate covered works that you do not | ||||
| convey, without conditions so long as your license otherwise remains | ||||
| in force.  You may convey covered works to others for the sole purpose | ||||
| of having them make modifications exclusively for you, or provide you | ||||
| with facilities for running those works, provided that you comply with | ||||
| the terms of this License in conveying all material for which you do | ||||
| not control copyright.  Those thus making or running the covered works | ||||
| for you must do so exclusively on your behalf, under your direction | ||||
| and control, on terms that prohibit them from making any copies of | ||||
| your copyrighted material outside their relationship with you. | ||||
|  | ||||
|   Conveying under any other circumstances is permitted solely under | ||||
| the conditions stated below.  Sublicensing is not allowed; section 10 | ||||
| makes it unnecessary. | ||||
|  | ||||
|   3. Protecting Users' Legal Rights From Anti-Circumvention Law. | ||||
|  | ||||
|   No covered work shall be deemed part of an effective technological | ||||
| measure under any applicable law fulfilling obligations under article | ||||
| 11 of the WIPO copyright treaty adopted on 20 December 1996, or | ||||
| similar laws prohibiting or restricting circumvention of such | ||||
| measures. | ||||
|  | ||||
|   When you convey a covered work, you waive any legal power to forbid | ||||
| circumvention of technological measures to the extent such circumvention | ||||
| is effected by exercising rights under this License with respect to | ||||
| the covered work, and you disclaim any intention to limit operation or | ||||
| modification of the work as a means of enforcing, against the work's | ||||
| users, your or third parties' legal rights to forbid circumvention of | ||||
| technological measures. | ||||
|  | ||||
|   4. Conveying Verbatim Copies. | ||||
|  | ||||
|   You may convey verbatim copies of the Program's source code as you | ||||
| receive it, in any medium, provided that you conspicuously and | ||||
| appropriately publish on each copy an appropriate copyright notice; | ||||
| keep intact all notices stating that this License and any | ||||
| non-permissive terms added in accord with section 7 apply to the code; | ||||
| keep intact all notices of the absence of any warranty; and give all | ||||
| recipients a copy of this License along with the Program. | ||||
|  | ||||
|   You may charge any price or no price for each copy that you convey, | ||||
| and you may offer support or warranty protection for a fee. | ||||
|  | ||||
|   5. Conveying Modified Source Versions. | ||||
|  | ||||
|   You may convey a work based on the Program, or the modifications to | ||||
| produce it from the Program, in the form of source code under the | ||||
| terms of section 4, provided that you also meet all of these conditions: | ||||
|  | ||||
|     a) The work must carry prominent notices stating that you modified | ||||
|     it, and giving a relevant date. | ||||
|  | ||||
|     b) The work must carry prominent notices stating that it is | ||||
|     released under this License and any conditions added under section | ||||
|     7.  This requirement modifies the requirement in section 4 to | ||||
|     "keep intact all notices". | ||||
|  | ||||
|     c) You must license the entire work, as a whole, under this | ||||
|     License to anyone who comes into possession of a copy.  This | ||||
|     License will therefore apply, along with any applicable section 7 | ||||
|     additional terms, to the whole of the work, and all its parts, | ||||
|     regardless of how they are packaged.  This License gives no | ||||
|     permission to license the work in any other way, but it does not | ||||
|     invalidate such permission if you have separately received it. | ||||
|  | ||||
|     d) If the work has interactive user interfaces, each must display | ||||
|     Appropriate Legal Notices; however, if the Program has interactive | ||||
|     interfaces that do not display Appropriate Legal Notices, your | ||||
|     work need not make them do so. | ||||
|  | ||||
|   A compilation of a covered work with other separate and independent | ||||
| works, which are not by their nature extensions of the covered work, | ||||
| and which are not combined with it such as to form a larger program, | ||||
| in or on a volume of a storage or distribution medium, is called an | ||||
| "aggregate" if the compilation and its resulting copyright are not | ||||
| used to limit the access or legal rights of the compilation's users | ||||
| beyond what the individual works permit.  Inclusion of a covered work | ||||
| in an aggregate does not cause this License to apply to the other | ||||
| parts of the aggregate. | ||||
|  | ||||
|   6. Conveying Non-Source Forms. | ||||
|  | ||||
|   You may convey a covered work in object code form under the terms | ||||
| of sections 4 and 5, provided that you also convey the | ||||
| machine-readable Corresponding Source under the terms of this License, | ||||
| in one of these ways: | ||||
|  | ||||
|     a) Convey the object code in, or embodied in, a physical product | ||||
|     (including a physical distribution medium), accompanied by the | ||||
|     Corresponding Source fixed on a durable physical medium | ||||
|     customarily used for software interchange. | ||||
|  | ||||
|     b) Convey the object code in, or embodied in, a physical product | ||||
|     (including a physical distribution medium), accompanied by a | ||||
|     written offer, valid for at least three years and valid for as | ||||
|     long as you offer spare parts or customer support for that product | ||||
|     model, to give anyone who possesses the object code either (1) a | ||||
|     copy of the Corresponding Source for all the software in the | ||||
|     product that is covered by this License, on a durable physical | ||||
|     medium customarily used for software interchange, for a price no | ||||
|     more than your reasonable cost of physically performing this | ||||
|     conveying of source, or (2) access to copy the | ||||
|     Corresponding Source from a network server at no charge. | ||||
|  | ||||
|     c) Convey individual copies of the object code with a copy of the | ||||
|     written offer to provide the Corresponding Source.  This | ||||
|     alternative is allowed only occasionally and noncommercially, and | ||||
|     only if you received the object code with such an offer, in accord | ||||
|     with subsection 6b. | ||||
|  | ||||
|     d) Convey the object code by offering access from a designated | ||||
|     place (gratis or for a charge), and offer equivalent access to the | ||||
|     Corresponding Source in the same way through the same place at no | ||||
|     further charge.  You need not require recipients to copy the | ||||
|     Corresponding Source along with the object code.  If the place to | ||||
|     copy the object code is a network server, the Corresponding Source | ||||
|     may be on a different server (operated by you or a third party) | ||||
|     that supports equivalent copying facilities, provided you maintain | ||||
|     clear directions next to the object code saying where to find the | ||||
|     Corresponding Source.  Regardless of what server hosts the | ||||
|     Corresponding Source, you remain obligated to ensure that it is | ||||
|     available for as long as needed to satisfy these requirements. | ||||
|  | ||||
|     e) Convey the object code using peer-to-peer transmission, provided | ||||
|     you inform other peers where the object code and Corresponding | ||||
|     Source of the work are being offered to the general public at no | ||||
|     charge under subsection 6d. | ||||
|  | ||||
|   A separable portion of the object code, whose source code is excluded | ||||
| from the Corresponding Source as a System Library, need not be | ||||
| included in conveying the object code work. | ||||
|  | ||||
|   A "User Product" is either (1) a "consumer product", which means any | ||||
| tangible personal property which is normally used for personal, family, | ||||
| or household purposes, or (2) anything designed or sold for incorporation | ||||
| into a dwelling.  In determining whether a product is a consumer product, | ||||
| doubtful cases shall be resolved in favor of coverage.  For a particular | ||||
| product received by a particular user, "normally used" refers to a | ||||
| typical or common use of that class of product, regardless of the status | ||||
| of the particular user or of the way in which the particular user | ||||
| actually uses, or expects or is expected to use, the product.  A product | ||||
| is a consumer product regardless of whether the product has substantial | ||||
| commercial, industrial or non-consumer uses, unless such uses represent | ||||
| the only significant mode of use of the product. | ||||
|  | ||||
|   "Installation Information" for a User Product means any methods, | ||||
| procedures, authorization keys, or other information required to install | ||||
| and execute modified versions of a covered work in that User Product from | ||||
| a modified version of its Corresponding Source.  The information must | ||||
| suffice to ensure that the continued functioning of the modified object | ||||
| code is in no case prevented or interfered with solely because | ||||
| modification has been made. | ||||
|  | ||||
|   If you convey an object code work under this section in, or with, or | ||||
| specifically for use in, a User Product, and the conveying occurs as | ||||
| part of a transaction in which the right of possession and use of the | ||||
| User Product is transferred to the recipient in perpetuity or for a | ||||
| fixed term (regardless of how the transaction is characterized), the | ||||
| Corresponding Source conveyed under this section must be accompanied | ||||
| by the Installation Information.  But this requirement does not apply | ||||
| if neither you nor any third party retains the ability to install | ||||
| modified object code on the User Product (for example, the work has | ||||
| been installed in ROM). | ||||
|  | ||||
|   The requirement to provide Installation Information does not include a | ||||
| requirement to continue to provide support service, warranty, or updates | ||||
| for a work that has been modified or installed by the recipient, or for | ||||
| the User Product in which it has been modified or installed.  Access to a | ||||
| network may be denied when the modification itself materially and | ||||
| adversely affects the operation of the network or violates the rules and | ||||
| protocols for communication across the network. | ||||
|  | ||||
|   Corresponding Source conveyed, and Installation Information provided, | ||||
| in accord with this section must be in a format that is publicly | ||||
| documented (and with an implementation available to the public in | ||||
| source code form), and must require no special password or key for | ||||
| unpacking, reading or copying. | ||||
|  | ||||
|   7. Additional Terms. | ||||
|  | ||||
|   "Additional permissions" are terms that supplement the terms of this | ||||
| License by making exceptions from one or more of its conditions. | ||||
| Additional permissions that are applicable to the entire Program shall | ||||
| be treated as though they were included in this License, to the extent | ||||
| that they are valid under applicable law.  If additional permissions | ||||
| apply only to part of the Program, that part may be used separately | ||||
| under those permissions, but the entire Program remains governed by | ||||
| this License without regard to the additional permissions. | ||||
|  | ||||
|   When you convey a copy of a covered work, you may at your option | ||||
| remove any additional permissions from that copy, or from any part of | ||||
| it.  (Additional permissions may be written to require their own | ||||
| removal in certain cases when you modify the work.)  You may place | ||||
| additional permissions on material, added by you to a covered work, | ||||
| for which you have or can give appropriate copyright permission. | ||||
|  | ||||
|   Notwithstanding any other provision of this License, for material you | ||||
| add to a covered work, you may (if authorized by the copyright holders of | ||||
| that material) supplement the terms of this License with terms: | ||||
|  | ||||
|     a) Disclaiming warranty or limiting liability differently from the | ||||
|     terms of sections 15 and 16 of this License; or | ||||
|  | ||||
|     b) Requiring preservation of specified reasonable legal notices or | ||||
|     author attributions in that material or in the Appropriate Legal | ||||
|     Notices displayed by works containing it; or | ||||
|  | ||||
|     c) Prohibiting misrepresentation of the origin of that material, or | ||||
|     requiring that modified versions of such material be marked in | ||||
|     reasonable ways as different from the original version; or | ||||
|  | ||||
|     d) Limiting the use for publicity purposes of names of licensors or | ||||
|     authors of the material; or | ||||
|  | ||||
|     e) Declining to grant rights under trademark law for use of some | ||||
|     trade names, trademarks, or service marks; or | ||||
|  | ||||
|     f) Requiring indemnification of licensors and authors of that | ||||
|     material by anyone who conveys the material (or modified versions of | ||||
|     it) with contractual assumptions of liability to the recipient, for | ||||
|     any liability that these contractual assumptions directly impose on | ||||
|     those licensors and authors. | ||||
|  | ||||
|   All other non-permissive additional terms are considered "further | ||||
| restrictions" within the meaning of section 10.  If the Program as you | ||||
| received it, or any part of it, contains a notice stating that it is | ||||
| governed by this License along with a term that is a further | ||||
| restriction, you may remove that term.  If a license document contains | ||||
| a further restriction but permits relicensing or conveying under this | ||||
| License, you may add to a covered work material governed by the terms | ||||
| of that license document, provided that the further restriction does | ||||
| not survive such relicensing or conveying. | ||||
|  | ||||
|   If you add terms to a covered work in accord with this section, you | ||||
| must place, in the relevant source files, a statement of the | ||||
| additional terms that apply to those files, or a notice indicating | ||||
| where to find the applicable terms. | ||||
|  | ||||
|   Additional terms, permissive or non-permissive, may be stated in the | ||||
| form of a separately written license, or stated as exceptions; | ||||
| the above requirements apply either way. | ||||
|  | ||||
|   8. Termination. | ||||
|  | ||||
|   You may not propagate or modify a covered work except as expressly | ||||
| provided under this License.  Any attempt otherwise to propagate or | ||||
| modify it is void, and will automatically terminate your rights under | ||||
| this License (including any patent licenses granted under the third | ||||
| paragraph of section 11). | ||||
|  | ||||
|   However, if you cease all violation of this License, then your | ||||
| license from a particular copyright holder is reinstated (a) | ||||
| provisionally, unless and until the copyright holder explicitly and | ||||
| finally terminates your license, and (b) permanently, if the copyright | ||||
| holder fails to notify you of the violation by some reasonable means | ||||
| prior to 60 days after the cessation. | ||||
|  | ||||
|   Moreover, your license from a particular copyright holder is | ||||
| reinstated permanently if the copyright holder notifies you of the | ||||
| violation by some reasonable means, this is the first time you have | ||||
| received notice of violation of this License (for any work) from that | ||||
| copyright holder, and you cure the violation prior to 30 days after | ||||
| your receipt of the notice. | ||||
|  | ||||
|   Termination of your rights under this section does not terminate the | ||||
| licenses of parties who have received copies or rights from you under | ||||
| this License.  If your rights have been terminated and not permanently | ||||
| reinstated, you do not qualify to receive new licenses for the same | ||||
| material under section 10. | ||||
|  | ||||
|   9. Acceptance Not Required for Having Copies. | ||||
|  | ||||
|   You are not required to accept this License in order to receive or | ||||
| run a copy of the Program.  Ancillary propagation of a covered work | ||||
| occurring solely as a consequence of using peer-to-peer transmission | ||||
| to receive a copy likewise does not require acceptance.  However, | ||||
| nothing other than this License grants you permission to propagate or | ||||
| modify any covered work.  These actions infringe copyright if you do | ||||
| not accept this License.  Therefore, by modifying or propagating a | ||||
| covered work, you indicate your acceptance of this License to do so. | ||||
|  | ||||
|   10. Automatic Licensing of Downstream Recipients. | ||||
|  | ||||
|   Each time you convey a covered work, the recipient automatically | ||||
| receives a license from the original licensors, to run, modify and | ||||
| propagate that work, subject to this License.  You are not responsible | ||||
| for enforcing compliance by third parties with this License. | ||||
|  | ||||
|   An "entity transaction" is a transaction transferring control of an | ||||
| organization, or substantially all assets of one, or subdividing an | ||||
| organization, or merging organizations.  If propagation of a covered | ||||
| work results from an entity transaction, each party to that | ||||
| transaction who receives a copy of the work also receives whatever | ||||
| licenses to the work the party's predecessor in interest had or could | ||||
| give under the previous paragraph, plus a right to possession of the | ||||
| Corresponding Source of the work from the predecessor in interest, if | ||||
| the predecessor has it or can get it with reasonable efforts. | ||||
|  | ||||
|   You may not impose any further restrictions on the exercise of the | ||||
| rights granted or affirmed under this License.  For example, you may | ||||
| not impose a license fee, royalty, or other charge for exercise of | ||||
| rights granted under this License, and you may not initiate litigation | ||||
| (including a cross-claim or counterclaim in a lawsuit) alleging that | ||||
| any patent claim is infringed by making, using, selling, offering for | ||||
| sale, or importing the Program or any portion of it. | ||||
|  | ||||
|   11. Patents. | ||||
|  | ||||
|   A "contributor" is a copyright holder who authorizes use under this | ||||
| License of the Program or a work on which the Program is based.  The | ||||
| work thus licensed is called the contributor's "contributor version". | ||||
|  | ||||
|   A contributor's "essential patent claims" are all patent claims | ||||
| owned or controlled by the contributor, whether already acquired or | ||||
| hereafter acquired, that would be infringed by some manner, permitted | ||||
| by this License, of making, using, or selling its contributor version, | ||||
| but do not include claims that would be infringed only as a | ||||
| consequence of further modification of the contributor version.  For | ||||
| purposes of this definition, "control" includes the right to grant | ||||
| patent sublicenses in a manner consistent with the requirements of | ||||
| this License. | ||||
|  | ||||
|   Each contributor grants you a non-exclusive, worldwide, royalty-free | ||||
| patent license under the contributor's essential patent claims, to | ||||
| make, use, sell, offer for sale, import and otherwise run, modify and | ||||
| propagate the contents of its contributor version. | ||||
|  | ||||
|   In the following three paragraphs, a "patent license" is any express | ||||
| agreement or commitment, however denominated, not to enforce a patent | ||||
| (such as an express permission to practice a patent or covenant not to | ||||
| sue for patent infringement).  To "grant" such a patent license to a | ||||
| party means to make such an agreement or commitment not to enforce a | ||||
| patent against the party. | ||||
|  | ||||
|   If you convey a covered work, knowingly relying on a patent license, | ||||
| and the Corresponding Source of the work is not available for anyone | ||||
| to copy, free of charge and under the terms of this License, through a | ||||
| publicly available network server or other readily accessible means, | ||||
| then you must either (1) cause the Corresponding Source to be so | ||||
| available, or (2) arrange to deprive yourself of the benefit of the | ||||
| patent license for this particular work, or (3) arrange, in a manner | ||||
| consistent with the requirements of this License, to extend the patent | ||||
| license to downstream recipients.  "Knowingly relying" means you have | ||||
| actual knowledge that, but for the patent license, your conveying the | ||||
| covered work in a country, or your recipient's use of the covered work | ||||
| in a country, would infringe one or more identifiable patents in that | ||||
| country that you have reason to believe are valid. | ||||
|  | ||||
|   If, pursuant to or in connection with a single transaction or | ||||
| arrangement, you convey, or propagate by procuring conveyance of, a | ||||
| covered work, and grant a patent license to some of the parties | ||||
| receiving the covered work authorizing them to use, propagate, modify | ||||
| or convey a specific copy of the covered work, then the patent license | ||||
| you grant is automatically extended to all recipients of the covered | ||||
| work and works based on it. | ||||
|  | ||||
|   A patent license is "discriminatory" if it does not include within | ||||
| the scope of its coverage, prohibits the exercise of, or is | ||||
| conditioned on the non-exercise of one or more of the rights that are | ||||
| specifically granted under this License.  You may not convey a covered | ||||
| work if you are a party to an arrangement with a third party that is | ||||
| in the business of distributing software, under which you make payment | ||||
| to the third party based on the extent of your activity of conveying | ||||
| the work, and under which the third party grants, to any of the | ||||
| parties who would receive the covered work from you, a discriminatory | ||||
| patent license (a) in connection with copies of the covered work | ||||
| conveyed by you (or copies made from those copies), or (b) primarily | ||||
| for and in connection with specific products or compilations that | ||||
| contain the covered work, unless you entered into that arrangement, | ||||
| or that patent license was granted, prior to 28 March 2007. | ||||
|  | ||||
|   Nothing in this License shall be construed as excluding or limiting | ||||
| any implied license or other defenses to infringement that may | ||||
| otherwise be available to you under applicable patent law. | ||||
|  | ||||
|   12. No Surrender of Others' Freedom. | ||||
|  | ||||
|   If conditions are imposed on you (whether by court order, agreement or | ||||
| otherwise) that contradict the conditions of this License, they do not | ||||
| excuse you from the conditions of this License.  If you cannot convey a | ||||
| covered work so as to satisfy simultaneously your obligations under this | ||||
| License and any other pertinent obligations, then as a consequence you may | ||||
| not convey it at all.  For example, if you agree to terms that obligate you | ||||
| to collect a royalty for further conveying from those to whom you convey | ||||
| the Program, the only way you could satisfy both those terms and this | ||||
| License would be to refrain entirely from conveying the Program. | ||||
|  | ||||
|   13. Use with the GNU Affero General Public License. | ||||
|  | ||||
|   Notwithstanding any other provision of this License, you have | ||||
| permission to link or combine any covered work with a work licensed | ||||
| under version 3 of the GNU Affero General Public License into a single | ||||
| combined work, and to convey the resulting work.  The terms of this | ||||
| License will continue to apply to the part which is the covered work, | ||||
| but the special requirements of the GNU Affero General Public License, | ||||
| section 13, concerning interaction through a network will apply to the | ||||
| combination as such. | ||||
|  | ||||
|   14. Revised Versions of this License. | ||||
|  | ||||
|   The Free Software Foundation may publish revised and/or new versions of | ||||
| the GNU General Public License from time to time.  Such new versions will | ||||
| be similar in spirit to the present version, but may differ in detail to | ||||
| address new problems or concerns. | ||||
|  | ||||
|   Each version is given a distinguishing version number.  If the | ||||
| Program specifies that a certain numbered version of the GNU General | ||||
| Public License "or any later version" applies to it, you have the | ||||
| option of following the terms and conditions either of that numbered | ||||
| version or of any later version published by the Free Software | ||||
| Foundation.  If the Program does not specify a version number of the | ||||
| GNU General Public License, you may choose any version ever published | ||||
| by the Free Software Foundation. | ||||
|  | ||||
|   If the Program specifies that a proxy can decide which future | ||||
| versions of the GNU General Public License can be used, that proxy's | ||||
| public statement of acceptance of a version permanently authorizes you | ||||
| to choose that version for the Program. | ||||
|  | ||||
|   Later license versions may give you additional or different | ||||
| permissions.  However, no additional obligations are imposed on any | ||||
| author or copyright holder as a result of your choosing to follow a | ||||
| later version. | ||||
|  | ||||
|   15. Disclaimer of Warranty. | ||||
|  | ||||
|   THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY | ||||
| APPLICABLE LAW.  EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT | ||||
| HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY | ||||
| OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, | ||||
| THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | ||||
| PURPOSE.  THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM | ||||
| IS WITH YOU.  SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF | ||||
| ALL NECESSARY SERVICING, REPAIR OR CORRECTION. | ||||
|  | ||||
|   16. Limitation of Liability. | ||||
|  | ||||
|   IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING | ||||
| WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS | ||||
| THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY | ||||
| GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE | ||||
| USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF | ||||
| DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD | ||||
| PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), | ||||
| EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF | ||||
| SUCH DAMAGES. | ||||
|  | ||||
|   17. Interpretation of Sections 15 and 16. | ||||
|  | ||||
|   If the disclaimer of warranty and limitation of liability provided | ||||
| above cannot be given local legal effect according to their terms, | ||||
| reviewing courts shall apply local law that most closely approximates | ||||
| an absolute waiver of all civil liability in connection with the | ||||
| Program, unless a warranty or assumption of liability accompanies a | ||||
| copy of the Program in return for a fee. | ||||
|  | ||||
|                      END OF TERMS AND CONDITIONS | ||||
|  | ||||
|             How to Apply These Terms to Your New Programs | ||||
|  | ||||
|   If you develop a new program, and you want it to be of the greatest | ||||
| possible use to the public, the best way to achieve this is to make it | ||||
| free software which everyone can redistribute and change under these terms. | ||||
|  | ||||
|   To do so, attach the following notices to the program.  It is safest | ||||
| to attach them to the start of each source file to most effectively | ||||
| state the exclusion of warranty; and each file should have at least | ||||
| the "copyright" line and a pointer to where the full notice is found. | ||||
|  | ||||
|     <one line to give the program's name and a brief idea of what it does.> | ||||
|     Copyright (C) 2020  Animath | ||||
|  | ||||
|     This program is free software: you can redistribute it and/or modify | ||||
|     it under the terms of the GNU General Public License as published by | ||||
|     the Free Software Foundation, either version 3 of the License, or | ||||
|     (at your option) any later version. | ||||
|  | ||||
|     This program is distributed in the hope that it will be useful, | ||||
|     but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|     GNU General Public License for more details. | ||||
|  | ||||
|     You should have received a copy of the GNU General Public License | ||||
|     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  | ||||
| Also add information on how to contact you by electronic and paper mail. | ||||
|  | ||||
|   If the program does terminal interaction, make it output a short | ||||
| notice like this when it starts in an interactive mode: | ||||
|  | ||||
|     <program>  Copyright (C) 2020  Animath | ||||
|     This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. | ||||
|     This is free software, and you are welcome to redistribute it | ||||
|     under certain conditions; type `show c' for details. | ||||
|  | ||||
| The hypothetical commands `show w' and `show c' should show the appropriate | ||||
| parts of the General Public License.  Of course, your program's commands | ||||
| might be different; for a GUI interface, you would use an "about box". | ||||
|  | ||||
|   You should also get your employer (if you work as a programmer) or school, | ||||
| if any, to sign a "copyright disclaimer" for the program, if necessary. | ||||
| For more information on this, and how to apply and follow the GNU GPL, see | ||||
| <https://www.gnu.org/licenses/>. | ||||
|  | ||||
|   The GNU General Public License does not permit incorporating your program | ||||
| into proprietary programs.  If your program is a subroutine library, you | ||||
| may consider it more useful to permit linking proprietary applications with | ||||
| the library.  If this is what you want to do, use the GNU Lesser General | ||||
| Public License instead of this License.  But first, please read | ||||
| <https://www.gnu.org/licenses/why-not-lgpl.html>. | ||||
							
								
								
									
										70
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | ||||
| # Plateforme des Correspondances des Jeunes Mathématicien·nes | ||||
|  | ||||
| [](https://gitlab.com/animath/si/plateforme-corres2math/-/commits/master) | ||||
| [](https://gitlab.com/animath/si/plateforme-corres2math/-/commits/master) | ||||
|  | ||||
| La plateforme des Correspondances des Jeunes Mathématicien·nes est née pour la seconde édition en 2019 de l'action. | ||||
| D'abord codée en PHP, elle a subi une refonte totale en Python, à l'aide du framework Web [Django](https://www.djangoproject.com/). | ||||
|  | ||||
| Cette plateforme permet aux participants et encadrants de s'inscrire et de déposer leurs autorisations nécessaires. | ||||
| Ils pourront ensuite déposer leurs solutions et notes de synthèse pour le premier tour en temps voulu. La plateforme  | ||||
| offre également un accès pour les organisateurs et les jurys leur permettant de communiquer avec les équipes et de | ||||
| récupérer les documents nécessaires. | ||||
|  | ||||
| Un wiki plus détaillé arrivera ultérieurement. L'interface organisateur et jury est vouée à être plus poussée. | ||||
|  | ||||
| L'instance de production est disponible à l'adresse [inscription.correspondances-maths.fr](https://inscription.correspondances-maths.fr). | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| Le plus simple pour installer la plateforme est d'utiliser l'image Docker incluse, qui fait tourner un serveur Nginx | ||||
| exposé sur le port 80 avec le serveur Django. Ci-dessous une configuration Docker-Compose, à adapter selon vos besoins : | ||||
|  | ||||
| ```yaml | ||||
|   plateforme-corres2math: | ||||
|     build: https://gitlab.com/animath/si/plateforme-corres2math.git | ||||
|     links: | ||||
|       - postgres | ||||
|     ports: | ||||
|       - "80:80" | ||||
|     env_file: | ||||
|       - ./inscription-corres2math.env | ||||
|     volumes: | ||||
|     # - ./inscription-corres2math:/code | ||||
|       - ./inscription-corres2math/media:/code/media | ||||
| ``` | ||||
|  | ||||
| Le volume `/code` n'est à ajouter uniquement en développement, et jamais en production. | ||||
|  | ||||
| Il faut remplir les variables d'environnement suivantes : | ||||
|  | ||||
| ```env | ||||
| CORRES2MATH_STAGE=              # dev ou prod | ||||
| DJANGO_DB_TYPE=                 # MySQL, PostgreSQL ou SQLite (par défaut) | ||||
| DJANGO_DB_HOST=                 # Hôte de la base de données | ||||
| DJANGO_DB_NAME=                 # Nom de la base de données | ||||
| DJANGO_DB_USER=                 # Utilisateur de la base de données | ||||
| DJANGO_DB_PASSWORD=             # Mot de passe pour accéder à la base de données | ||||
| SMTP_HOST=                      # Hôte SMTP pour l'envoi de mails | ||||
| SMTP_PORT=465                   # Port du serveur SMTP | ||||
| SMTP_HOST_USER=                 # Utilisateur du compte SMTP | ||||
| SMTP_HOST_PASSWORD=             # Mot de passe du compte SMTP | ||||
| FROM_EMAIL=contact@correspondances-maths.fr     # Nom de l'expéditeur des mails | ||||
| SERVER_EMAIL=contact@correspondances-maths.fr   # Adresse e-mail expéditrice | ||||
| SYMPA_URL=lists.example.com     # Serveur Sympa à utiliser | ||||
| SYMPA_EMAIL=                    # Adresse e-mail du compte administrateur de Sympa | ||||
| SYMPA_PASSWORD=                 # Mot de passe du compte administrateur de Sympa | ||||
| SYNAPSE_PASSWORD=               # Mot de passe du robot Matrix  | ||||
| ``` | ||||
|  | ||||
| Si le type de base de données sélectionné est SQLite, la variable `DJANGO_DB_HOST` sera utilisée en guise de chemin vers | ||||
| le fichier de base de données (par défaut, `db.sqlite3`). | ||||
|  | ||||
| En développement, il est recommandé d'utiliser SQLite pour des raisons de simplicité. Les paramètres de mail ne seront | ||||
| pas utilisés, et les mails qui doivent être envoyés seront envoyés dans la console. Les intégrations mail et Matrix | ||||
| seront également désactivées. | ||||
|  | ||||
| En production, il est recommandé de ne pas utiliser SQLite pour des raisons de performances. | ||||
|  | ||||
| La dernière différence entre le développment et la production est qu'en développement, chaque modification d'un fichier | ||||
| est détectée et le serveur se relance automatiquement dès lors. | ||||
							
								
								
									
										4
									
								
								apps/api/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								apps/api/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| default_app_config = 'api.apps.APIConfig' | ||||
							
								
								
									
										13
									
								
								apps/api/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								apps/api/apps.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.apps import AppConfig | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
|  | ||||
| class APIConfig(AppConfig): | ||||
|     """ | ||||
|     Manage the inscription through a JSON API. | ||||
|     """ | ||||
|     name = 'api' | ||||
|     verbose_name = _('API') | ||||
							
								
								
									
										19
									
								
								apps/api/serializers.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								apps/api/serializers.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.contrib.auth.models import User | ||||
| from rest_framework import serializers | ||||
|  | ||||
|  | ||||
| class UserSerializer(serializers.ModelSerializer): | ||||
|     """ | ||||
|     Serialize a User object into JSON. | ||||
|     """ | ||||
|     class Meta: | ||||
|         model = User | ||||
|         exclude = ( | ||||
|             'username', | ||||
|             'password', | ||||
|             'groups', | ||||
|             'user_permissions', | ||||
|         ) | ||||
							
								
								
									
										27
									
								
								apps/api/tests.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								apps/api/tests.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from unittest.case import skipIf | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.contrib.auth.models import User | ||||
| from django.test import TestCase | ||||
|  | ||||
|  | ||||
| class TestAPIPages(TestCase): | ||||
|     def setUp(self): | ||||
|         self.user = User.objects.create_superuser( | ||||
|             username="admin", | ||||
|             password="apitest", | ||||
|             email="", | ||||
|         ) | ||||
|         self.client.force_login(self.user) | ||||
|  | ||||
|     def test_user_page(self): | ||||
|         response = self.client.get("/api/user/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|     @skipIf("logs" not in settings.INSTALLED_APPS, reason="logs app is not used") | ||||
|     def test_logs_page(self): | ||||
|         response = self.client.get("/api/logs/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
							
								
								
									
										26
									
								
								apps/api/urls.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								apps/api/urls.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.conf.urls import include, url | ||||
| from rest_framework import routers | ||||
|  | ||||
| from .viewsets import UserViewSet | ||||
|  | ||||
| # Routers provide an easy way of automatically determining the URL conf. | ||||
| # Register each app API router and user viewset | ||||
| router = routers.DefaultRouter() | ||||
| router.register('user', UserViewSet) | ||||
|  | ||||
| if "logs" in settings.INSTALLED_APPS: | ||||
|     from logs.api.urls import register_logs_urls | ||||
|     register_logs_urls(router, "logs") | ||||
|  | ||||
| app_name = 'api' | ||||
|  | ||||
| # Wire up our API using automatic URL routing. | ||||
| # Additionally, we include login URLs for the browsable API. | ||||
| urlpatterns = [ | ||||
|     url('^', include(router.urls)), | ||||
|     url('^api-auth/', include('rest_framework.urls', namespace='rest_framework')), | ||||
| ] | ||||
							
								
								
									
										20
									
								
								apps/api/viewsets.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								apps/api/viewsets.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.contrib.auth.models import User | ||||
| from django_filters.rest_framework import DjangoFilterBackend | ||||
| from rest_framework.filters import SearchFilter | ||||
| from rest_framework.viewsets import ModelViewSet | ||||
|  | ||||
| from .serializers import UserSerializer | ||||
|  | ||||
|  | ||||
| class UserViewSet(ModelViewSet): | ||||
|     """ | ||||
|     Display list of users. | ||||
|     """ | ||||
|     queryset = User.objects.order_by("id").all() | ||||
|     serializer_class = UserSerializer | ||||
|     filter_backends = [DjangoFilterBackend, SearchFilter] | ||||
|     filterset_fields = ['id', 'first_name', 'last_name', 'email', 'is_superuser', 'is_staff', 'is_active', ] | ||||
|     search_fields = ['$first_name', '$last_name', ] | ||||
							
								
								
									
										4
									
								
								apps/eastereggs/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								apps/eastereggs/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| default_app_config = 'eastereggs.apps.EastereggsConfig' | ||||
							
								
								
									
										8
									
								
								apps/eastereggs/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								apps/eastereggs/apps.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.apps import AppConfig | ||||
|  | ||||
|  | ||||
| class EastereggsConfig(AppConfig): | ||||
|     name = 'eastereggs' | ||||
							
								
								
									
										2
									
								
								apps/eastereggs/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								apps/eastereggs/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
							
								
								
									
										19
									
								
								apps/eastereggs/templates/eastereggs/xp.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								apps/eastereggs/templates/eastereggs/xp.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| {% extends "index.html" %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="index-content"></div> | ||||
| {% include "eastereggs/xp_modal.html" %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block extrajavascript %} | ||||
|     <script> | ||||
|         $(document).ready(function() { | ||||
|             $("#index-content").load("{% url "index" %} #content"); | ||||
|             function displayModal() { | ||||
|                 $("#xpModal").modal('toggle'); | ||||
|                 setTimeout(displayModal, 400); | ||||
|             } | ||||
|             displayModal(); | ||||
|         }); | ||||
|     </script> | ||||
| {% endblock %} | ||||
							
								
								
									
										20
									
								
								apps/eastereggs/templates/eastereggs/xp_modal.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								apps/eastereggs/templates/eastereggs/xp_modal.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| <div id="xpModal" class="modal fade" tabindex="-1" role="dialog"> | ||||
|     <div class="modal-dialog" role="document"> | ||||
|         <div class="modal-content"> | ||||
|             <div class="modal-header"> | ||||
|                 <h5 class="modal-title">{% trans "Error" %}</h5> | ||||
|                 <button type="button" class="close" data-dismiss="modal" aria-label="Close"> | ||||
|                     <span aria-hidden="true">×</span> | ||||
|                 </button> | ||||
|             </div> | ||||
|             <div class="modal-body"> | ||||
|                 {% trans "This task failed successfully." %} | ||||
|             </div> | ||||
|             <div class="modal-footer"> | ||||
|                 <button type="button" class="btn btn-secondary" data-dismiss="modal">{% trans "Close" %}</button> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
| </div> | ||||
							
								
								
									
										11
									
								
								apps/eastereggs/urls.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								apps/eastereggs/urls.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.urls import path | ||||
| from django.views.generic import TemplateView | ||||
|  | ||||
| app_name = "eastereggs" | ||||
|  | ||||
| urlpatterns = [ | ||||
|     path("xp/", TemplateView.as_view(template_name="eastereggs/xp.html")), | ||||
| ] | ||||
							
								
								
									
										4
									
								
								apps/logs/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								apps/logs/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| default_app_config = 'logs.apps.LogsConfig' | ||||
							
								
								
									
										2
									
								
								apps/logs/api/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								apps/logs/api/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
							
								
								
									
										19
									
								
								apps/logs/api/serializers.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								apps/logs/api/serializers.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from rest_framework import serializers | ||||
|  | ||||
| from ..models import Changelog | ||||
|  | ||||
|  | ||||
| class ChangelogSerializer(serializers.ModelSerializer): | ||||
|     """ | ||||
|     REST API Serializer for Changelog types. | ||||
|     The djangorestframework plugin will analyse the model `Changelog` and parse all fields in the API. | ||||
|     """ | ||||
|  | ||||
|     class Meta: | ||||
|         model = Changelog | ||||
|         fields = '__all__' | ||||
|         # noinspection PyProtectedMember | ||||
|         read_only_fields = [f.name for f in model._meta.get_fields()]  # Changelogs are read-only protected | ||||
							
								
								
									
										11
									
								
								apps/logs/api/urls.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								apps/logs/api/urls.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from .views import ChangelogViewSet | ||||
|  | ||||
|  | ||||
| def register_logs_urls(router, path): | ||||
|     """ | ||||
|     Configure router for Activity REST API. | ||||
|     """ | ||||
|     router.register(path, ChangelogViewSet) | ||||
							
								
								
									
										28
									
								
								apps/logs/api/views.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								apps/logs/api/views.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django_filters.rest_framework import DjangoFilterBackend | ||||
| from rest_framework.filters import OrderingFilter | ||||
| from rest_framework.viewsets import ModelViewSet | ||||
|  | ||||
| from .serializers import ChangelogSerializer | ||||
| from ..models import Changelog | ||||
|  | ||||
|  | ||||
| class ChangelogViewSet(ModelViewSet): | ||||
|     """ | ||||
|     REST API View set. | ||||
|     The djangorestframework plugin will get all `Changelog` objects, serialize it to JSON with the given serializer, | ||||
|     then render it on /api/logs/ | ||||
|     """ | ||||
|  | ||||
|     def check_permissions(self, request): | ||||
|         # Only superusers can get access to logs | ||||
|         return self.request.user and self.request.user.is_superuser | ||||
|  | ||||
|     queryset = Changelog.objects.all() | ||||
|     serializer_class = ChangelogSerializer | ||||
|     filter_backends = [DjangoFilterBackend, OrderingFilter] | ||||
|     filterset_fields = ['model', 'action', "instance_pk", 'user', 'ip', ] | ||||
|     ordering_fields = ['timestamp', 'id', ] | ||||
|     ordering = ['-id', ] | ||||
							
								
								
									
										18
									
								
								apps/logs/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								apps/logs/apps.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.apps import AppConfig | ||||
| from django.db.models.signals import post_delete, post_save, pre_save | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
|  | ||||
| class LogsConfig(AppConfig): | ||||
|     name = 'logs' | ||||
|     verbose_name = _('Logs') | ||||
|  | ||||
|     def ready(self): | ||||
|         # noinspection PyUnresolvedReferences | ||||
|         from . import signals | ||||
|         pre_save.connect(signals.pre_save_object) | ||||
|         post_save.connect(signals.save_object) | ||||
|         post_delete.connect(signals.delete_object) | ||||
							
								
								
									
										37
									
								
								apps/logs/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								apps/logs/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| # Generated by Django 3.1.3 on 2020-11-04 12:05 | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.db import migrations, models | ||||
| import django.db.models.deletion | ||||
| import django.utils.timezone | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     initial = True | ||||
|  | ||||
|     dependencies = [ | ||||
|         migrations.swappable_dependency(settings.AUTH_USER_MODEL), | ||||
|         ('contenttypes', '0002_remove_content_type_name'), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.CreateModel( | ||||
|             name='Changelog', | ||||
|             fields=[ | ||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||
|                 ('ip', models.GenericIPAddressField(blank=True, null=True, verbose_name='IP Address')), | ||||
|                 ('instance_pk', models.CharField(max_length=255, verbose_name='identifier')), | ||||
|                 ('previous', models.TextField(blank=True, default='', verbose_name='previous data')), | ||||
|                 ('data', models.TextField(blank=True, default='', verbose_name='new data')), | ||||
|                 ('action', models.CharField(choices=[('create', 'create'), ('edit', 'edit'), ('delete', 'delete')], default='edit', max_length=16, verbose_name='action')), | ||||
|                 ('timestamp', models.DateTimeField(default=django.utils.timezone.now, verbose_name='timestamp')), | ||||
|                 ('model', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, to='contenttypes.contenttype', verbose_name='model')), | ||||
|                 ('user', models.ForeignKey(null=True, on_delete=django.db.models.deletion.PROTECT, to=settings.AUTH_USER_MODEL, verbose_name='user')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'changelog', | ||||
|                 'verbose_name_plural': 'changelogs', | ||||
|             }, | ||||
|         ), | ||||
|     ] | ||||
							
								
								
									
										2
									
								
								apps/logs/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								apps/logs/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| # Copyright (C) 2020 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
							
								
								
									
										88
									
								
								apps/logs/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								apps/logs/models.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | ||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.contrib.contenttypes.models import ContentType | ||||
| from django.core.exceptions import ValidationError | ||||
| from django.db import models | ||||
| from django.utils import timezone | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
|  | ||||
| class Changelog(models.Model): | ||||
|     """ | ||||
|     Store each modification in the database (except sessions and logging), | ||||
|     including creating, editing and deleting models. | ||||
|     """ | ||||
|  | ||||
|     user = models.ForeignKey( | ||||
|         settings.AUTH_USER_MODEL, | ||||
|         on_delete=models.PROTECT, | ||||
|         null=True, | ||||
|         verbose_name=_('user'), | ||||
|     ) | ||||
|  | ||||
|     ip = models.GenericIPAddressField( | ||||
|         null=True, | ||||
|         blank=True, | ||||
|         verbose_name=_("IP Address") | ||||
|     ) | ||||
|  | ||||
|     model = models.ForeignKey( | ||||
|         ContentType, | ||||
|         on_delete=models.PROTECT, | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         verbose_name=_('model'), | ||||
|     ) | ||||
|  | ||||
|     instance_pk = models.CharField( | ||||
|         max_length=255, | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         verbose_name=_('identifier'), | ||||
|     ) | ||||
|  | ||||
|     previous = models.TextField( | ||||
|         blank=True, | ||||
|         default="", | ||||
|         verbose_name=_('previous data'), | ||||
|     ) | ||||
|  | ||||
|     data = models.TextField( | ||||
|         blank=True, | ||||
|         default="", | ||||
|         verbose_name=_('new data'), | ||||
|     ) | ||||
|  | ||||
|     action = models.CharField(  # create, edit or delete | ||||
|         max_length=16, | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         choices=[ | ||||
|             ('create', _('create')), | ||||
|             ('edit', _('edit')), | ||||
|             ('delete', _('delete')), | ||||
|         ], | ||||
|         default='edit', | ||||
|         verbose_name=_('action'), | ||||
|     ) | ||||
|  | ||||
|     timestamp = models.DateTimeField( | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         default=timezone.now, | ||||
|         name='timestamp', | ||||
|         verbose_name=_('timestamp'), | ||||
|     ) | ||||
|  | ||||
|     def delete(self, using=None, keep_parents=False): | ||||
|         raise ValidationError(_("Logs cannot be destroyed.")) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("changelog") | ||||
|         verbose_name_plural = _("changelogs") | ||||
|  | ||||
|     def __str__(self): | ||||
|         return _("Changelog of type \"{action}\" for model {model} at {timestamp}").format( | ||||
|             action=self.get_action_display(), model=str(self.model), timestamp=str(self.timestamp)) | ||||
							
								
								
									
										132
									
								
								apps/logs/signals.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								apps/logs/signals.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,132 @@ | ||||
| import getpass | ||||
|  | ||||
| from corres2math.middlewares import get_current_authenticated_user, get_current_ip | ||||
| from django.contrib.auth.models import User | ||||
| from django.contrib.contenttypes.models import ContentType | ||||
| from rest_framework.renderers import JSONRenderer | ||||
| from rest_framework.serializers import ModelSerializer | ||||
|  | ||||
| from .models import Changelog | ||||
|  | ||||
|  | ||||
| # Ces modèles ne nécessitent pas de logs | ||||
| EXCLUDED = [ | ||||
|     'admin.logentry', | ||||
|     'authtoken.token', | ||||
|     'contenttypes.contenttype', | ||||
|     'logs.changelog',  # Never remove this line | ||||
|     'mailer.dontsendentry', | ||||
|     'mailer.message', | ||||
|     'mailer.messagelog', | ||||
|     'migrations.migration', | ||||
|     'sessions.session', | ||||
| ] | ||||
|  | ||||
|  | ||||
| def pre_save_object(sender, instance, **kwargs): | ||||
|     """ | ||||
|     Before a model get saved, we get the previous instance that is currently in the database | ||||
|     """ | ||||
|     qs = sender.objects.filter(pk=instance.pk).all() | ||||
|     if qs.exists(): | ||||
|         instance._previous = qs.get() | ||||
|     else: | ||||
|         instance._previous = None | ||||
|  | ||||
|  | ||||
| def save_object(sender, instance, **kwargs): | ||||
|     """ | ||||
|     Each time a model is saved, an entry in the table `Changelog` is added in the database | ||||
|     in order to store each modification made | ||||
|     """ | ||||
|     # noinspection PyProtectedMember | ||||
|     if instance._meta.label_lower in EXCLUDED or hasattr(instance, "_no_signal"): | ||||
|         return | ||||
|  | ||||
|     # noinspection PyProtectedMember | ||||
|     previous = instance._previous | ||||
|  | ||||
|     # Si un utilisateur est connecté, on récupère l'utilisateur courant ainsi que son adresse IP | ||||
|     user, ip = get_current_authenticated_user(), get_current_ip() | ||||
|  | ||||
|     if user is None: | ||||
|         # Si la modification n'a pas été faite via le client Web, on suppose que c'est du à `manage.py` | ||||
|         # On récupère alors l'utilisateur·trice connecté·e à la VM, et on récupère la note associée | ||||
|         ip = "127.0.0.1" | ||||
|         username = getpass.getuser() | ||||
|         user = User.objects.get(username=username) if User.objects.filter(username=username).exists() else None | ||||
|  | ||||
|     # On n'enregistre pas les connexions | ||||
|     # noinspection PyProtectedMember | ||||
|     if user is not None and instance._meta.label_lower == "auth.user" and previous \ | ||||
|             and instance.last_login != previous.last_login: | ||||
|         return | ||||
|  | ||||
|     changed_fields = '__all__' | ||||
|     if previous: | ||||
|         # On ne garde que les champs modifiés | ||||
|         changed_fields = [] | ||||
|         for field in instance._meta.fields: | ||||
|             if field.name.endswith("_ptr"): | ||||
|                 # A field ending with _ptr is a OneToOneRel with a subclass, e.g. NoteClub.note_ptr -> Note | ||||
|                 continue | ||||
|             if getattr(instance, field.name) != getattr(previous, field.name): | ||||
|                 changed_fields.append(field.name) | ||||
|  | ||||
|     if len(changed_fields) == 0: | ||||
|         # Pas de log s'il n'y a pas de modification | ||||
|         return | ||||
|  | ||||
|     # On crée notre propre sérialiseur JSON pour pouvoir sauvegarder les modèles avec uniquement les champs modifiés | ||||
|     class CustomSerializer(ModelSerializer): | ||||
|         class Meta: | ||||
|             model = instance.__class__ | ||||
|             fields = changed_fields | ||||
|  | ||||
|     previous_json = JSONRenderer().render(CustomSerializer(previous).data).decode("UTF-8") if previous else "" | ||||
|     instance_json = JSONRenderer().render(CustomSerializer(instance).data).decode("UTF-8") | ||||
|  | ||||
|     Changelog.objects.create(user=user, | ||||
|                              ip=ip, | ||||
|                              model=ContentType.objects.get_for_model(instance), | ||||
|                              instance_pk=instance.pk, | ||||
|                              previous=previous_json, | ||||
|                              data=instance_json, | ||||
|                              action=("edit" if previous else "create") | ||||
|                              ).save() | ||||
|  | ||||
|  | ||||
| def delete_object(sender, instance, **kwargs): | ||||
|     """ | ||||
|     Each time a model is deleted, an entry in the table `Changelog` is added in the database | ||||
|     """ | ||||
|     # noinspection PyProtectedMember | ||||
|     if instance._meta.label_lower in EXCLUDED or hasattr(instance, "_no_signal"): | ||||
|         return | ||||
|  | ||||
|     # Si un utilisateur est connecté, on récupère l'utilisateur courant ainsi que son adresse IP | ||||
|     user, ip = get_current_authenticated_user(), get_current_ip() | ||||
|  | ||||
|     if user is None: | ||||
|         # Si la modification n'a pas été faite via le client Web, on suppose que c'est du à `manage.py` | ||||
|         # On récupère alors l'utilisateur·trice connecté·e à la VM, et on récupère la note associée | ||||
|         ip = "127.0.0.1" | ||||
|         username = getpass.getuser() | ||||
|         user = User.objects.get(username=username) if User.objects.filter(username=username).exists() else None | ||||
|  | ||||
|     # On crée notre propre sérialiseur JSON pour pouvoir sauvegarder les modèles | ||||
|     class CustomSerializer(ModelSerializer): | ||||
|         class Meta: | ||||
|             model = instance.__class__ | ||||
|             fields = '__all__' | ||||
|  | ||||
|     instance_json = JSONRenderer().render(CustomSerializer(instance).data).decode("UTF-8") | ||||
|  | ||||
|     Changelog.objects.create(user=user, | ||||
|                              ip=ip, | ||||
|                              model=ContentType.objects.get_for_model(instance), | ||||
|                              instance_pk=instance.pk, | ||||
|                              previous=instance_json, | ||||
|                              data="", | ||||
|                              action="delete" | ||||
|                              ) | ||||
							
								
								
									
										21
									
								
								apps/logs/tests.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								apps/logs/tests.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| from django.contrib.auth.models import User | ||||
| from django.contrib.contenttypes.models import ContentType | ||||
| from django.core.exceptions import ValidationError | ||||
| from django.test import TestCase | ||||
|  | ||||
| from .models import Changelog | ||||
|  | ||||
|  | ||||
| class TestChangelog(TestCase): | ||||
|     def test_logs(self): | ||||
|         user = User.objects.create(email="admin@example.com") | ||||
|         self.assertTrue(Changelog.objects.filter(action="create", instance_pk=user.pk, | ||||
|                                                  model=ContentType.objects.get_for_model(User)).exists()) | ||||
|         old_user_pk = user.pk | ||||
|         user.delete() | ||||
|         self.assertTrue(Changelog.objects.filter(action="delete", instance_pk=old_user_pk, | ||||
|                                                  model=ContentType.objects.get_for_model(User)).exists()) | ||||
|  | ||||
|         changelog = Changelog.objects.first() | ||||
|         self.assertRaises(ValidationError, changelog.delete) | ||||
|         str(Changelog.objects.all()) | ||||
							
								
								
									
										4
									
								
								apps/participation/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								apps/participation/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| default_app_config = 'participation.apps.ParticipationConfig' | ||||
							
								
								
									
										49
									
								
								apps/participation/admin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								apps/participation/admin.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.contrib import admin | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
| from .models import Participation, Phase, Question, Team, Video | ||||
|  | ||||
|  | ||||
| @admin.register(Team) | ||||
| class TeamAdmin(admin.ModelAdmin): | ||||
|     list_display = ('name', 'trigram', 'problem', 'valid',) | ||||
|     search_fields = ('name', 'trigram',) | ||||
|     list_filter = ('participation__problem', 'participation__valid',) | ||||
|  | ||||
|     def problem(self, team): | ||||
|         return team.participation.get_problem_display() | ||||
|  | ||||
|     problem.short_description = _('problem number') | ||||
|  | ||||
|     def valid(self, team): | ||||
|         return team.participation.valid | ||||
|  | ||||
|     valid.short_description = _('valid') | ||||
|  | ||||
|  | ||||
| @admin.register(Participation) | ||||
| class ParticipationAdmin(admin.ModelAdmin): | ||||
|     list_display = ('team', 'problem', 'valid',) | ||||
|     search_fields = ('team__name', 'team__trigram',) | ||||
|     list_filter = ('problem', 'valid',) | ||||
|  | ||||
|  | ||||
| @admin.register(Video) | ||||
| class VideoAdmin(admin.ModelAdmin): | ||||
|     list_display = ('participation', 'link',) | ||||
|     search_fields = ('participation__team__name', 'participation__team__trigram', 'link',) | ||||
|  | ||||
|  | ||||
| @admin.register(Question) | ||||
| class QuestionAdmin(admin.ModelAdmin): | ||||
|     list_display = ('participation', 'question',) | ||||
|     search_fields = ('participation__team__name', 'participation__team__trigram', 'question',) | ||||
|  | ||||
|  | ||||
| @admin.register(Phase) | ||||
| class PhaseAdmin(admin.ModelAdmin): | ||||
|     list_display = ('phase_number', 'start', 'end',) | ||||
|     ordering = ('phase_number', 'start',) | ||||
							
								
								
									
										18
									
								
								apps/participation/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								apps/participation/apps.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.apps import AppConfig | ||||
| from django.db.models.signals import post_save, pre_delete, pre_save | ||||
|  | ||||
|  | ||||
| class ParticipationConfig(AppConfig): | ||||
|     """ | ||||
|     The participation app contains the data about the teams, videos, ... | ||||
|     """ | ||||
|     name = 'participation' | ||||
|  | ||||
|     def ready(self): | ||||
|         from participation.signals import create_team_participation, delete_related_videos, update_mailing_list | ||||
|         pre_save.connect(update_mailing_list, "participation.Team") | ||||
|         pre_delete.connect(delete_related_videos, "participation.Participation") | ||||
|         post_save.connect(create_team_participation, "participation.Team") | ||||
							
								
								
									
										194
									
								
								apps/participation/forms.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										194
									
								
								apps/participation/forms.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,194 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| import re | ||||
|  | ||||
| from bootstrap_datepicker_plus import DateTimePickerInput | ||||
| from django import forms | ||||
| from django.core.exceptions import ObjectDoesNotExist, ValidationError | ||||
| from django.db.models import Q | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
| from .models import Participation, Phase, Question, Team, Video | ||||
|  | ||||
|  | ||||
| class TeamForm(forms.ModelForm): | ||||
|     """ | ||||
|     Form to create a team, with the name and the trigram, | ||||
|     and if the team accepts that Animath diffuse the videos. | ||||
|     """ | ||||
|     def clean_trigram(self): | ||||
|         trigram = self.cleaned_data["trigram"].upper() | ||||
|         if not re.match("[A-Z]{3}", trigram): | ||||
|             raise ValidationError(_("The trigram must be composed of three uppercase letters.")) | ||||
|         return trigram | ||||
|  | ||||
|     class Meta: | ||||
|         model = Team | ||||
|         fields = ('name', 'trigram', 'grant_animath_access_videos',) | ||||
|  | ||||
|  | ||||
| class JoinTeamForm(forms.ModelForm): | ||||
|     """ | ||||
|     Form to join a team by the access code. | ||||
|     """ | ||||
|     def clean_access_code(self): | ||||
|         access_code = self.cleaned_data["access_code"] | ||||
|         if not Team.objects.filter(access_code=access_code).exists(): | ||||
|             raise ValidationError(_("No team was found with this access code.")) | ||||
|         return access_code | ||||
|  | ||||
|     def clean(self): | ||||
|         cleaned_data = super().clean() | ||||
|         if "access_code" in cleaned_data: | ||||
|             team = Team.objects.get(access_code=cleaned_data["access_code"]) | ||||
|             self.instance = team | ||||
|         return cleaned_data | ||||
|  | ||||
|     class Meta: | ||||
|         model = Team | ||||
|         fields = ('access_code',) | ||||
|  | ||||
|  | ||||
| class ParticipationForm(forms.ModelForm): | ||||
|     """ | ||||
|     Form to update the problem of a team participation. | ||||
|     """ | ||||
|     class Meta: | ||||
|         model = Participation | ||||
|         fields = ('problem',) | ||||
|  | ||||
|  | ||||
| class RequestValidationForm(forms.Form): | ||||
|     """ | ||||
|     Form to ask about validation. | ||||
|     """ | ||||
|     _form_type = forms.CharField( | ||||
|         initial="RequestValidationForm", | ||||
|         widget=forms.HiddenInput(), | ||||
|     ) | ||||
|  | ||||
|     engagement = forms.BooleanField( | ||||
|         label=_("I engage myself to participate to the whole \"Correspondances\"."), | ||||
|         required=True, | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class ValidateParticipationForm(forms.Form): | ||||
|     """ | ||||
|     Form to let administrators to accept or refuse a team. | ||||
|     """ | ||||
|     _form_type = forms.CharField( | ||||
|         initial="ValidateParticipationForm", | ||||
|         widget=forms.HiddenInput(), | ||||
|     ) | ||||
|  | ||||
|     message = forms.CharField( | ||||
|         label=_("Message to address to the team:"), | ||||
|         widget=forms.Textarea(), | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class UploadVideoForm(forms.ModelForm): | ||||
|     """ | ||||
|     Form to upload a video, for a solution or a synthesis. | ||||
|     """ | ||||
|     class Meta: | ||||
|         model = Video | ||||
|         fields = ('link',) | ||||
|  | ||||
|     def clean(self): | ||||
|         if Phase.current_phase().phase_number != 1 and Phase.current_phase().phase_number != 4 and self.instance.link: | ||||
|             self.add_error("link", _("You can't upload your video after the deadline.")) | ||||
|         return super().clean() | ||||
|  | ||||
|  | ||||
| class ReceiveParticipationForm(forms.ModelForm): | ||||
|     """ | ||||
|     Update the received participation of a participation. | ||||
|     """ | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.fields["received_participation"].queryset = Participation.objects.filter( | ||||
|             ~Q(pk=self.instance.pk) & Q(problem=self.instance.problem, valid=True) | ||||
|         ) | ||||
|  | ||||
|     class Meta: | ||||
|         model = Participation | ||||
|         fields = ('received_participation',) | ||||
|  | ||||
|  | ||||
| class SendParticipationForm(forms.ModelForm): | ||||
|     """ | ||||
|     Update the sent participation of a participation. | ||||
|     """ | ||||
|     sent_participation = forms.ModelChoiceField( | ||||
|         queryset=Participation.objects, | ||||
|         label=lambda: _("Send to team"), | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         try: | ||||
|             self.fields["sent_participation"].initial = self.instance.sent_participation | ||||
|         except ObjectDoesNotExist:     # No sent participation | ||||
|             pass | ||||
|         self.fields["sent_participation"].queryset = Participation.objects.filter( | ||||
|             ~Q(pk=self.instance.pk) & Q(problem=self.instance.problem, valid=True) | ||||
|         ) | ||||
|  | ||||
|     def clean(self, commit=True): | ||||
|         cleaned_data = super().clean() | ||||
|         if "sent_participation" in cleaned_data: | ||||
|             participation = cleaned_data["sent_participation"] | ||||
|             participation.received_participation = self.instance | ||||
|             self.instance = participation | ||||
|         return cleaned_data | ||||
|  | ||||
|     class Meta: | ||||
|         model = Participation | ||||
|         fields = ('sent_participation',) | ||||
|  | ||||
|  | ||||
| class QuestionForm(forms.ModelForm): | ||||
|     """ | ||||
|     Create or update a question. | ||||
|     """ | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.fields["question"].widget.attrs.update({"placeholder": _("How did you get the idea to ...?")}) | ||||
|  | ||||
|     def clean(self): | ||||
|         if Phase.current_phase().phase_number != 2: | ||||
|             self.add_error(None, _("You can only create or update a question during the second phase.")) | ||||
|         return super().clean() | ||||
|  | ||||
|     class Meta: | ||||
|         model = Question | ||||
|         fields = ('question',) | ||||
|  | ||||
|  | ||||
| class PhaseForm(forms.ModelForm): | ||||
|     """ | ||||
|     Form to update the calendar of a phase. | ||||
|     """ | ||||
|     class Meta: | ||||
|         model = Phase | ||||
|         fields = ('start', 'end',) | ||||
|         widgets = { | ||||
|             'start': DateTimePickerInput(format='%d/%m/%Y %H:%M'), | ||||
|             'end': DateTimePickerInput(format='%d/%m/%Y %H:%M'), | ||||
|         } | ||||
|  | ||||
|     def clean(self): | ||||
|         # Ensure that dates are in a right order | ||||
|         cleaned_data = super().clean() | ||||
|         start = cleaned_data["start"] | ||||
|         end = cleaned_data["end"] | ||||
|         if end <= start: | ||||
|             self.add_error("end", _("Start date must be before the end date.")) | ||||
|         if Phase.objects.filter(phase_number__lt=self.instance.phase_number, end__gt=start).exists(): | ||||
|             self.add_error("start", _("This phase must start after the previous phases.")) | ||||
|         if Phase.objects.filter(phase_number__gt=self.instance.phase_number, start__lt=end).exists(): | ||||
|             self.add_error("end", _("This phase must end after the next phases.")) | ||||
|         return cleaned_data | ||||
							
								
								
									
										2
									
								
								apps/participation/management/commands/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								apps/participation/management/commands/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
| @@ -0,0 +1,92 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| import os | ||||
|  | ||||
| from asgiref.sync import async_to_sync | ||||
| from corres2math.matrix import Matrix, RoomPreset, RoomVisibility | ||||
| from django.core.management import BaseCommand | ||||
| from registration.models import AdminRegistration, Registration | ||||
|  | ||||
|  | ||||
| class Command(BaseCommand): | ||||
|     def handle(self, *args, **options): | ||||
|         Matrix.set_display_name("Bot des Correspondances") | ||||
|  | ||||
|         if not os.getenv("SYNAPSE_PASSWORD"): | ||||
|             avatar_uri = "plop" | ||||
|         else:  # pragma: no cover | ||||
|             if not os.path.isfile(".matrix_avatar"): | ||||
|                 stat_file = os.stat("corres2math/static/logo.png") | ||||
|                 with open("corres2math/static/logo.png", "rb") as f: | ||||
|                     resp = Matrix.upload(f, filename="logo.png", content_type="image/png", | ||||
|                                          filesize=stat_file.st_size)[0][0] | ||||
|                 avatar_uri = resp.content_uri | ||||
|                 with open(".matrix_avatar", "w") as f: | ||||
|                     f.write(avatar_uri) | ||||
|                 Matrix.set_avatar(avatar_uri) | ||||
|  | ||||
|             with open(".matrix_avatar", "r") as f: | ||||
|                 avatar_uri = f.read().rstrip(" \t\r\n") | ||||
|  | ||||
|         if not async_to_sync(Matrix.resolve_room_alias)("#faq:correspondances-maths.fr"): | ||||
|             Matrix.create_room( | ||||
|                 visibility=RoomVisibility.public, | ||||
|                 alias="faq", | ||||
|                 name="FAQ", | ||||
|                 topic="Posez toutes vos questions ici !", | ||||
|                 federate=False, | ||||
|                 preset=RoomPreset.public_chat, | ||||
|             ) | ||||
|  | ||||
|         if not async_to_sync(Matrix.resolve_room_alias)("#annonces:correspondances-maths.fr"): | ||||
|             Matrix.create_room( | ||||
|                 visibility=RoomVisibility.public, | ||||
|                 alias="annonces", | ||||
|                 name="Annonces", | ||||
|                 topic="Informations importantes des Correspondances", | ||||
|                 federate=False, | ||||
|                 preset=RoomPreset.public_chat, | ||||
|             ) | ||||
|  | ||||
|         if not async_to_sync(Matrix.resolve_room_alias)("#je-cherche-une-equipe:correspondances-maths.fr"): | ||||
|             Matrix.create_room( | ||||
|                 visibility=RoomVisibility.public, | ||||
|                 alias="je-cherche-une-equipe", | ||||
|                 name="Je cherche une équipe", | ||||
|                 topic="Le Tinder des Correspondances", | ||||
|                 federate=False, | ||||
|                 preset=RoomPreset.public_chat, | ||||
|             ) | ||||
|  | ||||
|         if not async_to_sync(Matrix.resolve_room_alias)("#flood:correspondances-maths.fr"): | ||||
|             Matrix.create_room( | ||||
|                 visibility=RoomVisibility.public, | ||||
|                 alias="flood", | ||||
|                 name="Flood", | ||||
|                 topic="Discutez de tout et de rien !", | ||||
|                 federate=False, | ||||
|                 preset=RoomPreset.public_chat, | ||||
|             ) | ||||
|  | ||||
|         Matrix.set_room_avatar("#annonces:correspondances-maths.fr", avatar_uri) | ||||
|         Matrix.set_room_avatar("#faq:correspondances-maths.fr", avatar_uri) | ||||
|         Matrix.set_room_avatar("#je-cherche-une-equipe:correspondances-maths.fr", avatar_uri) | ||||
|         Matrix.set_room_avatar("#flood:correspondances-maths.fr", avatar_uri) | ||||
|  | ||||
|         Matrix.set_room_power_level_event("#annonces:correspondances-maths.fr", "events_default", 50) | ||||
|  | ||||
|         for r in Registration.objects.all(): | ||||
|             Matrix.invite("#annonces:correspondances-maths.fr", f"@{r.matrix_username}:correspondances-maths.fr") | ||||
|             Matrix.invite("#faq:correspondances-maths.fr", f"@{r.matrix_username}:correspondances-maths.fr") | ||||
|             Matrix.invite("#je-cherche-une-equipe:correspondances-maths.fr", | ||||
|                           f"@{r.matrix_username}:correspondances-maths.fr") | ||||
|             Matrix.invite("#flood:correspondances-maths.fr", f"@{r.matrix_username}:correspondances-maths.fr") | ||||
|  | ||||
|         for admin in AdminRegistration.objects.all(): | ||||
|             Matrix.set_room_power_level("#annonces:correspondances-maths.fr", | ||||
|                                         f"@{admin.matrix_username}:correspondances-maths.fr", 95) | ||||
|             Matrix.set_room_power_level("#faq:correspondances-maths.fr", | ||||
|                                         f"@{admin.matrix_username}:correspondances-maths.fr", 95) | ||||
|             Matrix.set_room_power_level("#flood:correspondances-maths.fr", | ||||
|                                         f"@{admin.matrix_username}:correspondances-maths.fr", 95) | ||||
							
								
								
									
										43
									
								
								apps/participation/management/commands/fix_sympa_lists.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								apps/participation/management/commands/fix_sympa_lists.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from corres2math.lists import get_sympa_client | ||||
| from django.core.management import BaseCommand | ||||
| from django.db.models import Q | ||||
| from participation.models import Team | ||||
| from registration.models import CoachRegistration, StudentRegistration | ||||
|  | ||||
|  | ||||
| class Command(BaseCommand): | ||||
|     def handle(self, *args, **options): | ||||
|         """ | ||||
|         Create Sympa mailing lists and register teams. | ||||
|         """ | ||||
|         sympa = get_sympa_client() | ||||
|  | ||||
|         sympa.create_list("equipes", "Équipes des Correspondances", "hotline", | ||||
|                           "Liste de diffusion pour contacter toutes les équipes validées des Correspondances.", | ||||
|                           "education", raise_error=False) | ||||
|         sympa.create_list("equipes-non-valides", "Équipes des Correspondances", "hotline", | ||||
|                           "Liste de diffusion pour contacter toutes les équipes non-validées des Correspondances.", | ||||
|                           "education", raise_error=False) | ||||
|  | ||||
|         for problem in range(1, 4): | ||||
|             sympa.create_list(f"probleme-{problem}", | ||||
|                               f"Équipes des Correspondances participant au problème {problem}", "hotline", | ||||
|                               f"Liste de diffusion pour contacter les équipes participant au problème {problem}" | ||||
|                               f" des Correspondances.", "education", raise_error=False) | ||||
|  | ||||
|         for team in Team.objects.filter(participation__valid=True).all(): | ||||
|             team.create_mailing_list() | ||||
|             sympa.subscribe(team.email, "equipes", f"Equipe {team.name}", True) | ||||
|             sympa.subscribe(team.email, f"probleme-{team.participation.problem}", f"Equipe {team.name}", True) | ||||
|  | ||||
|         for team in Team.objects.filter(Q(participation__valid=False) | Q(participation__valid__isnull=True)).all(): | ||||
|             team.create_mailing_list() | ||||
|             sympa.subscribe(team.email, "equipes-non-valides", f"Equipe {team.name}", True) | ||||
|  | ||||
|         for student in StudentRegistration.objects.filter(team__isnull=False).all(): | ||||
|             sympa.subscribe(student.user.email, f"equipe-{student.team.trigram.lower()}", True, f"{student}") | ||||
|         for coach in CoachRegistration.objects.filter(team__isnull=False).all(): | ||||
|             sympa.subscribe(coach.user.email, f"equipe-{coach.team.trigram.lower()}", True, f"{coach}") | ||||
							
								
								
									
										44
									
								
								apps/participation/management/commands/setup_third_phase.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								apps/participation/management/commands/setup_third_phase.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from corres2math.matrix import Matrix, RoomVisibility | ||||
| from django.core.management import BaseCommand | ||||
| from participation.models import Participation | ||||
|  | ||||
|  | ||||
| class Command(BaseCommand): | ||||
|     def handle(self, *args, **options): | ||||
|         for participation in Participation.objects.filter(valid=True).all(): | ||||
|             for i, question in enumerate(participation.questions.order_by("id").all()): | ||||
|                 solution_author = participation.received_participation.team | ||||
|                 alias = f"equipe-{solution_author.trigram.lower()}-question-{i}" | ||||
|                 room_id = f"#{alias}:correspondances-maths.fr" | ||||
|                 Matrix.create_room( | ||||
|                     visibility=RoomVisibility.public, | ||||
|                     alias=alias, | ||||
|                     name=f"Solution équipe {solution_author.trigram} - question {i+1}", | ||||
|                     topic=f"Échange entre l'équipe {solution_author.name} ({solution_author.trigram}) " | ||||
|                           f"et l'équipe {participation.team.name} ({participation.team.trigram}) " | ||||
|                           f"autour de la question {i+1} sur le problème {participation.problem}", | ||||
|                     federate=False, | ||||
|                     invite=[f"@{registration.matrix_username}:correspondances-maths.fr" for registration in | ||||
|                             list(participation.team.students.all()) + list(participation.team.coachs.all()) + | ||||
|                             list(solution_author.students.all()) + list(solution_author.coachs.all())], | ||||
|                 ) | ||||
|                 Matrix.set_room_power_level_event(room_id, "events_default", 21) | ||||
|                 for registration in solution_author.students.all(): | ||||
|                     Matrix.set_room_power_level(room_id, | ||||
|                                                 f"@{registration.matrix_username}:correspondances-maths.fr", 42) | ||||
|  | ||||
|                 Matrix.send_message(room_id, "Bienvenue dans la troisième phase des Correspondances !") | ||||
|                 Matrix.send_message(room_id, f"L'équipe {participation.team.name} a visionné la vidéo de l'équipe " | ||||
|                                     f"{solution_author.name} sur le problème {participation.problem}, et a posé " | ||||
|                                     "une série de questions.") | ||||
|                 Matrix.send_message(room_id, "L'équipe ayant composé la vidéo doit maintenant proposer une réponse.") | ||||
|                 Matrix.send_message(room_id, "Une fois la réponse apportée, vous pourrez ensuite échanger plus " | ||||
|                                              "librement autour de la question, au travers de ce canal.") | ||||
|                 Matrix.send_message(room_id, "**Question posée :**", formatted_body="<strong>Question posée :</strong>") | ||||
|                 Matrix.send_message(room_id, question.question, | ||||
|                                     formatted_body=f"<font color=\"#ff0000\">{question.question}</font>") | ||||
|  | ||||
|                 # TODO Setup the bot the set the power level of all members of the room to 42 | ||||
							
								
								
									
										138
									
								
								apps/participation/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										138
									
								
								apps/participation/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,138 @@ | ||||
| # Generated by Django 3.1.3 on 2020-11-04 12:05 | ||||
|  | ||||
| import django.core.validators | ||||
| from django.db import migrations, models | ||||
| import django.utils.timezone | ||||
|  | ||||
|  | ||||
| def register_phases(apps, _): | ||||
|     """ | ||||
|     Import the different phases of the action | ||||
|     """ | ||||
|     Phase = apps.get_model("participation", "phase") | ||||
|     Phase.objects.get_or_create( | ||||
|         phase_number=1, | ||||
|         description="Soumission des vidéos", | ||||
|     ) | ||||
|     Phase.objects.get_or_create( | ||||
|         phase_number=2, | ||||
|         description="Phase de questions", | ||||
|     ) | ||||
|     Phase.objects.get_or_create( | ||||
|         phase_number=3, | ||||
|         description="Phase d'échanges entre les équipes", | ||||
|     ) | ||||
|     Phase.objects.get_or_create( | ||||
|         phase_number=4, | ||||
|         description="Synthèse de l'échange", | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def reverse_phase_registering(apps, _):  # pragma: no cover | ||||
|     """ | ||||
|     Drop all phases in order to unapply this migration. | ||||
|     """ | ||||
|     Phase = apps.get_model("participation", "phase") | ||||
|     Phase.objects.all().delete() | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     initial = True | ||||
|  | ||||
|     dependencies = [ | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.CreateModel( | ||||
|             name='Participation', | ||||
|             fields=[ | ||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||
|                 ('problem', models.IntegerField(choices=[(1, 'Problem #1'), (2, 'Problem #2'), (3, 'Problem #3')], default=None, null=True, verbose_name='problem number')), | ||||
|                 ('valid', models.BooleanField(default=None, help_text='The video got the validation of the administrators.', null=True, verbose_name='valid')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'participation', | ||||
|                 'verbose_name_plural': 'participations', | ||||
|             }, | ||||
|         ), | ||||
|         migrations.CreateModel( | ||||
|             name='Phase', | ||||
|             fields=[ | ||||
|                 ('phase_number', models.AutoField(primary_key=True, serialize=False, unique=True, verbose_name='phase number')), | ||||
|                 ('description', models.CharField(max_length=255, verbose_name='phase description')), | ||||
|                 ('start', models.DateTimeField(default=django.utils.timezone.now, verbose_name='start date of the given phase')), | ||||
|                 ('end', models.DateTimeField(default=django.utils.timezone.now, verbose_name='end date of the given phase')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'phase', | ||||
|                 'verbose_name_plural': 'phases', | ||||
|             }, | ||||
|         ), | ||||
|         migrations.CreateModel( | ||||
|             name='Question', | ||||
|             fields=[ | ||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||
|                 ('question', models.TextField(verbose_name='question')), | ||||
|             ], | ||||
|         ), | ||||
|         migrations.CreateModel( | ||||
|             name='Team', | ||||
|             fields=[ | ||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||
|                 ('name', models.CharField(max_length=255, unique=True, verbose_name='name')), | ||||
|                 ('trigram', models.CharField(help_text='The trigram must be composed of three uppercase letters.', max_length=3, unique=True, validators=[django.core.validators.RegexValidator('[A-Z]{3}')], verbose_name='trigram')), | ||||
|                 ('access_code', models.CharField(help_text='The access code let other people to join the team.', max_length=6, verbose_name='access code')), | ||||
|                 ('grant_animath_access_videos', models.BooleanField(default=False, help_text='Give the authorisation to publish the video on the main website to promote the action.', verbose_name='Grant Animath to publish my video')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'team', | ||||
|                 'verbose_name_plural': 'teams', | ||||
|             }, | ||||
|         ), | ||||
|         migrations.CreateModel( | ||||
|             name='Video', | ||||
|             fields=[ | ||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||
|                 ('link', models.URLField(help_text='The full video link.', verbose_name='link')), | ||||
|                 ('valid', models.BooleanField(default=None, help_text='The video got the validation of the administrators.', null=True, verbose_name='valid')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'video', | ||||
|                 'verbose_name_plural': 'videos', | ||||
|             }, | ||||
|         ), | ||||
|         migrations.AddIndex( | ||||
|             model_name='team', | ||||
|             index=models.Index(fields=['trigram'], name='participati_trigram_239255_idx'), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name='question', | ||||
|             name='participation', | ||||
|             field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='questions', to='participation.participation', verbose_name='participation'), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name='participation', | ||||
|             name='received_participation', | ||||
|             field=models.OneToOneField(default=None, null=True, on_delete=django.db.models.deletion.PROTECT, related_name='sent_participation', to='participation.participation', verbose_name='received participation'), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name='participation', | ||||
|             name='solution', | ||||
|             field=models.OneToOneField(default=None, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='participation_solution', to='participation.video', verbose_name='solution video'), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name='participation', | ||||
|             name='synthesis', | ||||
|             field=models.OneToOneField(default=None, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='participation_synthesis', to='participation.video', verbose_name='synthesis video'), | ||||
|         ), | ||||
|         migrations.AddField( | ||||
|             model_name='participation', | ||||
|             name='team', | ||||
|             field=models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to='participation.team', verbose_name='team'), | ||||
|         ), | ||||
|         migrations.RunPython( | ||||
|             register_phases, | ||||
|             reverse_code=reverse_phase_registering, | ||||
|         ) | ||||
|     ] | ||||
							
								
								
									
										2
									
								
								apps/participation/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								apps/participation/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
							
								
								
									
										307
									
								
								apps/participation/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										307
									
								
								apps/participation/models.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,307 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| import os | ||||
| import re | ||||
|  | ||||
| from corres2math.lists import get_sympa_client | ||||
| from corres2math.matrix import Matrix, RoomPreset, RoomVisibility | ||||
| from django.core.exceptions import ObjectDoesNotExist | ||||
| from django.core.validators import RegexValidator | ||||
| from django.db import models | ||||
| from django.db.models import Index | ||||
| from django.template.loader import render_to_string | ||||
| from django.urls import reverse_lazy | ||||
| from django.utils import timezone | ||||
| from django.utils.crypto import get_random_string | ||||
| from django.utils.text import format_lazy | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
|  | ||||
| class Team(models.Model): | ||||
|     """ | ||||
|     The Team model represents a real team that participates to the Correspondances. | ||||
|     This only includes the registration detail. | ||||
|     """ | ||||
|     name = models.CharField( | ||||
|         max_length=255, | ||||
|         verbose_name=_("name"), | ||||
|         unique=True, | ||||
|     ) | ||||
|  | ||||
|     trigram = models.CharField( | ||||
|         max_length=3, | ||||
|         verbose_name=_("trigram"), | ||||
|         help_text=_("The trigram must be composed of three uppercase letters."), | ||||
|         unique=True, | ||||
|         validators=[RegexValidator("[A-Z]{3}")], | ||||
|     ) | ||||
|  | ||||
|     access_code = models.CharField( | ||||
|         max_length=6, | ||||
|         verbose_name=_("access code"), | ||||
|         help_text=_("The access code let other people to join the team."), | ||||
|     ) | ||||
|  | ||||
|     grant_animath_access_videos = models.BooleanField( | ||||
|         verbose_name=_("Grant Animath to publish my video"), | ||||
|         help_text=_("Give the authorisation to publish the video on the main website to promote the action."), | ||||
|         default=False, | ||||
|     ) | ||||
|  | ||||
|     @property | ||||
|     def email(self): | ||||
|         """ | ||||
|         :return: The mailing list to contact the team members. | ||||
|         """ | ||||
|         return f"equipe-{self.trigram.lower()}@{os.getenv('SYMPA_HOST', 'localhost')}" | ||||
|  | ||||
|     def create_mailing_list(self): | ||||
|         """ | ||||
|         Create a new Sympa mailing list to contact the team. | ||||
|         """ | ||||
|         get_sympa_client().create_list( | ||||
|             f"equipe-{self.trigram.lower()}", | ||||
|             f"Équipe {self.name} ({self.trigram})", | ||||
|             "hotline",  # TODO Use a custom sympa template | ||||
|             f"Liste de diffusion pour contacter l'équipe {self.name} des Correspondances", | ||||
|             "education", | ||||
|             raise_error=False, | ||||
|         ) | ||||
|         if self.pk and self.participation.valid:  # pragma: no cover | ||||
|             get_sympa_client().subscribe(self.email, "equipes", False, f"Equipe {self.name}") | ||||
|             get_sympa_client().subscribe(self.email, f"probleme-{self.participation.problem}", False, | ||||
|                                          f"Equipe {self.name}") | ||||
|         else: | ||||
|             get_sympa_client().subscribe(self.email, "equipes-non-valides", False) | ||||
|  | ||||
|     def delete_mailing_list(self): | ||||
|         """ | ||||
|         Drop the Sympa mailing list, if the team is empty or if the trigram changed. | ||||
|         """ | ||||
|         if self.participation.valid:  # pragma: no cover | ||||
|             get_sympa_client().unsubscribe(self.email, "equipes", False) | ||||
|             get_sympa_client().unsubscribe(self.email, f"probleme-{self.participation.problem}", False) | ||||
|         else: | ||||
|             get_sympa_client().unsubscribe(self.email, "equipes-non-valides", False) | ||||
|         get_sympa_client().delete_list(f"equipe-{self.trigram}") | ||||
|  | ||||
|     def save(self, *args, **kwargs): | ||||
|         if not self.access_code: | ||||
|             # if the team got created, generate the access code, create the contact mailing list | ||||
|             # and create a dedicated Matrix room. | ||||
|             self.access_code = get_random_string(6) | ||||
|             self.create_mailing_list() | ||||
|  | ||||
|             Matrix.create_room( | ||||
|                 visibility=RoomVisibility.private, | ||||
|                 name=f"#équipe-{self.trigram.lower()}", | ||||
|                 alias=f"equipe-{self.trigram.lower()}", | ||||
|                 topic=f"Discussion de l'équipe {self.name}", | ||||
|                 preset=RoomPreset.private_chat, | ||||
|             ) | ||||
|  | ||||
|         return super().save(*args, **kwargs) | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse_lazy("participation:team_detail", args=(self.pk,)) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return _("Team {name} ({trigram})").format(name=self.name, trigram=self.trigram) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("team") | ||||
|         verbose_name_plural = _("teams") | ||||
|         indexes = [ | ||||
|             Index(fields=("trigram", )), | ||||
|         ] | ||||
|  | ||||
|  | ||||
| class Participation(models.Model): | ||||
|     """ | ||||
|     The Participation model contains all data that are related to the participation: | ||||
|     chosen problem, validity status, videos,... | ||||
|     """ | ||||
|     team = models.OneToOneField( | ||||
|         Team, | ||||
|         on_delete=models.CASCADE, | ||||
|         verbose_name=_("team"), | ||||
|     ) | ||||
|  | ||||
|     problem = models.IntegerField( | ||||
|         choices=[(i, format_lazy(_("Problem #{problem:d}"), problem=i)) for i in range(1, 4)], | ||||
|         null=True, | ||||
|         default=None, | ||||
|         verbose_name=_("problem number"), | ||||
|     ) | ||||
|  | ||||
|     valid = models.BooleanField( | ||||
|         null=True, | ||||
|         default=None, | ||||
|         verbose_name=_("valid"), | ||||
|         help_text=_("The video got the validation of the administrators."), | ||||
|     ) | ||||
|  | ||||
|     solution = models.OneToOneField( | ||||
|         "participation.Video", | ||||
|         on_delete=models.SET_NULL, | ||||
|         related_name="participation_solution", | ||||
|         null=True, | ||||
|         default=None, | ||||
|         verbose_name=_("solution video"), | ||||
|     ) | ||||
|  | ||||
|     received_participation = models.OneToOneField( | ||||
|         "participation.Participation", | ||||
|         on_delete=models.PROTECT, | ||||
|         related_name="sent_participation", | ||||
|         null=True, | ||||
|         default=None, | ||||
|         verbose_name=_("received participation"), | ||||
|     ) | ||||
|  | ||||
|     synthesis = models.OneToOneField( | ||||
|         "participation.Video", | ||||
|         on_delete=models.SET_NULL, | ||||
|         related_name="participation_synthesis", | ||||
|         null=True, | ||||
|         default=None, | ||||
|         verbose_name=_("synthesis video"), | ||||
|     ) | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse_lazy("participation:participation_detail", args=(self.pk,)) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return _("Participation of the team {name} ({trigram})").format(name=self.team.name, trigram=self.team.trigram) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("participation") | ||||
|         verbose_name_plural = _("participations") | ||||
|  | ||||
|  | ||||
| class Video(models.Model): | ||||
|     """ | ||||
|     The Video model only contains a link and a validity status. | ||||
|     """ | ||||
|     link = models.URLField( | ||||
|         verbose_name=_("link"), | ||||
|         help_text=_("The full video link."), | ||||
|     ) | ||||
|  | ||||
|     valid = models.BooleanField( | ||||
|         null=True, | ||||
|         default=None, | ||||
|         verbose_name=_("valid"), | ||||
|         help_text=_("The video got the validation of the administrators."), | ||||
|     ) | ||||
|  | ||||
|     @property | ||||
|     def participation(self): | ||||
|         """ | ||||
|         Retrives the participation that is associated to this video, | ||||
|         whatever it is a solution or a synthesis. | ||||
|         """ | ||||
|         try: | ||||
|             # If this is a solution | ||||
|             return self.participation_solution | ||||
|         except ObjectDoesNotExist: | ||||
|             # If this is a synthesis | ||||
|             return self.participation_synthesis | ||||
|  | ||||
|     @property | ||||
|     def platform(self): | ||||
|         """ | ||||
|         According to the link, retrieve the platform that is used to upload the video. | ||||
|         """ | ||||
|         if "youtube.com" in self.link or "youtu.be" in self.link: | ||||
|             return "youtube" | ||||
|         return "unknown" | ||||
|  | ||||
|     @property | ||||
|     def youtube_code(self): | ||||
|         """ | ||||
|         If the video is uploaded on Youtube, search in the URL the video code. | ||||
|         """ | ||||
|         return re.compile("(https?://|)(www\\.|)(youtube\\.com/watch\\?v=|youtu\\.be/)([a-zA-Z0-9-_]*)?.*?")\ | ||||
|             .match(self.link).group(4) | ||||
|  | ||||
|     def as_iframe(self): | ||||
|         """ | ||||
|         Generate the HTML code to embed the video in an iframe, according to the type of the host platform. | ||||
|         """ | ||||
|         if self.platform == "youtube": | ||||
|             return render_to_string("participation/youtube_iframe.html", context=dict(youtube_code=self.youtube_code)) | ||||
|         return None | ||||
|  | ||||
|     def __str__(self): | ||||
|         return _("Video of team {name} ({trigram})")\ | ||||
|             .format(name=self.participation.team.name, trigram=self.participation.team.trigram) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("video") | ||||
|         verbose_name_plural = _("videos") | ||||
|  | ||||
|  | ||||
| class Question(models.Model): | ||||
|     """ | ||||
|     Question to ask to the team that sent a solution. | ||||
|     """ | ||||
|     participation = models.ForeignKey( | ||||
|         Participation, | ||||
|         on_delete=models.CASCADE, | ||||
|         verbose_name=_("participation"), | ||||
|         related_name="questions", | ||||
|     ) | ||||
|  | ||||
|     question = models.TextField( | ||||
|         verbose_name=_("question"), | ||||
|     ) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return self.question | ||||
|  | ||||
|  | ||||
| class Phase(models.Model): | ||||
|     """ | ||||
|     The Phase model corresponds to the dates of the phase. | ||||
|     """ | ||||
|     phase_number = models.AutoField( | ||||
|         primary_key=True, | ||||
|         unique=True, | ||||
|         verbose_name=_("phase number"), | ||||
|     ) | ||||
|  | ||||
|     description = models.CharField( | ||||
|         max_length=255, | ||||
|         verbose_name=_("phase description"), | ||||
|     ) | ||||
|  | ||||
|     start = models.DateTimeField( | ||||
|         verbose_name=_("start date of the given phase"), | ||||
|         default=timezone.now, | ||||
|     ) | ||||
|  | ||||
|     end = models.DateTimeField( | ||||
|         verbose_name=_("end date of the given phase"), | ||||
|         default=timezone.now, | ||||
|     ) | ||||
|  | ||||
|     @classmethod | ||||
|     def current_phase(cls) -> "Phase": | ||||
|         """ | ||||
|         Retrieve the current phase of this day | ||||
|         """ | ||||
|         qs = Phase.objects.filter(start__lte=timezone.now(), end__gte=timezone.now()) | ||||
|         if qs.exists(): | ||||
|             return qs.get() | ||||
|         qs = Phase.objects.filter(start__lte=timezone.now()).order_by("phase_number").all() | ||||
|         return qs.last() if qs.exists() else None | ||||
|  | ||||
|     def __str__(self): | ||||
|         return _("Phase {phase_number:d} starts on {start:%Y-%m-%d %H:%M} and ends on {end:%Y-%m-%d %H:%M}")\ | ||||
|             .format(phase_number=self.phase_number, start=self.start, end=self.end) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("phase") | ||||
|         verbose_name_plural = _("phases") | ||||
							
								
								
									
										36
									
								
								apps/participation/search_indexes.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								apps/participation/search_indexes.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from haystack import indexes | ||||
|  | ||||
| from .models import Participation, Team, Video | ||||
|  | ||||
|  | ||||
| class TeamIndex(indexes.ModelSearchIndex, indexes.Indexable): | ||||
|     """ | ||||
|     Index all teams by their name and trigram. | ||||
|     """ | ||||
|     text = indexes.NgramField(document=True, use_template=True) | ||||
|  | ||||
|     class Meta: | ||||
|         model = Team | ||||
|  | ||||
|  | ||||
| class ParticipationIndex(indexes.ModelSearchIndex, indexes.Indexable): | ||||
|     """ | ||||
|     Index all participations by their team name and team trigram. | ||||
|     """ | ||||
|     text = indexes.NgramField(document=True, use_template=True) | ||||
|  | ||||
|     class Meta: | ||||
|         model = Participation | ||||
|  | ||||
|  | ||||
| class VideoIndex(indexes.ModelSearchIndex, indexes.Indexable): | ||||
|     """ | ||||
|     Index all teams by their team name and team trigram. | ||||
|     """ | ||||
|     text = indexes.NgramField(document=True, use_template=True) | ||||
|  | ||||
|     class Meta: | ||||
|         model = Video | ||||
							
								
								
									
										46
									
								
								apps/participation/signals.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								apps/participation/signals.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from corres2math.lists import get_sympa_client | ||||
| from participation.models import Participation, Team, Video | ||||
|  | ||||
|  | ||||
| def create_team_participation(instance, created, **_): | ||||
|     """ | ||||
|     When a team got created, create an associated team and create Video objects. | ||||
|     """ | ||||
|     participation = Participation.objects.get_or_create(team=instance)[0] | ||||
|     if not participation.solution: | ||||
|         participation.solution = Video.objects.create() | ||||
|     if not participation.synthesis: | ||||
|         participation.synthesis = Video.objects.create() | ||||
|     participation.save() | ||||
|     if not created: | ||||
|         participation.team.create_mailing_list() | ||||
|  | ||||
|  | ||||
| def update_mailing_list(instance: Team, **_): | ||||
|     """ | ||||
|     When a team name or trigram got updated, update mailing lists and Matrix rooms | ||||
|     """ | ||||
|     if instance.pk: | ||||
|         old_team = Team.objects.get(pk=instance.pk) | ||||
|         if old_team.name != instance.name or old_team.trigram != instance.trigram: | ||||
|             # TODO Rename Matrix room | ||||
|             # Delete old mailing list, create a new one | ||||
|             old_team.delete_mailing_list() | ||||
|             instance.create_mailing_list() | ||||
|             # Subscribe all team members in the mailing list | ||||
|             for student in instance.students.all(): | ||||
|                 get_sympa_client().subscribe(student.user.email, f"equipe-{instance.trigram.lower()}", False, | ||||
|                                              f"{student.user.first_name} {student.user.last_name}") | ||||
|             for coach in instance.coachs.all(): | ||||
|                 get_sympa_client().subscribe(coach.user.email, f"equipe-{instance.trigram.lower()}", False, | ||||
|                                              f"{coach.user.first_name} {coach.user.last_name}") | ||||
|  | ||||
|  | ||||
| def delete_related_videos(instance: Participation, **_): | ||||
|     if instance.solution: | ||||
|         instance.solution.delete() | ||||
|     if instance.synthesis: | ||||
|         instance.synthesis.delete() | ||||
							
								
								
									
										94
									
								
								apps/participation/tables.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								apps/participation/tables.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,94 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.utils import timezone | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| import django_tables2 as tables | ||||
|  | ||||
| from .models import Phase, Team | ||||
|  | ||||
|  | ||||
| class CalendarTable(tables.Table): | ||||
|     class Meta: | ||||
|         attrs = { | ||||
|             'class': 'table table condensed table-striped', | ||||
|         } | ||||
|         row_attrs = { | ||||
|             'class': lambda record: 'bg-success' if timezone.now() > record.end else | ||||
|                                     'bg-warning' if timezone.now() > record.start else | ||||
|                                     'bg-danger', | ||||
|             'data-id': lambda record: str(record.phase_number), | ||||
|         } | ||||
|         model = Phase | ||||
|         fields = ('phase_number', 'description', 'start', 'end',) | ||||
|         template_name = 'django_tables2/bootstrap4.html' | ||||
|         order_by = ('phase_number',) | ||||
|  | ||||
|  | ||||
| # noinspection PyTypeChecker | ||||
| class TeamTable(tables.Table): | ||||
|     name = tables.LinkColumn( | ||||
|         'participation:team_detail', | ||||
|         args=[tables.A("id")], | ||||
|         verbose_name=lambda: _("name").capitalize(), | ||||
|     ) | ||||
|  | ||||
|     problem = tables.Column( | ||||
|         accessor="participation__problem", | ||||
|         verbose_name=lambda: _("problem number").capitalize(), | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         attrs = { | ||||
|             'class': 'table table condensed table-striped', | ||||
|         } | ||||
|         row_attrs = { | ||||
|             'class': lambda record: '' if record.participation.solution.link else 'bg-warning', | ||||
|         } | ||||
|         model = Team | ||||
|         fields = ('name', 'trigram', 'problem',) | ||||
|         template_name = 'django_tables2/bootstrap4.html' | ||||
|  | ||||
|  | ||||
| # noinspection PyTypeChecker | ||||
| class ParticipationTable(tables.Table): | ||||
|     name = tables.LinkColumn( | ||||
|         'participation:participation_detail', | ||||
|         args=[tables.A("id")], | ||||
|         verbose_name=lambda: _("name").capitalize(), | ||||
|         accessor="team__name", | ||||
|     ) | ||||
|  | ||||
|     trigram = tables.Column( | ||||
|         verbose_name=lambda: _("trigram").capitalize(), | ||||
|         accessor="team__trigram", | ||||
|     ) | ||||
|  | ||||
|     problem = tables.Column( | ||||
|         verbose_name=lambda: _("problem number").capitalize(), | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         attrs = { | ||||
|             'class': 'table table condensed table-striped', | ||||
|         } | ||||
|         model = Team | ||||
|         fields = ('name', 'trigram', 'problem',) | ||||
|         template_name = 'django_tables2/bootstrap4.html' | ||||
|  | ||||
|  | ||||
| class VideoTable(tables.Table): | ||||
|     participation_name = tables.LinkColumn( | ||||
|         'participation:participation_detail', | ||||
|         args=[tables.A("participation__pk")], | ||||
|         verbose_name=lambda: _("name").capitalize(), | ||||
|         accessor=tables.A("participation__team__name"), | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         attrs = { | ||||
|             'class': 'table table condensed table-striped', | ||||
|         } | ||||
|         model = Team | ||||
|         fields = ('participation_name', 'link',) | ||||
|         template_name = 'django_tables2/bootstrap4.html' | ||||
							
								
								
									
										39
									
								
								apps/participation/templates/participation/chat.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								apps/participation/templates/participation/chat.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div class="alert alert-warning"> | ||||
|     {% blocktrans trimmed %} | ||||
|         The chat is located on the dedicated Matrix server: | ||||
|     {% endblocktrans %} | ||||
|     </div> | ||||
|  | ||||
|     <div class="alert text-center"> | ||||
|     <a class="btn btn-success" href="https://element.correspondances-maths.fr/#/room/#faq:correspondances-maths.fr" target="_blank"> | ||||
|         <i class="fas fa-server"></i> {% trans "Access to the Matrix server" %} | ||||
|     </a> | ||||
|     </div> | ||||
|  | ||||
|     <div class="alert alert-info"> | ||||
|     <p> | ||||
|     {% blocktrans trimmed %} | ||||
|         To connect to the server, you can select "Log in", then use your credentials of this platform to connect | ||||
|         with the central authentication server, then you must trust the connection between the Matrix account and the | ||||
|         platform. Finally, you will be able to access to the chat platform. | ||||
|     {% endblocktrans %} | ||||
|     </p> | ||||
|  | ||||
|     <p> | ||||
|     {% blocktrans trimmed %} | ||||
|         You will be invited in some basic rooms. You must confirm the invitations to join channels. | ||||
|     {% endblocktrans %} | ||||
|     </p> | ||||
|  | ||||
|     <p> | ||||
|     {% blocktrans trimmed %} | ||||
|         If you have any trouble, don't hesitate to contact us :) | ||||
|     {% endblocktrans %} | ||||
|     </p> | ||||
|     </div> | ||||
| {% endblock %} | ||||
							
								
								
									
										13
									
								
								apps/participation/templates/participation/create_team.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								apps/participation/templates/participation/create_team.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-success" type="submit">{% trans "Create" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
							
								
								
									
										13
									
								
								apps/participation/templates/participation/join_team.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								apps/participation/templates/participation/join_team.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-primary" type="submit">{% trans "Join" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
| @@ -0,0 +1,29 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="fr"> | ||||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <title>Demande de validation - Correspondances des Jeunes Mathématicien·ne·s</title> | ||||
| </head> | ||||
| <body> | ||||
| <p> | ||||
| Bonjour {{ user.registration }}, | ||||
| </p> | ||||
|  | ||||
| <p> | ||||
| L'équipe « {{ team.name }} » ({{ team.trigram }}) vient de demander à valider son équipe pour participer | ||||
| au {{ team.participation.get_problem_display }} des Correspondances des Jeunes Mathématicien·ne·s. | ||||
| Vous pouvez décider d'accepter ou de refuser l'équipe en vous rendant sur la page de l'équipe : | ||||
| <a href="https://{{ domain }}{% url "participation:team_detail" pk=team.pk %}"> | ||||
|     https://{{ domain }}{% url "participation:team_detail" pk=team.pk %} | ||||
| </a> | ||||
| </p> | ||||
|  | ||||
| <p> | ||||
| Cordialement, | ||||
| </p> | ||||
|  | ||||
| <p> | ||||
| L'organisation des Correspondances des Jeunes Mathématicien·ne·s | ||||
| </p> | ||||
| </body> | ||||
| </html> | ||||
| @@ -0,0 +1,10 @@ | ||||
| Bonjour {{ user.registration }}, | ||||
|  | ||||
| L'équipe « {{ team.name }} » ({{ team.trigram }}) vient de demander à valider son équipe pour participer | ||||
| au {{ team.participation.get_problem_display }} des Correspondances des Jeunes Mathématicien·ne·s. | ||||
| Vous pouvez décider d'accepter ou de refuser l'équipe en vous rendant sur la page de l'équipe : | ||||
| https://{{ domain }}{% url "participation:team_detail" pk=team.pk %} | ||||
|  | ||||
| Cordialement, | ||||
|  | ||||
| L'organisation des Correspondances des Jeunes Mathématicien·ne·s | ||||
| @@ -0,0 +1,22 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="fr"> | ||||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <title>Équipe non validée – Correspondances des Jeunes Mathématicien·ne·s</title> | ||||
| </head> | ||||
| <body> | ||||
| Bonjour,<br/> | ||||
| <br /> | ||||
| Maleureusement, votre équipe « {{ team.name }} » ({{ team.trigram }}) n'a pas été validée. Veuillez vérifier que vos autorisations | ||||
| de droit à l'image sont correctes. Les organisateurs vous adressent ce message :<br /> | ||||
| <br /> | ||||
| {{ message }}<br /> | ||||
| <br /> | ||||
| N'hésitez pas à nous contacter à l'adresse <a href="mailto:contact@correspondances-maths.fr">contact@correspondances-maths.fr</a> | ||||
| pour plus d'informations. | ||||
| <br/> | ||||
| Cordialement,<br/> | ||||
| <br/> | ||||
| Le comité d'organisation des Correspondances des Jeunes Mathématicien·ne·s | ||||
| </body> | ||||
| </html> | ||||
| @@ -0,0 +1,12 @@ | ||||
| Bonjour, | ||||
|  | ||||
| Maleureusement, votre équipe « {{ team.name }} » ({{ team.trigram }}) n'a pas été validée. Veuillez vérifier que vos | ||||
| autorisations de droit à l'image sont correctes. Les organisateurs vous adressent ce message : | ||||
|  | ||||
| {{ message }} | ||||
|  | ||||
| N'hésitez pas à nous contacter à l'adresse contact@correspondances-maths.fr pour plus d'informations. | ||||
|  | ||||
| Cordialement, | ||||
|  | ||||
| Le comité d'organisation des Correspondances des Jeunes Mathématicien·ne·s | ||||
| @@ -2,14 +2,16 @@ | ||||
| <html lang="fr"> | ||||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <title>Équipe validée – Correspondances des Jeunes Mathématicien·ne·s {YEAR}</title> | ||||
|     <title>Équipe validée – Correspondances des Jeunes Mathématicien·ne·s</title> | ||||
| </head> | ||||
| <body> | ||||
| Bonjour {FIRST_NAME} {SURNAME},<br/> | ||||
| Bonjour,<br/> | ||||
| <br/> | ||||
| Félicitations ! Votre équipe « {TEAM_NAME} » ({TRIGRAM}) est désormais validée ! Vous êtes désormais apte à travailler sur | ||||
| votre problème. Lorsque les Correspondances auront débutées, vous pourrez soumettre votre vidéo sur la plateforme d'inscription. | ||||
| {MESSAGE}<br /> | ||||
| Félicitations ! Votre équipe « {{ team.name }} » ({{ team.trigram }}) est désormais validée ! Vous êtes désormais apte à travailler sur | ||||
| votre problème. Lorsque les Correspondances auront débutées, vous pourrez soumettre votre vidéo sur la plateforme d'inscription.<br> | ||||
| Les organisateurs vous adressent ce message :<br/> | ||||
| <br/> | ||||
| {{ message }}<br /> | ||||
| <br/> | ||||
| Cordialement,<br/> | ||||
| <br/> | ||||
| @@ -0,0 +1,12 @@ | ||||
| Bonjour, | ||||
|  | ||||
| Félicitations ! Votre équipe « {{ team.name }} » ({{ team.trigram }}) est désormais validée ! Vous êtes désormais apte à travailler sur | ||||
| votre problème. Lorsque les Correspondances auront débutées, vous pourrez soumettre votre vidéo sur la plateforme d'inscription. | ||||
|  | ||||
| Les organisateurs vous adressent ce message : | ||||
|  | ||||
| {{ message }} | ||||
|  | ||||
| Cordialement, | ||||
|  | ||||
| Le comité d'organisation des Correspondances des Jeunes Mathématicien·ne·s | ||||
| @@ -0,0 +1,303 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
| {% trans "any" as any %} | ||||
|     <div class="card bg-light shadow"> | ||||
|         <div class="card-header text-center"> | ||||
|             <h4>{% trans "Participation of team" %} {{ participation.team.name }} ({{ participation.team.trigram }})</h4> | ||||
|         </div> | ||||
|         <div class="card-body"> | ||||
|             <dl class="row"> | ||||
|                 <dt class="col-sm-2">{% trans "Team:" %}</dt> | ||||
|                 <dd class="col-sm-10"><a href="{% url "participation:team_detail" pk=participation.team.pk %}">{{ participation.team }}</a></dd> | ||||
|  | ||||
|                 <dt class="col-sm-2">{% trans "Chosen problem:" %}</dt> | ||||
|                 <dd class="col-sm-10">{{ participation.get_problem_display }}</dd> | ||||
|             </dl> | ||||
|  | ||||
|             <div id="solution-container"> | ||||
|                 <dl class="row"> | ||||
|                     {% trans "No video sent" as novideo %} | ||||
|                     <dt class="col-sm-2">{% trans "Proposed solution:" %}</dt> | ||||
|                     <dd class="col-sm-10"><a href="{{ participation.solution.link|default:"#" }}"{% if participation.solution.link %} target="_blank"{% endif %}> | ||||
|                         {{ participation.solution.link|default:novideo }}</a> | ||||
|                         {% if current_phase.phase_number == 1 or participation.solution.link == "" %} | ||||
|                             <button class="btn btn-primary" data-toggle="modal" data-target="#uploadSolutionModal">{% trans "Upload" %}</button> | ||||
|                         {% endif %} | ||||
|                         {% if participation.solution.link %} | ||||
|                             <button class="btn btn-info" data-toggle="modal" data-target="#displaySolutionModal">{% trans "Display" %}</button> | ||||
|                         {% endif %} | ||||
|                     </dd> | ||||
|                 </dl> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
|  | ||||
|     {% if user.registration.is_admin or current_phase.phase_number >= 2 %} | ||||
|     <hr> | ||||
|  | ||||
|     <div class="row"> | ||||
|         <div class="col-md-6"> | ||||
|             <div class="card bg-light shadow"> | ||||
|                 <div class="card-header text-center"> | ||||
|                     <h4>{% trans "Sent solution" %}</h4> | ||||
|                 </div> | ||||
|                 <div class="card-body"> | ||||
|                     <dl class="row"> | ||||
|                         <dt class="col-xl-5 text-right">{% trans "Team that received your solution:" %}</dt> | ||||
|                         <dd class="col-md-5">{{ participation.sent_participation.team|default:any }}</dd> | ||||
|                         {% if user.registration.is_admin %} | ||||
|                             <dd class="col-xs-2"> | ||||
|                                 <button class="btn btn-primary" data-toggle="modal" data-target="#defineSentParticipationModal">{% trans "Change" %}</button> | ||||
|                             </dd> | ||||
|                         {% endif %} | ||||
|                     </dl> | ||||
|  | ||||
|                     {% if current_phase.phase_number == 2 %} | ||||
|                         <div class="alert alert-info"> | ||||
|                             {% blocktrans trimmed %} | ||||
|                                 The mentioned team received your video. They are now watching your video, | ||||
|                                 and formulating questions. You would be able to exchange with the other phase during | ||||
|                                 the next phase. | ||||
|                             {% endblocktrans %} | ||||
|                         </div> | ||||
|                     {% elif current_phase.phase_number == 3 %} | ||||
|                         <div class="alert alert-info"> | ||||
|                             {% blocktrans trimmed with user_id=user.pk %} | ||||
|                                 The other team sent you questions about your solution. Your are now able to answer them, | ||||
|                                 then to exchange freely with the other team. You can click on the Chat button, or to | ||||
|                                 connect to your dedicated Matrix account: | ||||
|                                 <code>@corres2math_{{ user_id }}:correspondances-maths.fr</code>. | ||||
|                                 You can use your own Matrix client, or use the dedicated Element client: | ||||
|                                 <a href="https://element.correspondances-maths.fr">element.correpondances-maths.fr</a> | ||||
|                             {% endblocktrans %} | ||||
|                         </div> | ||||
|                     {% elif current_phase.phase_number == 4 %} | ||||
|                         <dl class="row"> | ||||
|                             <dt class="col-xl-5 text-right">{% trans "Synthesis from the other team:" %}</dt> | ||||
|                             <dd class="col-sm-7"><a href="{{ participation.received_participation.synthesis.link|default:"#" }}"{% if participation.received_participation.synthesis.link %} target="_blank"{% endif %}> | ||||
|                                 {{ participation.received_participation.synthesis.link|default:novideo }}</a> | ||||
|                                 {% if participation.received_participation.synthesis.link %} | ||||
|                                     <button class="btn btn-info" data-toggle="modal" data-target="#displayOtherSynthesisModal">{% trans "Display" %}</button> | ||||
|                                 {% endif %} | ||||
|                             </dd> | ||||
|                         </dl> | ||||
|                     {% endif %} | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|         <div class="col-md-6"> | ||||
|             <div class="card bg-light shadow"> | ||||
|                 <div class="card-header text-center"> | ||||
|                     <h4>{% trans "Received solution" %}</h4> | ||||
|                 </div> | ||||
|                 <div class="card-body"> | ||||
|                     <dl class="row"> | ||||
|                         <dt class="col-xl-5 text-right">{% trans "Team that sent you their solution:" %}</dt> | ||||
|                         <dd class="col-md-5">{{ participation.received_participation.team|default:any }}</dd> | ||||
|                         {% if user.registration.is_admin %} | ||||
|                             <dd class="col-xs-2"> | ||||
|                                 <button class="btn btn-primary" data-toggle="modal" data-target="#defineReceivedParticipationModal">{% trans "Change" %}</button> | ||||
|                             </dd> | ||||
|                         {% endif %} | ||||
|  | ||||
|                         <dt class="col-xl-5 text-right">{% trans "Proposed solution:" %}</dt> | ||||
|                         <dd class="col-sm-7"><a href="{{ participation.received_participation.solution.link|default:"#" }}"{% if participation.received_participation.solution.link %} target="_blank"{% endif %}> | ||||
|                             {{ participation.received_participation.solution.link|default:novideo }}</a> | ||||
|                             {% if participation.received_participation.solution.link %} | ||||
|                                 <button class="btn btn-info" data-toggle="modal" data-target="#displayOtherSolutionModal">{% trans "Display" %}</button> | ||||
|                             {% endif %} | ||||
|                         </dd> | ||||
|  | ||||
|                     {% if current_phase.phase_number == 2 %} | ||||
|                         <div class="alert alert-info"> | ||||
|                         {% blocktrans trimmed %} | ||||
|                             You received a solution about the same problem that you treated from another team. | ||||
|                             You are now encouraged to see the video, then to ask from 3 to 6 questions about the video. | ||||
|                             After that, you will be invited to exchange with the other team about the solution. | ||||
|                         {% endblocktrans %} | ||||
|                         </div> | ||||
|  | ||||
|                         {% for question in participation.questions.all %} | ||||
|                             <dd class="col-md-9 text-truncate">{{ question.question }}</dd> | ||||
|                             <dd class="col-md-3"> | ||||
|                                 <button class="btn btn-primary" data-toggle="modal" data-target="#updateQuestion{{ forloop.counter }}Modal">{% trans "Change" %}</button> | ||||
|                             </dd> | ||||
|                             <hr> | ||||
|                         {% endfor %} | ||||
|  | ||||
|                         {% if user.registration.participates %} | ||||
|                             <button class="btn btn-success" data-toggle="modal" data-target="#addQuestionModal"> | ||||
|                                 <i class="fas fa-plus-circle"></i> {% trans "Add a question" %} | ||||
|                             </button> | ||||
|                         {% endif %} | ||||
|                     {% elif current_phase.phase_number == 3 %} | ||||
|                         <div class="alert alert-info"> | ||||
|                             {% blocktrans trimmed with user_id=user.pk %} | ||||
|                                 You sent your questions to the other team about their solution. When they answer to | ||||
|                                 your questions, you will be able to exchange freely with the other team. | ||||
|                                 You can click on the Chat button, or to connect to your dedicated Matrix account: | ||||
|                                 <code>@corres2math_{{ user_id }}:correspondances-maths.fr</code>. | ||||
|                                 You can use your own Matrix client, or use the dedicated Element client: | ||||
|                                 <a href="https://element.correspondances-maths.fr">element.correpondances-maths.fr</a> | ||||
|                             {% endblocktrans %} | ||||
|                         </div> | ||||
|                     {% elif current_phase.phase_number == 4 %} | ||||
|                         <div id="solution-container"> | ||||
|                             <dl class="row"> | ||||
|                                 {% trans "No video sent" as novideo %} | ||||
|                                 <dt class="col-sm-5 text-right">{% trans "Your synthesis of the exchange:" %}</dt> | ||||
|                                 <dd class="col-sm-7"><a href="{{ participation.synthesis.link|default:"#" }}"{% if participation.synthesis.link %} target="_blank"{% endif %}> | ||||
|                                     {{ participation.synthesis.link|default:novideo }}</a> | ||||
|                                     <button class="btn btn-primary" data-toggle="modal" data-target="#uploadSynthesisModal">{% trans "Upload" %}</button> | ||||
|                                     {% if participation.synthesis.link %} | ||||
|                                         <button class="btn btn-info" data-toggle="modal" data-target="#displaySynthesisModal">{% trans "Display" %}</button> | ||||
|                                     {% endif %} | ||||
|                                 </dd> | ||||
|                             </dl> | ||||
|                         </div> | ||||
|                     {% endif %} | ||||
|                     </dl> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
|     {% endif %} | ||||
|  | ||||
| {% if user.registration.is_admin %} | ||||
|     {% trans "Define received video" as modal_title %} | ||||
|     {% trans "Update" as modal_button %} | ||||
|     {% url "participation:participation_receive_participation" pk=participation.pk as modal_action %} | ||||
|     {% include "base_modal.html" with modal_id="defineReceivedParticipation" %} | ||||
|  | ||||
|     {% trans "Define team that receives your video" as modal_title %} | ||||
|     {% trans "Update" as modal_button %} | ||||
|     {% url "participation:participation_send_participation" pk=participation.pk as modal_action %} | ||||
|     {% include "base_modal.html" with modal_id="defineSentParticipation" %} | ||||
| {% endif %} | ||||
|  | ||||
| {% trans "Upload video" as modal_title %} | ||||
| {% trans "Upload" as modal_button %} | ||||
| {% url "participation:upload_video" pk=participation.solution_id as modal_action %} | ||||
| {% include "base_modal.html" with modal_id="uploadSolution" %} | ||||
|  | ||||
| {% trans "Display solution" as modal_title %} | ||||
| {% trans "This video platform is not supported yet." as unsupported_platform %} | ||||
| {% include "base_modal.html" with modal_id="displaySolution" modal_action="" modal_button="" modal_additional_class="modal-lg" modal_content=participation.solution.as_iframe|default:unsupported_platform %} | ||||
|  | ||||
|  | ||||
| {% if user.registration.is_admin or current_phase.phase_number >= 2 %} | ||||
|     {% if participation.received_participation.solution.link %} | ||||
|         {% trans "Display solution" as modal_title %} | ||||
|         {% trans "This video platform is not supported yet." as unsupported_platform %} | ||||
|         {% include "base_modal.html" with modal_id="displayOtherSolution" modal_action="" modal_button="" modal_additional_class="modal-lg" modal_content=participation.received_participation.solution.as_iframe|default:unsupported_platform %} | ||||
|     {% endif %} | ||||
|  | ||||
|     {% if current_phase.phase_number == 2 %} | ||||
|         {% trans "Add question" as modal_title %} | ||||
|         {% trans "Add" as modal_button %} | ||||
|         {% url "participation:add_question" pk=participation.pk as modal_action %} | ||||
|         {% include "base_modal.html" with modal_id="addQuestion" modal_button_type="success" %} | ||||
|     {% endif %} | ||||
|  | ||||
|     {% for question in participation.questions.all %} | ||||
|         {% with number_str=forloop.counter|stringformat:"d"%} | ||||
|             {% with modal_id="updateQuestion"|add:number_str %} | ||||
|                 {% trans "Delete" as delete %} | ||||
|                 {% with extra_modal_button='<button class="btn btn-danger" type="button" data-dismiss="modal" data-toggle="modal" data-target="#deleteQuestion'|add:number_str|add:'Modal">'|add:delete|add:"</button>"|safe %} | ||||
|                     {% trans "Update question" as modal_title %} | ||||
|                     {% trans "Update" as modal_button %} | ||||
|                     {% url "participation:update_question" pk=question.pk as modal_action %} | ||||
|                     {% include "base_modal.html" %} | ||||
|                 {% endwith %} | ||||
|             {% endwith %} | ||||
|  | ||||
|             {% with modal_id="deleteQuestion"|add:number_str %} | ||||
|                 {% trans "Delete question" as modal_title %} | ||||
|                 {% trans "Delete" as modal_button %} | ||||
|                 {% url "participation:delete_question" pk=question.pk as modal_action %} | ||||
|                 {% include "base_modal.html" with modal_button_type="danger" %} | ||||
|             {% endwith %} | ||||
|         {% endwith %} | ||||
|     {% endfor %} | ||||
| {% endif %} | ||||
|  | ||||
| {% if current_phase.phase_number >= 4 %} | ||||
|     {% if participation.received_participation.synthesis.link %} | ||||
|         {% trans "Display synthesis" as modal_title %} | ||||
|         {% trans "This video platform is not supported yet." as unsupported_platform %} | ||||
|         {% include "base_modal.html" with modal_id="displayOtherSynthesis" modal_action="" modal_button="" modal_additional_class="modal-lg" modal_content=participation.received_participation.synthesis.as_iframe|default:unsupported_platform %} | ||||
|     {% endif %} | ||||
|  | ||||
|     {% trans "Upload video" as modal_title %} | ||||
|     {% trans "Upload" as modal_button %} | ||||
|     {% url "participation:upload_video" pk=participation.synthesis_id as modal_action %} | ||||
|     {% include "base_modal.html" with modal_id="uploadSynthesis" %} | ||||
|  | ||||
|     {% if participation.synthesis.link %} | ||||
|         {% trans "Display synthesis" as modal_title %} | ||||
|         {% trans "This video platform is not supported yet." as unsupported_platform %} | ||||
|         {% include "base_modal.html" with modal_id="displaySynthesis" modal_action="" modal_button="" modal_additional_class="modal-lg" modal_content=participation.synthesis.as_iframe|default:unsupported_platform %} | ||||
|     {% endif %} | ||||
| {% endif %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block extrajavascript %} | ||||
|     <script> | ||||
|         $(document).ready(function() { | ||||
|             {% if user.registration.is_admin %} | ||||
|                 $('button[data-target="#defineReceivedParticipationModal"]').click(function() { | ||||
|                     let modalBody = $("#defineReceivedParticipationModal div.modal-body"); | ||||
|                     if (!modalBody.html().trim()) | ||||
|                         modalBody.load("{% url "participation:participation_receive_participation" pk=participation.pk %} #form-content"); | ||||
|                 }); | ||||
|  | ||||
|                 $('button[data-target="#defineSentParticipationModal"]').click(function() { | ||||
|                     let modalBody = $("#defineSentParticipationModal div.modal-body"); | ||||
|                     if (!modalBody.html().trim()) | ||||
|                         modalBody.load("{% url "participation:participation_send_participation" pk=participation.pk %} #form-content"); | ||||
|                 }); | ||||
|             {% endif %} | ||||
|  | ||||
|             {% if current_phase.phase_number == 2 %} | ||||
|                 $('button[data-target="#addQuestionModal"]').click(function() { | ||||
|                     let modalBody = $("#addQuestionModal div.modal-body"); | ||||
|                     if (!modalBody.html().trim()) | ||||
|                         modalBody.load("{% url "participation:add_question" pk=participation.pk %} #form-content"); | ||||
|                 }); | ||||
|             {% endif %} | ||||
|  | ||||
|             {% for question in participation.questions.all %} | ||||
|                 $('button[data-target="#updateQuestion{{ forloop.counter }}Modal"]').click(function() { | ||||
|                     let modalBody = $("#updateQuestion{{ forloop.counter }}Modal div.modal-body"); | ||||
|                     if (!modalBody.html().trim()) | ||||
|                         modalBody.load("{% url "participation:update_question" pk=question.pk %} #form-content"); | ||||
|                 }); | ||||
|  | ||||
|                 {% if current_phase.phase_number == 2 %} | ||||
|                     $('button[data-target="#deleteQuestion{{ forloop.counter }}Modal"]').click(function() { | ||||
|                         let modalBody = $("#deleteQuestion{{ forloop.counter }}Modal div.modal-body"); | ||||
|                         if (!modalBody.html().trim()) | ||||
|                             modalBody.load("{% url "participation:delete_question" pk=question.pk %} #form-content"); | ||||
|                     }); | ||||
|                 {% endif %} | ||||
|             {% endfor %} | ||||
|  | ||||
|             $('button[data-target="#uploadSolutionModal"]').click(function() { | ||||
|                 let modalBody = $("#uploadSolutionModal div.modal-body"); | ||||
|                 if (!modalBody.html().trim()) | ||||
|                     modalBody.load("{% url "participation:upload_video" pk=participation.solution_id %} #form-content"); | ||||
|             }); | ||||
|  | ||||
|             {% if current_phase.phase_number == 4 %} | ||||
|                 $('button[data-target="#uploadSynthesisModal"]').click(function() { | ||||
|                     let modalBody = $("#uploadSynthesisModal div.modal-body"); | ||||
|                     if (!modalBody.html().trim()) | ||||
|                         modalBody.load("{% url "participation:upload_video" pk=participation.synthesis_id %} #form-content"); | ||||
|                 }); | ||||
|             {% endif %} | ||||
|         }); | ||||
|     </script> | ||||
| {% endblock %} | ||||
							
								
								
									
										14
									
								
								apps/participation/templates/participation/phase_form.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								apps/participation/templates/participation/phase_form.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post" action="{% url "participation:update_phase" pk=object.pk %}"> | ||||
|         <div id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-primary" type="submit">{% trans "Update" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
|  | ||||
							
								
								
									
										35
									
								
								apps/participation/templates/participation/phase_list.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								apps/participation/templates/participation/phase_list.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load django_tables2 i18n static %} | ||||
|  | ||||
| {% block extracss %} | ||||
|     <link rel="stylesheet" href="{% static "bootstrap_datepicker_plus/css/datepicker-widget.css" %}"> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block contenttitle %} | ||||
|     <h2>{% trans "Calendar" %}</h2> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="form-content"> | ||||
|         {% render_table table %} | ||||
|  | ||||
|         {% trans "Update phase" as modal_title %} | ||||
|         {% trans "Update" as modal_button %} | ||||
|         {% include "base_modal.html" with modal_id="updatePhase" %} | ||||
|     </div> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block extrajavascript %} | ||||
| {% if user.registration.is_admin %} | ||||
|     <script> | ||||
|         $("tr").click(function () { | ||||
|             let modalBody = $("#updatePhaseModal div.modal-body"); | ||||
|             if (!modalBody.html().trim()) | ||||
|                 modalBody.load("{% url "participation:calendar" %}" + $(this).data("id") + "/ #form-content"); | ||||
|             $("#updatePhase-form").attr("action", "{% url "participation:calendar" %}" + $(this).data("id") + "/") | ||||
|             $("#updatePhaseModal").modal(); | ||||
|         }) | ||||
|     </script> | ||||
| {% endif %} | ||||
| {% endblock %} | ||||
| @@ -0,0 +1,17 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div id="form-content"> | ||||
|             <div class="alert alert-danger"> | ||||
|                 {% trans "Are you sure you want to delete this question?" %} | ||||
|             </div> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-danger" type="submit">{% trans "Delete" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
|  | ||||
| @@ -0,0 +1,14 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-primary" type="submit">{% trans "Send" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
|  | ||||
| @@ -0,0 +1,14 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-primary" type="submit">{% trans "Update" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
|  | ||||
| @@ -0,0 +1,14 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-primary" type="submit">{% trans "Update" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
|  | ||||
							
								
								
									
										148
									
								
								apps/participation/templates/participation/team_detail.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										148
									
								
								apps/participation/templates/participation/team_detail.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,148 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load i18n %} | ||||
| {% load crispy_forms_filters %} | ||||
|  | ||||
| {% block content %} | ||||
|  | ||||
|     <div class="card bg-light shadow"> | ||||
|         <div class="card-header text-center"> | ||||
|             <h4>{{ team.name }}</h4> | ||||
|         </div> | ||||
|     <div class="card-body"> | ||||
|         <dl class="row"> | ||||
|             <dt class="col-sm-6 text-right">{% trans "Name:" %}</dt> | ||||
|             <dd class="col-sm-6">{{ team.name }}</dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Trigram:" %}</dt> | ||||
|             <dd class="col-sm-6">{{ team.trigram }}</dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Email:" %}</dt> | ||||
|             <dd class="col-sm-6"><a href="mailto:{{ team.email }}">{{ team.email }}</a></dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Access code:" %}</dt> | ||||
|             <dd class="col-sm-6">{{ team.access_code }}</dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Coachs:" %}</dt> | ||||
|             <dd class="col-sm-6"> | ||||
|                 {% for coach in team.coachs.all %} | ||||
|                     <a href="{% url "registration:user_detail" pk=coach.user.pk %}">{{ coach }}</a>{% if not forloop.last %},{% endif %} | ||||
|                 {% empty %} | ||||
|                     {% trans "any" %} | ||||
|                 {% endfor %} | ||||
|             </dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Participants:" %}</dt> | ||||
|             <dd class="col-sm-6"> | ||||
|                 {% for student in team.students.all %} | ||||
|                     <a href="{% url "registration:user_detail" pk=student.user.pk %}">{{ student }}</a>{% if not forloop.last %},{% endif %} | ||||
|                 {% empty %} | ||||
|                     {% trans "any" %} | ||||
|                 {% endfor %} | ||||
|             </dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Chosen problem:" %}</dt> | ||||
|             {% trans "any" as any %} | ||||
|             <dd class="col-sm-6">{{ team.participation.get_problem_display|default:any }}</dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Grant Animath to publish our video:" %}</dt> | ||||
|             <dd class="col-sm-6">{{ team.grant_animath_access_videos|yesno }}</dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Authorizations:" %}</dt> | ||||
|             <dd class="col-sm-6"> | ||||
|                 {% for student in team.students.all %} | ||||
|                     {% if student.photo_authorization %} | ||||
|                         <a href="{{ student.photo_authorization.url }}" data-turbolinks="false">{{ student }}</a>{% if not forloop.last %},{% endif %} | ||||
|                     {% else %} | ||||
|                         {{ student }} ({% trans "Not uploaded yet" %}){% if not forloop.last %},{% endif %} | ||||
|                     {% endif %} | ||||
|                 {% endfor %} | ||||
|             </dd> | ||||
|         </dl> | ||||
|     </div> | ||||
|     <div class="card-footer text-center"> | ||||
|         <button class="btn btn-primary" data-toggle="modal" data-target="#updateTeamModal">{% trans "Update" %}</button> | ||||
|         {% if not team.participation.valid %} | ||||
|             <button class="btn btn-danger" data-toggle="modal" data-target="#leaveTeamModal">{% trans "Leave" %}</button> | ||||
|         {% endif %} | ||||
|     </div> | ||||
|     </div> | ||||
|  | ||||
|     <hr> | ||||
|  | ||||
|     {% if team.participation.valid %} | ||||
|         <div class="text-center"> | ||||
|             <a class="btn btn-info" href="{% url "participation:participation_detail" pk=team.participation.pk %}"> | ||||
|                 <i class="fas fa-video"></i> {% trans "Access to team participation" %} <i class="fas fa-video"></i> | ||||
|             </a> | ||||
|         </div> | ||||
|     {% elif team.participation.valid == None %} {# Team did not ask for validation #} | ||||
|         {% if user.registration.participates %} | ||||
|             {% if can_validate %} | ||||
|                 <div class="alert alert-info"> | ||||
|                     {% trans "Your team has at least 3 members and all photo authorizations were given: the team can be validated." %} | ||||
|                     <div class="text-center"> | ||||
|                         <form method="post"> | ||||
|                             {% csrf_token %} | ||||
|                             {{ request_validation_form|crispy }} | ||||
|                             <button class="btn btn-success" name="request-validation">{% trans "Submit my team to validation" %}</button> | ||||
|                         </form> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             {% else %} | ||||
|                 <div class="alert alert-warning"> | ||||
|                     {% trans "Your team must be composed of 3 members and each member must upload its photo authorization and confirm its email address." %} | ||||
|                 </div> | ||||
|             {% endif %} | ||||
|         {% else %} | ||||
|             <div class="alert alert-warning"> | ||||
|             {% trans "This team didn't ask for validation yet." %} | ||||
|             </div> | ||||
|         {% endif %} | ||||
|     {% else %}  {# Team is waiting for validation #} | ||||
|         {% if user.registration.participates %} | ||||
|             <div class="alert alert-warning"> | ||||
|                 {% trans "Your validation is pending." %} | ||||
|             </div> | ||||
|         {% else %} | ||||
|             <div class="alert alert-info"> | ||||
|                 {% trans "The team requested to be validated. You may now control the authorizations and confirm that they can participate." %} | ||||
|             </div> | ||||
|              <form method="post"> | ||||
|                 {% csrf_token %} | ||||
|                 {{ validation_form|crispy }} | ||||
|                 <div class="input-group btn-group"> | ||||
|                     <button class="btn btn-success" name="validate" type="submit">{% trans "Validate" %}</button> | ||||
|                     <button class="btn btn-danger" name="invalidate" type="submit">{% trans "Invalidate" %}</button> | ||||
|                 </div> | ||||
|              </form> | ||||
|         {% endif %} | ||||
|     {% endif %} | ||||
|  | ||||
|     {% trans "Update team" as modal_title %} | ||||
|     {% trans "Update" as modal_button %} | ||||
|     {% url "participation:update_team" pk=team.pk as modal_action %} | ||||
|     {% include "base_modal.html" with modal_id="updateTeam" %} | ||||
|  | ||||
|     {% trans "Leave team" as modal_title %} | ||||
|     {% trans "Leave" as modal_button %} | ||||
|     {% url "participation:team_leave" as modal_action %} | ||||
|     {% include "base_modal.html" with modal_id="leaveTeam" modal_button_type="danger" %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block extrajavascript %} | ||||
|     <script> | ||||
|         $(document).ready(function() { | ||||
|             $('button[data-target="#updateTeamModal"]').click(function() { | ||||
|                 let modalBody = $("#updateTeamModal div.modal-body"); | ||||
|                 if (!modalBody.html().trim()) | ||||
|                     modalBody.load("{% url "participation:update_team" pk=team.pk %} #form-content"); | ||||
|             }); | ||||
|             $('button[data-target="#leaveTeamModal"]').click(function() { | ||||
|                 let modalBody = $("#leaveTeamModal div.modal-body"); | ||||
|                 if (!modalBody.html().trim()) | ||||
|                     modalBody.load("{% url "participation:team_leave" %} #form-content"); | ||||
|             }); | ||||
|         }); | ||||
|     </script> | ||||
| {% endblock %} | ||||
							
								
								
									
										13
									
								
								apps/participation/templates/participation/team_leave.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								apps/participation/templates/participation/team_leave.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div class="alert alert-warning" id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {% trans "Are you sure that you want to leave this team?" %} | ||||
|         </div> | ||||
|         <button class="btn btn-danger" type="submit">{% trans "Leave" %}</button> | ||||
|     </form> | ||||
| {% endblock %} | ||||
							
								
								
									
										13
									
								
								apps/participation/templates/participation/team_list.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								apps/participation/templates/participation/team_list.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load django_tables2 i18n %} | ||||
|  | ||||
| {% block contenttitle %} | ||||
|     <h1>{% trans "All teams" %}</h1> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <div id="form-content"> | ||||
|         {% render_table table %} | ||||
|     </div> | ||||
| {% endblock %} | ||||
							
								
								
									
										15
									
								
								apps/participation/templates/participation/update_team.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								apps/participation/templates/participation/update_team.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|             {{ participation_form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-success" type="submit">{% trans "Update" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
|  | ||||
							
								
								
									
										14
									
								
								apps/participation/templates/participation/upload_video.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								apps/participation/templates/participation/upload_video.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-success" type="submit">{% trans "Upload" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
|  | ||||
| @@ -0,0 +1,8 @@ | ||||
| <div style="position: relative; width: 100%; padding-bottom: 56.25%;"> | ||||
|     <iframe src="https://www.youtube.com/embed/{{ youtube_code }}" | ||||
|             frameborder="0" | ||||
|             style="position: absolute; top: 0; left: 0; width: 100%; height: 100%;" | ||||
|             allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture" | ||||
|             allowfullscreen> | ||||
|     </iframe> | ||||
| </div> | ||||
| @@ -0,0 +1,4 @@ | ||||
| {{ object.team.name }} | ||||
| {{ object.team.trigram }} | ||||
| {{ object.problem }} | ||||
| {{ object.get_problem_display }} | ||||
| @@ -0,0 +1,2 @@ | ||||
| {{ object.name }} | ||||
| {{ object.trigram }} | ||||
| @@ -0,0 +1,5 @@ | ||||
| {{ object.link }} | ||||
| {{ object.participation.team.name }} | ||||
| {{ object.participation.team.trigram }} | ||||
| {{ object.participation.problem }} | ||||
| {{ object.participation.get_problem_display }} | ||||
							
								
								
									
										2
									
								
								apps/participation/templatetags/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								apps/participation/templatetags/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
							
								
								
									
										15
									
								
								apps/participation/templatetags/calendar.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								apps/participation/templatetags/calendar.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django import template | ||||
|  | ||||
| from ..models import Phase | ||||
|  | ||||
|  | ||||
| def current_phase(nb): | ||||
|     phase = Phase.current_phase() | ||||
|     return phase is not None and phase.phase_number == nb | ||||
|  | ||||
|  | ||||
| register = template.Library() | ||||
| register.filter("current_phase", current_phase) | ||||
							
								
								
									
										853
									
								
								apps/participation/tests.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										853
									
								
								apps/participation/tests.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,853 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from datetime import timedelta | ||||
|  | ||||
| from django.contrib.auth.models import User | ||||
| from django.contrib.contenttypes.models import ContentType | ||||
| from django.contrib.sites.models import Site | ||||
| from django.core.management import call_command | ||||
| from django.test import TestCase | ||||
| from django.urls import reverse | ||||
| from django.utils import timezone | ||||
| from registration.models import CoachRegistration, StudentRegistration | ||||
|  | ||||
| from .models import Participation, Phase, Question, Team | ||||
|  | ||||
|  | ||||
| class TestStudentParticipation(TestCase): | ||||
|     def setUp(self) -> None: | ||||
|         self.superuser = User.objects.create_superuser( | ||||
|             username="admin", | ||||
|             email="admin@example.com", | ||||
|             password="toto1234", | ||||
|         ) | ||||
|  | ||||
|         self.user = User.objects.create( | ||||
|             first_name="Toto", | ||||
|             last_name="Toto", | ||||
|             email="toto@example.com", | ||||
|             password="toto", | ||||
|         ) | ||||
|         StudentRegistration.objects.create( | ||||
|             user=self.user, | ||||
|             student_class=12, | ||||
|             school="Earth", | ||||
|             give_contact_to_animath=True, | ||||
|             email_confirmed=True, | ||||
|         ) | ||||
|         self.team = Team.objects.create( | ||||
|             name="Super team", | ||||
|             trigram="AAA", | ||||
|             access_code="azerty", | ||||
|             grant_animath_access_videos=True, | ||||
|         ) | ||||
|         self.question = Question.objects.create(participation=self.team.participation, | ||||
|                                                 question="Pourquoi l'existence précède l'essence ?") | ||||
|         self.client.force_login(self.user) | ||||
|  | ||||
|         self.second_user = User.objects.create( | ||||
|             first_name="Lalala", | ||||
|             last_name="Lalala", | ||||
|             email="lalala@example.com", | ||||
|             password="lalala", | ||||
|         ) | ||||
|         StudentRegistration.objects.create( | ||||
|             user=self.second_user, | ||||
|             student_class=11, | ||||
|             school="Moon", | ||||
|             give_contact_to_animath=True, | ||||
|             email_confirmed=True, | ||||
|         ) | ||||
|         self.second_team = Team.objects.create( | ||||
|             name="Poor team", | ||||
|             trigram="FFF", | ||||
|             access_code="qwerty", | ||||
|             grant_animath_access_videos=True, | ||||
|         ) | ||||
|  | ||||
|         self.coach = User.objects.create( | ||||
|             first_name="Coach", | ||||
|             last_name="Coach", | ||||
|             email="coach@example.com", | ||||
|             password="coach", | ||||
|         ) | ||||
|         CoachRegistration.objects.create(user=self.coach) | ||||
|  | ||||
|     def test_admin_pages(self): | ||||
|         """ | ||||
|         Load Django-admin pages. | ||||
|         """ | ||||
|         self.client.force_login(self.superuser) | ||||
|  | ||||
|         # Test team pages | ||||
|         response = self.client.get(reverse("admin:index") + "participation/team/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.get(reverse("admin:index") | ||||
|                                    + f"participation/team/{self.team.pk}/change/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|         response = self.client.get(reverse("admin:index") + | ||||
|                                    f"r/{ContentType.objects.get_for_model(Team).id}/" | ||||
|                                    f"{self.team.pk}/") | ||||
|         self.assertRedirects(response, "http://" + Site.objects.get().domain + | ||||
|                              str(self.team.get_absolute_url()), 302, 200) | ||||
|  | ||||
|         # Test participation pages | ||||
|         self.team.participation.valid = True | ||||
|         self.team.participation.save() | ||||
|         response = self.client.get(reverse("admin:index") + "participation/participation/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.get(reverse("admin:index") | ||||
|                                    + f"participation/participation/{self.team.participation.pk}/change/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|         response = self.client.get(reverse("admin:index") + | ||||
|                                    f"r/{ContentType.objects.get_for_model(Participation).id}/" | ||||
|                                    f"{self.team.participation.pk}/") | ||||
|         self.assertRedirects(response, "http://" + Site.objects.get().domain + | ||||
|                              str(self.team.participation.get_absolute_url()), 302, 200) | ||||
|  | ||||
|         # Test video pages | ||||
|         response = self.client.get(reverse("admin:index") + "participation/video/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.get(reverse("admin:index") | ||||
|                                    + f"participation/video/{self.team.participation.solution.pk}/change/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # Test question pages | ||||
|         response = self.client.get(reverse("admin:index") + "participation/question/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.get(reverse("admin:index") | ||||
|                                    + f"participation/question/{self.question.pk}/change/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # Test phase pages | ||||
|         response = self.client.get(reverse("admin:index") + "participation/phase/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.get(reverse("admin:index") + "participation/phase/1/change/") | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|     def test_create_team(self): | ||||
|         """ | ||||
|         Try to create a team. | ||||
|         """ | ||||
|         response = self.client.get(reverse("participation:create_team")) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:create_team"), data=dict( | ||||
|             name="Test team", | ||||
|             trigram="123", | ||||
|             grant_animath_access_videos=False, | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:create_team"), data=dict( | ||||
|             name="Test team", | ||||
|             trigram="TES", | ||||
|             grant_animath_access_videos=False, | ||||
|         )) | ||||
|         self.assertTrue(Team.objects.filter(trigram="TES").exists()) | ||||
|         team = Team.objects.get(trigram="TES") | ||||
|         self.assertRedirects(response, reverse("participation:team_detail", args=(team.pk,)), 302, 200) | ||||
|  | ||||
|         # Already in a team | ||||
|         response = self.client.post(reverse("participation:create_team"), data=dict( | ||||
|             name="Test team 2", | ||||
|             trigram="TET", | ||||
|             grant_animath_access_videos=False, | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|     def test_join_team(self): | ||||
|         """ | ||||
|         Try to join an existing team. | ||||
|         """ | ||||
|         response = self.client.get(reverse("participation:join_team")) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         team = Team.objects.create(name="Test", trigram="TES") | ||||
|  | ||||
|         response = self.client.post(reverse("participation:join_team"), data=dict( | ||||
|             access_code="éééééé", | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:join_team"), data=dict( | ||||
|             access_code=team.access_code, | ||||
|         )) | ||||
|         self.assertRedirects(response, reverse("participation:team_detail", args=(team.pk,)), 302, 200) | ||||
|         self.assertTrue(Team.objects.filter(trigram="TES").exists()) | ||||
|  | ||||
|         # Already joined | ||||
|         response = self.client.post(reverse("participation:join_team"), data=dict( | ||||
|             access_code=team.access_code, | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|     def test_team_list(self): | ||||
|         """ | ||||
|         Test to display the list of teams. | ||||
|         """ | ||||
|         response = self.client.get(reverse("participation:team_list")) | ||||
|         self.assertTrue(response.status_code, 200) | ||||
|  | ||||
|     def test_no_myteam_redirect_noteam(self): | ||||
|         """ | ||||
|         Test redirection. | ||||
|         """ | ||||
|         response = self.client.get(reverse("participation:my_team_detail")) | ||||
|         self.assertTrue(response.status_code, 200) | ||||
|  | ||||
|     def test_team_detail(self): | ||||
|         """ | ||||
|         Try to display the information of a team. | ||||
|         """ | ||||
|         self.user.registration.team = self.team | ||||
|         self.user.registration.save() | ||||
|  | ||||
|         response = self.client.get(reverse("participation:my_team_detail")) | ||||
|         self.assertRedirects(response, reverse("participation:team_detail", args=(self.team.pk,)), 302, 200) | ||||
|  | ||||
|         response = self.client.get(reverse("participation:team_detail", args=(self.team.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # Can't see other teams | ||||
|         self.second_user.registration.team = self.second_team | ||||
|         self.second_user.registration.save() | ||||
|         self.client.force_login(self.second_user) | ||||
|         response = self.client.get(reverse("participation:team_detail", args=(self.team.participation.pk,))) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|     def test_request_validate_team(self): | ||||
|         """ | ||||
|         The team ask for validation. | ||||
|         """ | ||||
|         self.user.registration.team = self.team | ||||
|         self.user.registration.save() | ||||
|  | ||||
|         second_user = User.objects.create( | ||||
|             first_name="Blublu", | ||||
|             last_name="Blublu", | ||||
|             email="blublu@example.com", | ||||
|             password="blublu", | ||||
|         ) | ||||
|         StudentRegistration.objects.create( | ||||
|             user=second_user, | ||||
|             student_class=12, | ||||
|             school="Jupiter", | ||||
|             give_contact_to_animath=True, | ||||
|             email_confirmed=True, | ||||
|             team=self.team, | ||||
|             photo_authorization="authorization/photo/mai-linh", | ||||
|         ) | ||||
|  | ||||
|         third_user = User.objects.create( | ||||
|             first_name="Zupzup", | ||||
|             last_name="Zupzup", | ||||
|             email="zupzup@example.com", | ||||
|             password="zupzup", | ||||
|         ) | ||||
|         StudentRegistration.objects.create( | ||||
|             user=third_user, | ||||
|             student_class=10, | ||||
|             school="Sun", | ||||
|             give_contact_to_animath=False, | ||||
|             email_confirmed=True, | ||||
|             team=self.team, | ||||
|             photo_authorization="authorization/photo/yohann", | ||||
|         ) | ||||
|  | ||||
|         self.client.force_login(self.superuser) | ||||
|         # Admin users can't ask for validation | ||||
|         resp = self.client.post(reverse("participation:team_detail", args=(self.team.pk,)), data=dict( | ||||
|             _form_type="RequestValidationForm", | ||||
|             engagement=True, | ||||
|         )) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|  | ||||
|         self.client.force_login(self.user) | ||||
|  | ||||
|         self.assertIsNone(self.team.participation.valid) | ||||
|  | ||||
|         resp = self.client.get(reverse("participation:team_detail", args=(self.team.pk,))) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|         self.assertFalse(resp.context["can_validate"]) | ||||
|         # Can't validate | ||||
|         resp = self.client.post(reverse("participation:team_detail", args=(self.team.pk,)), data=dict( | ||||
|             _form_type="RequestValidationForm", | ||||
|             engagement=True, | ||||
|         )) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|  | ||||
|         self.user.registration.photo_authorization = "authorization/photo/ananas" | ||||
|         self.user.registration.save() | ||||
|  | ||||
|         resp = self.client.get(reverse("participation:team_detail", args=(self.team.pk,))) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|         self.assertFalse(resp.context["can_validate"]) | ||||
|  | ||||
|         self.team.participation.problem = 2 | ||||
|         self.team.participation.save() | ||||
|  | ||||
|         resp = self.client.get(reverse("participation:team_detail", args=(self.team.pk,))) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|         self.assertTrue(resp.context["can_validate"]) | ||||
|  | ||||
|         resp = self.client.post(reverse("participation:team_detail", args=(self.team.pk,)), data=dict( | ||||
|             _form_type="RequestValidationForm", | ||||
|             engagement=True, | ||||
|         )) | ||||
|         self.assertRedirects(resp, reverse("participation:team_detail", args=(self.team.pk,)), 302, 200) | ||||
|         self.team.participation.refresh_from_db() | ||||
|         self.assertFalse(self.team.participation.valid) | ||||
|         self.assertIsNotNone(self.team.participation.valid) | ||||
|  | ||||
|         # Team already asked for validation | ||||
|         resp = self.client.post(reverse("participation:team_detail", args=(self.team.pk,)), data=dict( | ||||
|             _form_type="RequestValidationForm", | ||||
|             engagement=True, | ||||
|         )) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|  | ||||
|     def test_validate_team(self): | ||||
|         """ | ||||
|         A team asked for validation. Try to validate it. | ||||
|         """ | ||||
|         self.team.participation.valid = False | ||||
|         self.team.participation.save() | ||||
|  | ||||
|         # No right to do that | ||||
|         resp = self.client.post(reverse("participation:team_detail", args=(self.team.pk,)), data=dict( | ||||
|             _form_type="ValidateParticipationForm", | ||||
|             message="J'ai 4 ans", | ||||
|             validate=True, | ||||
|         )) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|  | ||||
|         self.client.force_login(self.superuser) | ||||
|  | ||||
|         resp = self.client.get(reverse("participation:team_detail", args=(self.team.pk,))) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|  | ||||
|         resp = self.client.post(reverse("participation:team_detail", args=(self.team.pk,)), data=dict( | ||||
|             _form_type="ValidateParticipationForm", | ||||
|             message="Woops I didn't said anything", | ||||
|         )) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|  | ||||
|         # Test invalidate team | ||||
|         resp = self.client.post(reverse("participation:team_detail", args=(self.team.pk,)), data=dict( | ||||
|             _form_type="ValidateParticipationForm", | ||||
|             message="Wsh nope", | ||||
|             invalidate=True, | ||||
|         )) | ||||
|         self.assertRedirects(resp, reverse("participation:team_detail", args=(self.team.pk,)), 302, 200) | ||||
|         self.team.participation.refresh_from_db() | ||||
|         self.assertIsNone(self.team.participation.valid) | ||||
|  | ||||
|         # Team did not ask validation | ||||
|         resp = self.client.post(reverse("participation:team_detail", args=(self.team.pk,)), data=dict( | ||||
|             _form_type="ValidateParticipationForm", | ||||
|             message="Bienvenue ça va être trop cool", | ||||
|             validate=True, | ||||
|         )) | ||||
|         self.assertEqual(resp.status_code, 200) | ||||
|  | ||||
|         self.team.participation.valid = False | ||||
|         self.team.participation.save() | ||||
|  | ||||
|         # Test validate team | ||||
|         resp = self.client.post(reverse("participation:team_detail", args=(self.team.pk,)), data=dict( | ||||
|             _form_type="ValidateParticipationForm", | ||||
|             message="Bienvenue ça va être trop cool", | ||||
|             validate=True, | ||||
|         )) | ||||
|         self.assertRedirects(resp, reverse("participation:team_detail", args=(self.team.pk,)), 302, 200) | ||||
|         self.team.participation.refresh_from_db() | ||||
|         self.assertTrue(self.team.participation.valid) | ||||
|  | ||||
|     def test_update_team(self): | ||||
|         """ | ||||
|         Try to update team information. | ||||
|         """ | ||||
|         self.user.registration.team = self.team | ||||
|         self.user.registration.save() | ||||
|  | ||||
|         self.coach.registration.team = self.team | ||||
|         self.coach.registration.save() | ||||
|  | ||||
|         response = self.client.get(reverse("participation:update_team", args=(self.team.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # Form is invalid | ||||
|         response = self.client.post(reverse("participation:update_team", args=(self.team.pk,)), data=dict( | ||||
|             name="Updated team name", | ||||
|             trigram="BBB", | ||||
|             grant_animath_access_videos=True, | ||||
|             problem=42, | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:update_team", args=(self.team.pk,)), data=dict( | ||||
|             name="Updated team name", | ||||
|             trigram="BBB", | ||||
|             grant_animath_access_videos=True, | ||||
|             problem=3, | ||||
|         )) | ||||
|         self.assertRedirects(response, reverse("participation:team_detail", args=(self.team.pk,)), 302, 200) | ||||
|         self.assertTrue(Team.objects.filter(trigram="BBB", participation__problem=3).exists()) | ||||
|  | ||||
|     def test_leave_team(self): | ||||
|         """ | ||||
|         A user is in a team, and leaves it. | ||||
|         """ | ||||
|         # User is not in a team | ||||
|         response = self.client.post(reverse("participation:team_leave")) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|         self.user.registration.team = self.team | ||||
|         self.user.registration.save() | ||||
|  | ||||
|         # Team is valid | ||||
|         self.team.participation.valid = True | ||||
|         self.team.participation.save() | ||||
|         response = self.client.post(reverse("participation:team_leave")) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|         # Unauthenticated users are redirected to login page | ||||
|         self.client.logout() | ||||
|         response = self.client.get(reverse("participation:team_leave")) | ||||
|         self.assertRedirects(response, reverse("login") + "?next=" + reverse("participation:team_leave"), 302, 200) | ||||
|  | ||||
|         self.client.force_login(self.user) | ||||
|  | ||||
|         self.team.participation.valid = None | ||||
|         self.team.participation.save() | ||||
|  | ||||
|         response = self.client.post(reverse("participation:team_leave")) | ||||
|         self.assertRedirects(response, reverse("index"), 302, 200) | ||||
|         self.user.registration.refresh_from_db() | ||||
|         self.assertIsNone(self.user.registration.team) | ||||
|         self.assertFalse(Team.objects.filter(pk=self.team.pk).exists()) | ||||
|  | ||||
|     def test_no_myparticipation_redirect_nomyparticipation(self): | ||||
|         """ | ||||
|         Ensure a permission denied when we search my team participation when we are in no team. | ||||
|         """ | ||||
|         response = self.client.get(reverse("participation:my_participation_detail")) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|     def test_participation_detail(self): | ||||
|         """ | ||||
|         Try to display the detail of a team participation. | ||||
|         """ | ||||
|         self.user.registration.team = self.team | ||||
|         self.user.registration.save() | ||||
|  | ||||
|         # Can't see the participation if it is not valid | ||||
|         response = self.client.get(reverse("participation:my_participation_detail")) | ||||
|         self.assertRedirects(response, | ||||
|                              reverse("participation:participation_detail", args=(self.team.participation.pk,)), | ||||
|                              302, 403) | ||||
|  | ||||
|         self.team.participation.valid = True | ||||
|         self.team.participation.save() | ||||
|         response = self.client.get(reverse("participation:my_participation_detail")) | ||||
|         self.assertRedirects(response, | ||||
|                              reverse("participation:participation_detail", args=(self.team.participation.pk,)), | ||||
|                              302, 200) | ||||
|  | ||||
|         response = self.client.get(reverse("participation:participation_detail", args=(self.team.participation.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # Can't see other participations | ||||
|         self.second_user.registration.team = self.second_team | ||||
|         self.second_user.registration.save() | ||||
|         self.client.force_login(self.second_user) | ||||
|         response = self.client.get(reverse("participation:participation_detail", args=(self.team.participation.pk,))) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|     def test_upload_video(self): | ||||
|         """ | ||||
|         Try to send a solution video link. | ||||
|         """ | ||||
|         self.user.registration.team = self.team | ||||
|         self.user.registration.save() | ||||
|  | ||||
|         self.team.participation.valid = True | ||||
|         self.team.participation.save() | ||||
|  | ||||
|         response = self.client.get(reverse("participation:upload_video", args=(self.team.participation.solution.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:upload_video", args=(self.team.participation.solution.pk,)), | ||||
|                                     data=dict(link="https://youtube.com/watch?v=73nsrixx7eI")) | ||||
|         self.assertRedirects(response, | ||||
|                              reverse("participation:participation_detail", args=(self.team.participation.id,)), | ||||
|                              302, 200) | ||||
|         self.team.participation.refresh_from_db() | ||||
|         self.assertEqual(self.team.participation.solution.platform, "youtube") | ||||
|         self.assertEqual(self.team.participation.solution.youtube_code, "73nsrixx7eI") | ||||
|  | ||||
|         response = self.client.get(reverse("participation:participation_detail", args=(self.team.participation.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # Set the second phase | ||||
|         for i in range(1, 5): | ||||
|             Phase.objects.filter(phase_number=i).update(start=timezone.now() + timedelta(days=i - 2), | ||||
|                                                         end=timezone.now() + timedelta(days=i - 1)) | ||||
|         self.assertEqual(Phase.current_phase().phase_number, 2) | ||||
|  | ||||
|         # Can't update the link during the second phase | ||||
|         response = self.client.post(reverse("participation:upload_video", args=(self.team.participation.solution.pk,)), | ||||
|                                     data=dict(link="https://youtube.com/watch?v=73nsrixx7eI")) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|     def test_questions(self): | ||||
|         """ | ||||
|         Ensure that creating/updating/deleting a question is working. | ||||
|         """ | ||||
|         self.user.registration.team = self.team | ||||
|         self.user.registration.save() | ||||
|  | ||||
|         self.team.participation.valid = True | ||||
|         self.team.participation.save() | ||||
|  | ||||
|         response = self.client.get(reverse("participation:add_question", args=(self.team.participation.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # We are not in second phase | ||||
|         response = self.client.post(reverse("participation:add_question", args=(self.team.participation.pk,)), | ||||
|                                     data=dict(question="I got censored!")) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # Set the second phase | ||||
|         for i in range(1, 5): | ||||
|             Phase.objects.filter(phase_number=i).update(start=timezone.now() + timedelta(days=i - 2), | ||||
|                                                         end=timezone.now() + timedelta(days=i - 1)) | ||||
|         self.assertEqual(Phase.current_phase().phase_number, 2) | ||||
|  | ||||
|         # Create a question | ||||
|         response = self.client.post(reverse("participation:add_question", args=(self.team.participation.pk,)), | ||||
|                                     data=dict(question="I asked a question!")) | ||||
|         self.assertRedirects(response, reverse("participation:participation_detail", | ||||
|                                                args=(self.team.participation.pk,)), 302, 200) | ||||
|         qs = Question.objects.filter(participation=self.team.participation, question="I asked a question!") | ||||
|         self.assertTrue(qs.exists()) | ||||
|         question = qs.get() | ||||
|  | ||||
|         # Update a question | ||||
|         response = self.client.get(reverse("participation:update_question", args=(question.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|         response = self.client.post(reverse("participation:update_question", args=(question.pk,)), data=dict( | ||||
|             question="The question changed!", | ||||
|         )) | ||||
|         self.assertRedirects(response, reverse("participation:participation_detail", | ||||
|                                                args=(self.team.participation.pk,)), 302, 200) | ||||
|         question.refresh_from_db() | ||||
|         self.assertEqual(question.question, "The question changed!") | ||||
|  | ||||
|         # Delete the question | ||||
|         response = self.client.get(reverse("participation:delete_question", args=(question.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|         response = self.client.post(reverse("participation:delete_question", args=(question.pk,))) | ||||
|         self.assertRedirects(response, reverse("participation:participation_detail", | ||||
|                                                args=(self.team.participation.pk,)), 302, 200) | ||||
|         self.assertFalse(Question.objects.filter(pk=question.pk).exists()) | ||||
|  | ||||
|         # Non-authenticated users are redirected to login page | ||||
|         self.client.logout() | ||||
|         response = self.client.get(reverse("participation:add_question", args=(self.team.participation.pk,))) | ||||
|         self.assertRedirects(response, reverse("login") + "?next=" + | ||||
|                              reverse("participation:add_question", args=(self.team.participation.pk,)), 302, 200) | ||||
|         response = self.client.get(reverse("participation:update_question", args=(self.question.pk,))) | ||||
|         self.assertRedirects(response, reverse("login") + "?next=" + | ||||
|                              reverse("participation:update_question", args=(self.question.pk,)), 302, 200) | ||||
|         response = self.client.get(reverse("participation:delete_question", args=(self.question.pk,))) | ||||
|         self.assertRedirects(response, reverse("login") + "?next=" + | ||||
|                              reverse("participation:delete_question", args=(self.question.pk,)), 302, 200) | ||||
|  | ||||
|     def test_current_phase(self): | ||||
|         """ | ||||
|         Ensure that the current phase is the good one. | ||||
|         """ | ||||
|         # We are before the beginning | ||||
|         for i in range(1, 5): | ||||
|             Phase.objects.filter(phase_number=i).update(start=timezone.now() + timedelta(days=2 * i), | ||||
|                                                         end=timezone.now() + timedelta(days=2 * i + 1)) | ||||
|         self.assertEqual(Phase.current_phase(), None) | ||||
|  | ||||
|         # We are after the end | ||||
|         for i in range(1, 5): | ||||
|             Phase.objects.filter(phase_number=i).update(start=timezone.now() - timedelta(days=2 * i), | ||||
|                                                         end=timezone.now() - timedelta(days=2 * i + 1)) | ||||
|         self.assertEqual(Phase.current_phase().phase_number, Phase.objects.count()) | ||||
|  | ||||
|         # First phase | ||||
|         for i in range(1, 5): | ||||
|             Phase.objects.filter(phase_number=i).update(start=timezone.now() + timedelta(days=i - 1), | ||||
|                                                         end=timezone.now() + timedelta(days=i)) | ||||
|         self.assertEqual(Phase.current_phase().phase_number, 1) | ||||
|  | ||||
|         # Second phase | ||||
|         for i in range(1, 5): | ||||
|             Phase.objects.filter(phase_number=i).update(start=timezone.now() + timedelta(days=i - 2), | ||||
|                                                         end=timezone.now() + timedelta(days=i - 1)) | ||||
|         self.assertEqual(Phase.current_phase().phase_number, 2) | ||||
|  | ||||
|         # Third phase | ||||
|         for i in range(1, 5): | ||||
|             Phase.objects.filter(phase_number=i).update(start=timezone.now() + timedelta(days=i - 3), | ||||
|                                                         end=timezone.now() + timedelta(days=i - 2)) | ||||
|         self.assertEqual(Phase.current_phase().phase_number, 3) | ||||
|  | ||||
|         # Fourth phase | ||||
|         for i in range(1, 5): | ||||
|             Phase.objects.filter(phase_number=i).update(start=timezone.now() + timedelta(days=i - 4), | ||||
|                                                         end=timezone.now() + timedelta(days=i - 3)) | ||||
|         self.assertEqual(Phase.current_phase().phase_number, 4) | ||||
|  | ||||
|         response = self.client.get(reverse("participation:calendar")) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.get(reverse("participation:update_phase", args=(4,))) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:update_phase", args=(4,)), data=dict( | ||||
|             start=timezone.now(), | ||||
|             end=timezone.now() + timedelta(days=3), | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|         self.client.force_login(self.superuser) | ||||
|         response = self.client.get(reverse("participation:update_phase", args=(4,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:update_phase", args=(4,)), data=dict( | ||||
|             start=timezone.now(), | ||||
|             end=timezone.now() + timedelta(days=3), | ||||
|         )) | ||||
|         self.assertRedirects(response, reverse("participation:calendar"), 302, 200) | ||||
|         fourth_phase = Phase.objects.get(phase_number=4) | ||||
|         self.assertEqual((fourth_phase.end - fourth_phase.start).days, 3) | ||||
|  | ||||
|         # First phase must be before the other phases | ||||
|         response = self.client.post(reverse("participation:update_phase", args=(1,)), data=dict( | ||||
|             start=timezone.now() + timedelta(days=8), | ||||
|             end=timezone.now() + timedelta(days=9), | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # Fourth phase must be after the other phases | ||||
|         response = self.client.post(reverse("participation:update_phase", args=(4,)), data=dict( | ||||
|             start=timezone.now() - timedelta(days=9), | ||||
|             end=timezone.now() - timedelta(days=8), | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # End must be after start | ||||
|         response = self.client.post(reverse("participation:update_phase", args=(4,)), data=dict( | ||||
|             start=timezone.now() + timedelta(days=3), | ||||
|             end=timezone.now(), | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         # Unauthenticated user can't update the calendar | ||||
|         self.client.logout() | ||||
|         response = self.client.get(reverse("participation:calendar")) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|         response = self.client.get(reverse("participation:update_phase", args=(2,))) | ||||
|         self.assertRedirects(response, reverse("login") + "?next=" + | ||||
|                              reverse("participation:update_phase", args=(2,)), 302, 200) | ||||
|  | ||||
|     def test_forbidden_access(self): | ||||
|         """ | ||||
|         Load personal pages and ensure that these are protected. | ||||
|         """ | ||||
|         self.user.registration.team = self.team | ||||
|         self.user.registration.save() | ||||
|  | ||||
|         resp = self.client.get(reverse("participation:team_detail", args=(self.second_team.pk,))) | ||||
|         self.assertEqual(resp.status_code, 403) | ||||
|         resp = self.client.get(reverse("participation:update_team", args=(self.second_team.pk,))) | ||||
|         self.assertEqual(resp.status_code, 403) | ||||
|         resp = self.client.get(reverse("participation:team_authorizations", args=(self.second_team.pk,))) | ||||
|         self.assertEqual(resp.status_code, 403) | ||||
|         resp = self.client.get(reverse("participation:participation_detail", args=(self.second_team.pk,))) | ||||
|         self.assertEqual(resp.status_code, 403) | ||||
|         resp = self.client.get(reverse("participation:upload_video", | ||||
|                                        args=(self.second_team.participation.solution.pk,))) | ||||
|         self.assertEqual(resp.status_code, 403) | ||||
|         resp = self.client.get(reverse("participation:upload_video", | ||||
|                                        args=(self.second_team.participation.synthesis.pk,))) | ||||
|         self.assertEqual(resp.status_code, 403) | ||||
|         resp = self.client.get(reverse("participation:add_question", args=(self.second_team.pk,))) | ||||
|         self.assertEqual(resp.status_code, 403) | ||||
|         question = Question.objects.create(participation=self.second_team.participation, | ||||
|                                            question=self.question.question) | ||||
|         resp = self.client.get(reverse("participation:update_question", args=(question.pk,))) | ||||
|         self.assertEqual(resp.status_code, 403) | ||||
|         resp = self.client.get(reverse("participation:delete_question", args=(question.pk,))) | ||||
|         self.assertEqual(resp.status_code, 403) | ||||
|  | ||||
|     def test_cover_matrix(self): | ||||
|         """ | ||||
|         Load matrix scripts, to cover them and ensure that they can run. | ||||
|         """ | ||||
|         self.user.registration.team = self.team | ||||
|         self.user.registration.save() | ||||
|         self.second_user.registration.team = self.second_team | ||||
|         self.second_user.registration.save() | ||||
|         self.team.participation.valid = True | ||||
|         self.team.participation.received_participation = self.second_team.participation | ||||
|         self.team.participation.save() | ||||
|  | ||||
|         call_command('fix_matrix_channels') | ||||
|         call_command('setup_third_phase') | ||||
|  | ||||
|  | ||||
| class TestAdmin(TestCase): | ||||
|     def setUp(self) -> None: | ||||
|         self.user = User.objects.create_superuser( | ||||
|             username="admin@example.com", | ||||
|             email="admin@example.com", | ||||
|             password="admin", | ||||
|         ) | ||||
|         self.client.force_login(self.user) | ||||
|  | ||||
|         self.team1 = Team.objects.create( | ||||
|             name="Toto", | ||||
|             trigram="TOT", | ||||
|         ) | ||||
|         self.team1.participation.valid = True | ||||
|         self.team1.participation.problem = 1 | ||||
|         self.team1.participation.save() | ||||
|  | ||||
|         self.team2 = Team.objects.create( | ||||
|             name="Bliblu", | ||||
|             trigram="BIU", | ||||
|         ) | ||||
|         self.team2.participation.valid = True | ||||
|         self.team2.participation.problem = 1 | ||||
|         self.team2.participation.save() | ||||
|  | ||||
|         self.team3 = Team.objects.create( | ||||
|             name="Zouplop", | ||||
|             trigram="ZPL", | ||||
|         ) | ||||
|         self.team3.participation.valid = True | ||||
|         self.team3.participation.problem = 1 | ||||
|         self.team3.participation.save() | ||||
|  | ||||
|         self.other_team = Team.objects.create( | ||||
|             name="I am different", | ||||
|             trigram="IAD", | ||||
|         ) | ||||
|         self.other_team.participation.valid = True | ||||
|         self.other_team.participation.problem = 2 | ||||
|         self.other_team.participation.save() | ||||
|  | ||||
|     def test_research(self): | ||||
|         """ | ||||
|         Try to search some things. | ||||
|         """ | ||||
|         call_command("rebuild_index", "--noinput", "--verbosity", 0) | ||||
|  | ||||
|         response = self.client.get(reverse("haystack_search") + "?q=" + self.team1.name) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|         self.assertTrue(response.context["object_list"]) | ||||
|  | ||||
|         response = self.client.get(reverse("haystack_search") + "?q=" + self.team2.trigram) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|         self.assertTrue(response.context["object_list"]) | ||||
|  | ||||
|     def test_set_received_video(self): | ||||
|         """ | ||||
|         Try to define the received video of a participation. | ||||
|         """ | ||||
|         response = self.client.get(reverse("participation:participation_receive_participation", | ||||
|                                            args=(self.team1.participation.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:participation_receive_participation", | ||||
|                                             args=(self.team1.participation.pk,)), | ||||
|                                     data=dict(received_participation=self.team2.participation.pk)) | ||||
|         self.assertRedirects(response, reverse("participation:participation_detail", | ||||
|                                                args=(self.team1.participation.pk,)), 302, 200) | ||||
|  | ||||
|         response = self.client.get(reverse("participation:participation_receive_participation", | ||||
|                                            args=(self.team1.participation.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:participation_send_participation", | ||||
|                                             args=(self.team1.participation.pk,)), | ||||
|                                     data=dict(sent_participation=self.team3.participation.pk)) | ||||
|         self.assertRedirects(response, reverse("participation:participation_detail", | ||||
|                                                args=(self.team1.participation.pk,)), 302, 200) | ||||
|  | ||||
|         self.team1.participation.refresh_from_db() | ||||
|         self.team2.participation.refresh_from_db() | ||||
|         self.team3.participation.refresh_from_db() | ||||
|  | ||||
|         self.assertEqual(self.team1.participation.received_participation.pk, self.team2.participation.pk) | ||||
|         self.assertEqual(self.team1.participation.sent_participation.pk, self.team3.participation.pk) | ||||
|         self.assertEqual(self.team2.participation.sent_participation.pk, self.team1.participation.pk) | ||||
|         self.assertEqual(self.team3.participation.received_participation.pk, self.team1.participation.pk) | ||||
|  | ||||
|         # The other team didn't work on the same problem | ||||
|         response = self.client.post(reverse("participation:participation_receive_participation", | ||||
|                                             args=(self.team1.participation.pk,)), | ||||
|                                     data=dict(received_participation=self.other_team.participation.pk)) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         response = self.client.post(reverse("participation:participation_send_participation", | ||||
|                                             args=(self.team1.participation.pk,)), | ||||
|                                     data=dict(sent_participation=self.other_team.participation.pk)) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|     def test_create_team_forbidden(self): | ||||
|         """ | ||||
|         Ensure that an admin can't create a team. | ||||
|         """ | ||||
|         response = self.client.post(reverse("participation:create_team"), data=dict( | ||||
|             name="Test team", | ||||
|             trigram="TES", | ||||
|             grant_animath_access_videos=False, | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|     def test_join_team_forbidden(self): | ||||
|         """ | ||||
|         Ensure that an admin can't join a team. | ||||
|         """ | ||||
|         team = Team.objects.create(name="Test", trigram="TES") | ||||
|  | ||||
|         response = self.client.post(reverse("participation:join_team"), data=dict( | ||||
|             access_code=team.access_code, | ||||
|         )) | ||||
|         self.assertTrue(response.status_code, 403) | ||||
|  | ||||
|     def test_leave_team_forbidden(self): | ||||
|         """ | ||||
|         Ensure that an admin can't leave a team. | ||||
|         """ | ||||
|         response = self.client.get(reverse("participation:team_leave")) | ||||
|         self.assertTrue(response.status_code, 403) | ||||
|  | ||||
|     def test_my_team_forbidden(self): | ||||
|         """ | ||||
|         Ensure that an admin can't access to "My team". | ||||
|         """ | ||||
|         response = self.client.get(reverse("participation:my_team_detail")) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
|  | ||||
|     def test_my_participation_forbidden(self): | ||||
|         """ | ||||
|         Ensure that an admin can't access to "My participation". | ||||
|         """ | ||||
|         response = self.client.get(reverse("participation:my_participation_detail")) | ||||
|         self.assertEqual(response.status_code, 403) | ||||
							
								
								
									
										37
									
								
								apps/participation/urls.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								apps/participation/urls.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.urls import path | ||||
| from django.views.generic import TemplateView | ||||
|  | ||||
| from .views import CalendarView, CreateQuestionView, CreateTeamView, DeleteQuestionView, JoinTeamView, \ | ||||
|     MyParticipationDetailView, MyTeamDetailView, ParticipationDetailView, PhaseUpdateView, \ | ||||
|     SetParticipationReceiveParticipationView, SetParticipationSendParticipationView, TeamAuthorizationsView, \ | ||||
|     TeamDetailView, TeamLeaveView, TeamListView, TeamUpdateView, UpdateQuestionView, UploadVideoView | ||||
|  | ||||
|  | ||||
| app_name = "participation" | ||||
|  | ||||
| urlpatterns = [ | ||||
|     path("create_team/", CreateTeamView.as_view(), name="create_team"), | ||||
|     path("join_team/", JoinTeamView.as_view(), name="join_team"), | ||||
|     path("teams/", TeamListView.as_view(), name="team_list"), | ||||
|     path("team/", MyTeamDetailView.as_view(), name="my_team_detail"), | ||||
|     path("team/<int:pk>/", TeamDetailView.as_view(), name="team_detail"), | ||||
|     path("team/<int:pk>/update/", TeamUpdateView.as_view(), name="update_team"), | ||||
|     path("team/<int:pk>/authorizations/", TeamAuthorizationsView.as_view(), name="team_authorizations"), | ||||
|     path("team/leave/", TeamLeaveView.as_view(), name="team_leave"), | ||||
|     path("detail/", MyParticipationDetailView.as_view(), name="my_participation_detail"), | ||||
|     path("detail/<int:pk>/", ParticipationDetailView.as_view(), name="participation_detail"), | ||||
|     path("detail/upload-video/<int:pk>/", UploadVideoView.as_view(), name="upload_video"), | ||||
|     path("detail/<int:pk>/receive-participation/", SetParticipationReceiveParticipationView.as_view(), | ||||
|          name="participation_receive_participation"), | ||||
|     path("detail/<int:pk>/send-participation/", SetParticipationSendParticipationView.as_view(), | ||||
|          name="participation_send_participation"), | ||||
|     path("detail/<int:pk>/add-question/", CreateQuestionView.as_view(), name="add_question"), | ||||
|     path("update-question/<int:pk>/", UpdateQuestionView.as_view(), name="update_question"), | ||||
|     path("delete-question/<int:pk>/", DeleteQuestionView.as_view(), name="delete_question"), | ||||
|     path("calendar/", CalendarView.as_view(), name="calendar"), | ||||
|     path("calendar/<int:pk>/", PhaseUpdateView.as_view(), name="update_phase"), | ||||
|     path("chat/", TemplateView.as_view(template_name="participation/chat.html"), name="chat") | ||||
| ] | ||||
							
								
								
									
										557
									
								
								apps/participation/views.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										557
									
								
								apps/participation/views.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,557 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from io import BytesIO | ||||
| from zipfile import ZipFile | ||||
|  | ||||
| from corres2math.lists import get_sympa_client | ||||
| from corres2math.matrix import Matrix | ||||
| from corres2math.views import AdminMixin | ||||
| from django.contrib.auth.mixins import LoginRequiredMixin | ||||
| from django.contrib.sites.models import Site | ||||
| from django.core.exceptions import PermissionDenied | ||||
| from django.core.mail import send_mail | ||||
| from django.db import transaction | ||||
| from django.http import HttpResponse | ||||
| from django.shortcuts import redirect | ||||
| from django.template.loader import render_to_string | ||||
| from django.urls import reverse_lazy | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from django.views.generic import CreateView, DeleteView, DetailView, FormView, RedirectView, TemplateView, UpdateView | ||||
| from django.views.generic.edit import FormMixin, ProcessFormView | ||||
| from django_tables2 import SingleTableView | ||||
| from magic import Magic | ||||
| from registration.models import AdminRegistration | ||||
|  | ||||
| from .forms import JoinTeamForm, ParticipationForm, PhaseForm, QuestionForm, \ | ||||
|     ReceiveParticipationForm, RequestValidationForm, SendParticipationForm, TeamForm, \ | ||||
|     UploadVideoForm, ValidateParticipationForm | ||||
| from .models import Participation, Phase, Question, Team, Video | ||||
| from .tables import CalendarTable, TeamTable | ||||
|  | ||||
|  | ||||
| class CreateTeamView(LoginRequiredMixin, CreateView): | ||||
|     """ | ||||
|     Display the page to create a team for new users. | ||||
|     """ | ||||
|  | ||||
|     model = Team | ||||
|     form_class = TeamForm | ||||
|     extra_context = dict(title=_("Create team")) | ||||
|     template_name = "participation/create_team.html" | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         user = request.user | ||||
|         if not user.is_authenticated: | ||||
|             return super().handle_no_permission() | ||||
|         registration = user.registration | ||||
|         if not registration.participates: | ||||
|             raise PermissionDenied(_("You don't participate, so you can't create a team.")) | ||||
|         elif registration.team: | ||||
|             raise PermissionDenied(_("You are already in a team.")) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     @transaction.atomic | ||||
|     def form_valid(self, form): | ||||
|         """ | ||||
|         When a team is about to be created, the user automatically | ||||
|         joins the team, a mailing list got created and the user is | ||||
|         automatically subscribed to this mailing list, and finally | ||||
|         a Matrix room is created and the user is invited in this room. | ||||
|         """ | ||||
|         ret = super().form_valid(form) | ||||
|         # The user joins the team | ||||
|         user = self.request.user | ||||
|         registration = user.registration | ||||
|         registration.team = form.instance | ||||
|         registration.save() | ||||
|  | ||||
|         # Subscribe the user mail address to the team mailing list | ||||
|         get_sympa_client().subscribe(user.email, f"equipe-{form.instance.trigram.lower()}", False, | ||||
|                                      f"{user.first_name} {user.last_name}") | ||||
|  | ||||
|         # Invite the user in the team Matrix room | ||||
|         Matrix.invite(f"#equipe-{form.instance.trigram.lower()}:correspondances-maths.fr", | ||||
|                       f"@{user.registration.matrix_username}:correspondances-maths.fr") | ||||
|         return ret | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:team_detail", args=(self.object.pk,)) | ||||
|  | ||||
|  | ||||
| class JoinTeamView(LoginRequiredMixin, FormView): | ||||
|     """ | ||||
|     Participants can join a team with the access code of the team. | ||||
|     """ | ||||
|     model = Team | ||||
|     form_class = JoinTeamForm | ||||
|     extra_context = dict(title=_("Join team")) | ||||
|     template_name = "participation/create_team.html" | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         user = request.user | ||||
|         if not user.is_authenticated: | ||||
|             return super().handle_no_permission() | ||||
|         registration = user.registration | ||||
|         if not registration.participates: | ||||
|             raise PermissionDenied(_("You don't participate, so you can't create a team.")) | ||||
|         elif registration.team: | ||||
|             raise PermissionDenied(_("You are already in a team.")) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     @transaction.atomic | ||||
|     def form_valid(self, form): | ||||
|         """ | ||||
|         When a user joins a team, the user is automatically subscribed to | ||||
|         the team mailing list,the user is invited in the team Matrix room. | ||||
|         """ | ||||
|         self.object = form.instance | ||||
|         ret = super().form_valid(form) | ||||
|  | ||||
|         # Join the team | ||||
|         user = self.request.user | ||||
|         registration = user.registration | ||||
|         registration.team = form.instance | ||||
|         registration.save() | ||||
|  | ||||
|         # Subscribe to the team mailing list | ||||
|         get_sympa_client().subscribe(user.email, f"equipe-{form.instance.trigram.lower()}", False, | ||||
|                                      f"{user.first_name} {user.last_name}") | ||||
|  | ||||
|         # Invite the user in the team Matrix room | ||||
|         Matrix.invite(f"#equipe-{form.instance.trigram.lower()}:correspondances-maths.fr", | ||||
|                       f"@{user.registration.matrix_username}:correspondances-maths.fr") | ||||
|         return ret | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:team_detail", args=(self.object.pk,)) | ||||
|  | ||||
|  | ||||
| class TeamListView(AdminMixin, SingleTableView): | ||||
|     """ | ||||
|     Display the whole list of teams | ||||
|     """ | ||||
|     model = Team | ||||
|     table_class = TeamTable | ||||
|     ordering = ('participation__problem', 'trigram',) | ||||
|  | ||||
|  | ||||
| class MyTeamDetailView(LoginRequiredMixin, RedirectView): | ||||
|     """ | ||||
|     Redirect to the detail of the team in which the user is. | ||||
|     """ | ||||
|  | ||||
|     def get_redirect_url(self, *args, **kwargs): | ||||
|         user = self.request.user | ||||
|         registration = user.registration | ||||
|         if registration.participates: | ||||
|             if registration.team: | ||||
|                 return reverse_lazy("participation:team_detail", args=(registration.team_id,)) | ||||
|             raise PermissionDenied(_("You are not in a team.")) | ||||
|         raise PermissionDenied(_("You don't participate, so you don't have any team.")) | ||||
|  | ||||
|  | ||||
| class TeamDetailView(LoginRequiredMixin, FormMixin, ProcessFormView, DetailView): | ||||
|     """ | ||||
|     Display the detail of a team. | ||||
|     """ | ||||
|     model = Team | ||||
|  | ||||
|     def get(self, request, *args, **kwargs): | ||||
|         user = request.user | ||||
|         self.object = self.get_object() | ||||
|         # Ensure that the user is an admin or a member of the team | ||||
|         if user.registration.is_admin or user.registration.participates and \ | ||||
|                 user.registration.team and user.registration.team.pk == kwargs["pk"]: | ||||
|             return super().get(request, *args, **kwargs) | ||||
|         raise PermissionDenied | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         context = super().get_context_data(**kwargs) | ||||
|  | ||||
|         team = self.get_object() | ||||
|         context["title"] = _("Detail of team {trigram}").format(trigram=self.object.trigram) | ||||
|         context["request_validation_form"] = RequestValidationForm(self.request.POST or None) | ||||
|         context["validation_form"] = ValidateParticipationForm(self.request.POST or None) | ||||
|         # A team is complete when there are at least 3 members that have sent their photo authorization | ||||
|         # and confirmed their email address | ||||
|         context["can_validate"] = team.students.count() >= 3 and \ | ||||
|             all(r.email_confirmed for r in team.students.all()) and \ | ||||
|             all(r.photo_authorization for r in team.students.all()) and \ | ||||
|             team.participation.problem | ||||
|  | ||||
|         return context | ||||
|  | ||||
|     def get_form_class(self): | ||||
|         if not self.request.POST: | ||||
|             return RequestValidationForm | ||||
|         elif self.request.POST["_form_type"] == "RequestValidationForm": | ||||
|             return RequestValidationForm | ||||
|         elif self.request.POST["_form_type"] == "ValidateParticipationForm": | ||||
|             return ValidateParticipationForm | ||||
|  | ||||
|     def form_valid(self, form): | ||||
|         self.object = self.get_object() | ||||
|         if isinstance(form, RequestValidationForm): | ||||
|             return self.handle_request_validation(form) | ||||
|         elif isinstance(form, ValidateParticipationForm): | ||||
|             return self.handle_validate_participation(form) | ||||
|  | ||||
|     def handle_request_validation(self, form): | ||||
|         """ | ||||
|         A team requests to be validated | ||||
|         """ | ||||
|         if not self.request.user.registration.participates: | ||||
|             form.add_error(None, _("You don't participate, so you can't request the validation of the team.")) | ||||
|             return self.form_invalid(form) | ||||
|         if self.object.participation.valid is not None: | ||||
|             form.add_error(None, _("The validation of the team is already done or pending.")) | ||||
|             return self.form_invalid(form) | ||||
|         if not self.get_context_data()["can_validate"]: | ||||
|             form.add_error(None, _("The team can't be validated: missing email address confirmations, " | ||||
|                                    "photo authorizations, people or the chosen problem is not set.")) | ||||
|             return self.form_invalid(form) | ||||
|  | ||||
|         self.object.participation.valid = False | ||||
|         self.object.participation.save() | ||||
|  | ||||
|         for admin in AdminRegistration.objects.all(): | ||||
|             mail_context = dict(user=admin.user, team=self.object, domain=Site.objects.first().domain) | ||||
|             mail_plain = render_to_string("participation/mails/request_validation.txt", mail_context) | ||||
|             mail_html = render_to_string("participation/mails/request_validation.html", mail_context) | ||||
|             admin.user.email_user("[Corres2math] Validation d'équipe", mail_plain, html_message=mail_html) | ||||
|         return super().form_valid(form) | ||||
|  | ||||
|     def handle_validate_participation(self, form): | ||||
|         """ | ||||
|         An admin validates the team (or not) | ||||
|         """ | ||||
|         if not self.request.user.registration.is_admin: | ||||
|             form.add_error(None, _("You are not an administrator.")) | ||||
|             return self.form_invalid(form) | ||||
|         elif self.object.participation.valid is not False: | ||||
|             form.add_error(None, _("This team has no pending validation.")) | ||||
|             return self.form_invalid(form) | ||||
|  | ||||
|         if "validate" in self.request.POST: | ||||
|             self.object.participation.valid = True | ||||
|             self.object.participation.save() | ||||
|             mail_context = dict(team=self.object, message=form.cleaned_data["message"]) | ||||
|             mail_plain = render_to_string("participation/mails/team_validated.txt", mail_context) | ||||
|             mail_html = render_to_string("participation/mails/team_validated.html", mail_context) | ||||
|             send_mail("[Corres2math] Équipe validée", mail_plain, None, [self.object.email], html_message=mail_html) | ||||
|  | ||||
|             get_sympa_client().subscribe(self.object.email, "equipes", False, f"Equipe {self.object.name}") | ||||
|             get_sympa_client().unsubscribe(self.object.email, "equipes-non-valides", False) | ||||
|             get_sympa_client().subscribe(self.object.email, f"probleme-{self.object.participation.problem}", False, | ||||
|                                          f"Equipe {self.object.name}") | ||||
|         elif "invalidate" in self.request.POST: | ||||
|             self.object.participation.valid = None | ||||
|             self.object.participation.save() | ||||
|             mail_context = dict(team=self.object, message=form.cleaned_data["message"]) | ||||
|             mail_plain = render_to_string("participation/mails/team_not_validated.txt", mail_context) | ||||
|             mail_html = render_to_string("participation/mails/team_not_validated.html", mail_context) | ||||
|             send_mail("[Corres2math] Équipe non validée", mail_plain, None, [self.object.email], | ||||
|                       html_message=mail_html) | ||||
|         else: | ||||
|             form.add_error(None, _("You must specify if you validate the registration or not.")) | ||||
|             return self.form_invalid(form) | ||||
|         return super().form_valid(form) | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return self.request.path | ||||
|  | ||||
|  | ||||
| class TeamUpdateView(LoginRequiredMixin, UpdateView): | ||||
|     """ | ||||
|     Update the detail of a team | ||||
|     """ | ||||
|     model = Team | ||||
|     form_class = TeamForm | ||||
|     template_name = "participation/update_team.html" | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         user = request.user | ||||
|         if not user.is_authenticated: | ||||
|             return super().handle_no_permission() | ||||
|         if user.registration.is_admin or user.registration.participates and \ | ||||
|                 user.registration.team and \ | ||||
|                 user.registration.team.pk == kwargs["pk"]: | ||||
|             return super().dispatch(request, *args, **kwargs) | ||||
|         raise PermissionDenied | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         context = super().get_context_data(**kwargs) | ||||
|         context["participation_form"] = ParticipationForm(data=self.request.POST or None, | ||||
|                                                           instance=self.object.participation) | ||||
|         context["title"] = _("Update team {trigram}").format(trigram=self.object.trigram) | ||||
|         return context | ||||
|  | ||||
|     @transaction.atomic | ||||
|     def form_valid(self, form): | ||||
|         participation_form = ParticipationForm(data=self.request.POST or None, instance=self.object.participation) | ||||
|         if not participation_form.is_valid(): | ||||
|             return self.form_invalid(form) | ||||
|  | ||||
|         participation_form.save() | ||||
|         return super().form_valid(form) | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:team_detail", args=(self.object.pk,)) | ||||
|  | ||||
|  | ||||
| class TeamAuthorizationsView(LoginRequiredMixin, DetailView): | ||||
|     """ | ||||
|     Get as a ZIP archive all the authorizations that are sent | ||||
|     """ | ||||
|     model = Team | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         user = request.user | ||||
|         if not user.is_authenticated: | ||||
|             return super().handle_no_permission() | ||||
|         if user.registration.is_admin or user.registration.participates and user.registration.team.pk == kwargs["pk"]: | ||||
|             return super().dispatch(request, *args, **kwargs) | ||||
|         raise PermissionDenied | ||||
|  | ||||
|     def get(self, request, *args, **kwargs): | ||||
|         team = self.get_object() | ||||
|         output = BytesIO() | ||||
|         zf = ZipFile(output, "w") | ||||
|         for student in team.students.all(): | ||||
|             magic = Magic(mime=True) | ||||
|             mime_type = magic.from_file("media/" + student.photo_authorization.name) | ||||
|             ext = mime_type.split("/")[1].replace("jpeg", "jpg") | ||||
|             zf.write("media/" + student.photo_authorization.name, | ||||
|                      _("Photo authorization of {student}.{ext}").format(student=str(student), ext=ext)) | ||||
|         zf.close() | ||||
|         response = HttpResponse(content_type="application/zip") | ||||
|         response["Content-Disposition"] = "attachment; filename=\"{filename}\"" \ | ||||
|             .format(filename=_("Photo authorizations of team {trigram}.zip").format(trigram=team.trigram)) | ||||
|         response.write(output.getvalue()) | ||||
|         return response | ||||
|  | ||||
|  | ||||
| class TeamLeaveView(LoginRequiredMixin, TemplateView): | ||||
|     """ | ||||
|     A team member leaves a team | ||||
|     """ | ||||
|     template_name = "participation/team_leave.html" | ||||
|     extra_context = dict(title=_("Leave team")) | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         if not request.user.is_authenticated: | ||||
|             return self.handle_no_permission() | ||||
|         if not request.user.registration.participates or not request.user.registration.team: | ||||
|             raise PermissionDenied(_("You are not in a team.")) | ||||
|         if request.user.registration.team.participation.valid: | ||||
|             raise PermissionDenied(_("The team is already validated or the validation is pending.")) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     @transaction.atomic() | ||||
|     def post(self, request, *args, **kwargs): | ||||
|         """ | ||||
|         When the team is left, the user is unsubscribed from the team mailing list | ||||
|         and kicked from the team room. | ||||
|         """ | ||||
|         team = request.user.registration.team | ||||
|         request.user.registration.team = None | ||||
|         request.user.registration.save() | ||||
|         get_sympa_client().unsubscribe(request.user.email, f"equipe-{team.trigram.lower()}", False) | ||||
|         Matrix.kick(f"#equipe-{team.trigram.lower()}:correspondances-maths.fr", | ||||
|                     f"@{request.user.registration.matrix_username}:correspondances-maths.fr", | ||||
|                     "Équipe quittée") | ||||
|         if team.students.count() + team.coachs.count() == 0: | ||||
|             team.delete() | ||||
|         return redirect(reverse_lazy("index")) | ||||
|  | ||||
|  | ||||
| class MyParticipationDetailView(LoginRequiredMixin, RedirectView): | ||||
|     """ | ||||
|     Redirects to the detail view of the participation of the team. | ||||
|     """ | ||||
|     def get_redirect_url(self, *args, **kwargs): | ||||
|         user = self.request.user | ||||
|         registration = user.registration | ||||
|         if registration.participates: | ||||
|             if registration.team: | ||||
|                 return reverse_lazy("participation:participation_detail", args=(registration.team.participation.id,)) | ||||
|             raise PermissionDenied(_("You are not in a team.")) | ||||
|         raise PermissionDenied(_("You don't participate, so you don't have any team.")) | ||||
|  | ||||
|  | ||||
| class ParticipationDetailView(LoginRequiredMixin, DetailView): | ||||
|     """ | ||||
|     Display detail about the participation of a team, and manage the video submission. | ||||
|     """ | ||||
|     model = Participation | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         user = request.user | ||||
|         if not user.is_authenticated: | ||||
|             return super().handle_no_permission() | ||||
|         if not self.get_object().valid: | ||||
|             raise PermissionDenied(_("The team is not validated yet.")) | ||||
|         if user.registration.is_admin or user.registration.participates \ | ||||
|                 and user.registration.team.participation \ | ||||
|                 and user.registration.team.participation.pk == kwargs["pk"]: | ||||
|             return super().dispatch(request, *args, **kwargs) | ||||
|         raise PermissionDenied | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         context = super().get_context_data(**kwargs) | ||||
|  | ||||
|         context["title"] = lambda: _("Participation of team {trigram}").format(trigram=self.object.team.trigram) | ||||
|         context["current_phase"] = Phase.current_phase() | ||||
|  | ||||
|         return context | ||||
|  | ||||
|  | ||||
| class SetParticipationReceiveParticipationView(AdminMixin, UpdateView): | ||||
|     """ | ||||
|     Define the solution that a team will receive. | ||||
|     """ | ||||
|     model = Participation | ||||
|     form_class = ReceiveParticipationForm | ||||
|     template_name = "participation/receive_participation_form.html" | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:participation_detail", args=(self.kwargs["pk"],)) | ||||
|  | ||||
|  | ||||
| class SetParticipationSendParticipationView(AdminMixin, UpdateView): | ||||
|     """ | ||||
|     Define the team where the solution will be sent. | ||||
|     """ | ||||
|     model = Participation | ||||
|     form_class = SendParticipationForm | ||||
|     template_name = "participation/send_participation_form.html" | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:participation_detail", args=(self.kwargs["pk"],)) | ||||
|  | ||||
|  | ||||
| class CreateQuestionView(LoginRequiredMixin, CreateView): | ||||
|     """ | ||||
|     Ask a question to another team. | ||||
|     """ | ||||
|     participation: Participation | ||||
|     model = Question | ||||
|     form_class = QuestionForm | ||||
|     extra_context = dict(title=_("Create question")) | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         if not request.user.is_authenticated: | ||||
|             return self.handle_no_permission() | ||||
|         self.participation = Participation.objects.get(pk=kwargs["pk"]) | ||||
|         if request.user.registration.is_admin or \ | ||||
|                 request.user.registration.participates and \ | ||||
|                 self.participation.valid and \ | ||||
|                 request.user.registration.team.pk == self.participation.team_id: | ||||
|             return super().dispatch(request, *args, **kwargs) | ||||
|         raise PermissionDenied | ||||
|  | ||||
|     def form_valid(self, form): | ||||
|         form.instance.participation = self.participation | ||||
|         return super().form_valid(form) | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:participation_detail", args=(self.participation.pk,)) | ||||
|  | ||||
|  | ||||
| class UpdateQuestionView(LoginRequiredMixin, UpdateView): | ||||
|     """ | ||||
|     Edit a question. | ||||
|     """ | ||||
|     model = Question | ||||
|     form_class = QuestionForm | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         self.object = self.get_object() | ||||
|         if not request.user.is_authenticated: | ||||
|             return self.handle_no_permission() | ||||
|         if request.user.registration.is_admin or \ | ||||
|                 request.user.registration.participates and \ | ||||
|                 self.object.participation.valid and \ | ||||
|                 request.user.registration.team.pk == self.object.participation.team_id: | ||||
|             return super().dispatch(request, *args, **kwargs) | ||||
|         raise PermissionDenied | ||||
|  | ||||
|     def form_valid(self, form): | ||||
|         if not self.request.user.registration.is_admin and Phase.current_phase().phase_number != 2: | ||||
|             form.add_error(None, _("You can update your questions now.")) | ||||
|             return self.form_invalid(form) | ||||
|         return super().form_valid(form) | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:participation_detail", args=(self.object.participation.pk,)) | ||||
|  | ||||
|  | ||||
| class DeleteQuestionView(LoginRequiredMixin, DeleteView): | ||||
|     """ | ||||
|     Remove a question. | ||||
|     """ | ||||
|     model = Question | ||||
|     extra_context = dict(title=_("Delete question")) | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         self.object = self.get_object() | ||||
|         if not request.user.is_authenticated: | ||||
|             return self.handle_no_permission() | ||||
|         if request.user.registration.is_admin or \ | ||||
|                 request.user.registration.participates and \ | ||||
|                 self.object.participation.valid and \ | ||||
|                 request.user.registration.team.pk == self.object.participation.team_id: | ||||
|             return super().dispatch(request, *args, **kwargs) | ||||
|         raise PermissionDenied | ||||
|  | ||||
|     def delete(self, request, *args, **kwargs): | ||||
|         if not request.user.registration.is_admin and Phase.current_phase().phase_number != 2: | ||||
|             raise PermissionDenied(_("You can update your questions now.")) | ||||
|         return super().delete(request, *args, **kwargs) | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:participation_detail", args=(self.object.participation.pk,)) | ||||
|  | ||||
|  | ||||
| class UploadVideoView(LoginRequiredMixin, UpdateView): | ||||
|     """ | ||||
|     Upload a solution video for a team. | ||||
|     """ | ||||
|     model = Video | ||||
|     form_class = UploadVideoForm | ||||
|     template_name = "participation/upload_video.html" | ||||
|     extra_context = dict(title=_("Upload video")) | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         user = request.user | ||||
|         if not user.is_authenticated: | ||||
|             return super().handle_no_permission() | ||||
|         if user.registration.is_admin or user.registration.participates \ | ||||
|                 and user.registration.team.participation.pk == self.get_object().participation.pk: | ||||
|             return super().dispatch(request, *args, **kwargs) | ||||
|         raise PermissionDenied | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:participation_detail", args=(self.object.participation.pk,)) | ||||
|  | ||||
|  | ||||
| class CalendarView(SingleTableView): | ||||
|     """ | ||||
|     Display the calendar of the action. | ||||
|     """ | ||||
|     table_class = CalendarTable | ||||
|     model = Phase | ||||
|     extra_context = dict(title=_("Calendar")) | ||||
|  | ||||
|  | ||||
| class PhaseUpdateView(AdminMixin, UpdateView): | ||||
|     """ | ||||
|     Update a phase of the calendar, if we have sufficient rights. | ||||
|     """ | ||||
|     model = Phase | ||||
|     form_class = PhaseForm | ||||
|     extra_context = dict(title=_("Calendar update")) | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("participation:calendar") | ||||
							
								
								
									
										4
									
								
								apps/registration/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								apps/registration/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| default_app_config = 'registration.apps.RegistrationConfig' | ||||
							
								
								
									
										29
									
								
								apps/registration/admin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								apps/registration/admin.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.contrib import admin | ||||
| from polymorphic.admin import PolymorphicChildModelAdmin, PolymorphicParentModelAdmin | ||||
|  | ||||
| from .models import AdminRegistration, CoachRegistration, Registration, StudentRegistration | ||||
|  | ||||
|  | ||||
| @admin.register(Registration) | ||||
| class RegistrationAdmin(PolymorphicParentModelAdmin): | ||||
|     child_models = (StudentRegistration, CoachRegistration, AdminRegistration,) | ||||
|     list_display = ("user", "type", "email_confirmed",) | ||||
|     polymorphic_list = True | ||||
|  | ||||
|  | ||||
| @admin.register(StudentRegistration) | ||||
| class StudentRegistrationAdmin(PolymorphicChildModelAdmin): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| @admin.register(CoachRegistration) | ||||
| class CoachRegistrationAdmin(PolymorphicChildModelAdmin): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| @admin.register(AdminRegistration) | ||||
| class AdminRegistrationAdmin(PolymorphicChildModelAdmin): | ||||
|     pass | ||||
							
								
								
									
										23
									
								
								apps/registration/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								apps/registration/apps.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.apps import AppConfig | ||||
| from django.db.models.signals import post_save, pre_save | ||||
|  | ||||
|  | ||||
| class RegistrationConfig(AppConfig): | ||||
|     """ | ||||
|     Registration app contains the detail about users only. | ||||
|     """ | ||||
|     name = 'registration' | ||||
|  | ||||
|     def ready(self): | ||||
|         from registration.signals import create_admin_registration, invite_to_public_rooms, \ | ||||
|             set_username, send_email_link | ||||
|         pre_save.connect(set_username, "auth.User") | ||||
|         pre_save.connect(send_email_link, "auth.User") | ||||
|         post_save.connect(create_admin_registration, "auth.User") | ||||
|         post_save.connect(invite_to_public_rooms, "registration.Registration") | ||||
|         post_save.connect(invite_to_public_rooms, "registration.StudentRegistration") | ||||
|         post_save.connect(invite_to_public_rooms, "registration.CoachRegistration") | ||||
|         post_save.connect(invite_to_public_rooms, "registration.AdminRegistration") | ||||
							
								
								
									
										17
									
								
								apps/registration/auth.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								apps/registration/auth.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from cas_server.auth import DjangoAuthUser  # pragma: no cover | ||||
|  | ||||
|  | ||||
| class CustomAuthUser(DjangoAuthUser):  # pragma: no cover | ||||
|     """ | ||||
|     Override Django Auth User model to define a custom Matrix username. | ||||
|     """ | ||||
|  | ||||
|     def attributs(self): | ||||
|         d = super().attributs() | ||||
|         if self.user: | ||||
|             d["matrix_username"] = self.user.registration.matrix_username | ||||
|             d["display_name"] = str(self.user.registration) | ||||
|         return d | ||||
							
								
								
									
										26
									
								
								apps/registration/fixtures/initial.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								apps/registration/fixtures/initial.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| [ | ||||
|     { | ||||
|         "model": "cas_server.servicepattern", | ||||
|         "pk": 1, | ||||
|         "fields": { | ||||
|             "pos": 100, | ||||
|             "name": "Plateforme des Correspondances", | ||||
|             "pattern": "^https://correspondances-maths.fr:8448/.*$", | ||||
|             "user_field": "matrix_username", | ||||
|             "restrict_users": false, | ||||
|             "proxy": true, | ||||
|             "proxy_callback": true, | ||||
|             "single_log_out": true, | ||||
|             "single_log_out_callback": "" | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
|         "model": "cas_server.replaceattributname", | ||||
|         "pk": 1, | ||||
|         "fields": { | ||||
|             "name": "display_name", | ||||
|             "replace": "", | ||||
|             "service_pattern": 1 | ||||
|         } | ||||
|     } | ||||
| ] | ||||
							
								
								
									
										110
									
								
								apps/registration/forms.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								apps/registration/forms.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django import forms | ||||
| from django.contrib.auth.forms import UserCreationForm | ||||
| from django.contrib.auth.models import User | ||||
| from django.core.exceptions import ValidationError | ||||
| from django.forms import FileInput | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
| from .models import AdminRegistration, CoachRegistration, StudentRegistration | ||||
|  | ||||
|  | ||||
| class SignupForm(UserCreationForm): | ||||
|     """ | ||||
|     Signup form to registers participants and coaches | ||||
|     They can choose the role at the registration. | ||||
|     """ | ||||
|  | ||||
|     role = forms.ChoiceField( | ||||
|         label=lambda: _("role").capitalize(), | ||||
|         choices=lambda: [ | ||||
|             ("participant", _("participant").capitalize()), | ||||
|             ("coach", _("coach").capitalize()), | ||||
|         ], | ||||
|     ) | ||||
|  | ||||
|     def clean_email(self): | ||||
|         """ | ||||
|         Ensure that the email address is unique. | ||||
|         """ | ||||
|         email = self.data["email"] | ||||
|         if User.objects.filter(email=email).exists(): | ||||
|             self.add_error("email", _("This email address is already used.")) | ||||
|         return email | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.fields["first_name"].required = True | ||||
|         self.fields["last_name"].required = True | ||||
|         self.fields["email"].required = True | ||||
|  | ||||
|     class Meta: | ||||
|         model = User | ||||
|         fields = ('first_name', 'last_name', 'email', 'password1', 'password2', 'role',) | ||||
|  | ||||
|  | ||||
| class UserForm(forms.ModelForm): | ||||
|     """ | ||||
|     Replace the default user form to require the first name, last name and the email. | ||||
|     The username is always equal to the email. | ||||
|     """ | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.fields["first_name"].required = True | ||||
|         self.fields["last_name"].required = True | ||||
|         self.fields["email"].required = True | ||||
|  | ||||
|     class Meta: | ||||
|         model = User | ||||
|         fields = ('first_name', 'last_name', 'email',) | ||||
|  | ||||
|  | ||||
| class StudentRegistrationForm(forms.ModelForm): | ||||
|     """ | ||||
|     A student can update its class, its school and if it allows Animath to contact him/her later. | ||||
|     """ | ||||
|     class Meta: | ||||
|         model = StudentRegistration | ||||
|         fields = ('team', 'student_class', 'school', 'give_contact_to_animath', 'email_confirmed',) | ||||
|  | ||||
|  | ||||
| class PhotoAuthorizationForm(forms.ModelForm): | ||||
|     """ | ||||
|     Form to send a photo authorization. | ||||
|     """ | ||||
|     def clean_photo_authorization(self): | ||||
|         if "photo_authorization" in self.files: | ||||
|             file = self.files["photo_authorization"] | ||||
|             if file.size > 2e6: | ||||
|                 raise ValidationError(_("The uploaded file size must be under 2 Mo.")) | ||||
|             if file.content_type not in ["application/pdf", "image/png", "image/jpeg"]: | ||||
|                 raise ValidationError(_("The uploaded file must be a PDF, PNG of JPEG file.")) | ||||
|             return self.cleaned_data["photo_authorization"] | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.fields["photo_authorization"].widget = FileInput() | ||||
|  | ||||
|     class Meta: | ||||
|         model = StudentRegistration | ||||
|         fields = ('photo_authorization',) | ||||
|  | ||||
|  | ||||
| class CoachRegistrationForm(forms.ModelForm): | ||||
|     """ | ||||
|     A coach can tell its professional activity. | ||||
|     """ | ||||
|     class Meta: | ||||
|         model = CoachRegistration | ||||
|         fields = ('team', 'professional_activity', 'give_contact_to_animath', 'email_confirmed',) | ||||
|  | ||||
|  | ||||
| class AdminRegistrationForm(forms.ModelForm): | ||||
|     """ | ||||
|     Admins can tell everything they want. | ||||
|     """ | ||||
|     class Meta: | ||||
|         model = AdminRegistration | ||||
|         fields = ('role', 'give_contact_to_animath', 'email_confirmed',) | ||||
							
								
								
									
										74
									
								
								apps/registration/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								apps/registration/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | ||||
| # Generated by Django 3.1.3 on 2020-11-04 12:05 | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.db import migrations, models | ||||
| import django.db.models.deletion | ||||
| import registration.models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     initial = True | ||||
|  | ||||
|     dependencies = [ | ||||
|         migrations.swappable_dependency(settings.AUTH_USER_MODEL), | ||||
|         ('participation', '0001_initial'), | ||||
|         ('contenttypes', '0002_remove_content_type_name'), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.CreateModel( | ||||
|             name='Registration', | ||||
|             fields=[ | ||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||
|                 ('give_contact_to_animath', models.BooleanField(default=False, verbose_name='Grant Animath to contact me in the future about other actions')), | ||||
|                 ('email_confirmed', models.BooleanField(default=False, verbose_name='email confirmed')), | ||||
|                 ('polymorphic_ctype', models.ForeignKey(editable=False, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='polymorphic_registration.registration_set+', to='contenttypes.contenttype')), | ||||
|                 ('user', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL, verbose_name='user')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'registration', | ||||
|                 'verbose_name_plural': 'registrations', | ||||
|             }, | ||||
|         ), | ||||
|         migrations.CreateModel( | ||||
|             name='AdminRegistration', | ||||
|             fields=[ | ||||
|                 ('registration_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='registration.registration')), | ||||
|                 ('role', models.TextField(verbose_name='role of the administrator')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'admin registration', | ||||
|                 'verbose_name_plural': 'admin registrations', | ||||
|             }, | ||||
|             bases=('registration.registration',), | ||||
|         ), | ||||
|         migrations.CreateModel( | ||||
|             name='StudentRegistration', | ||||
|             fields=[ | ||||
|                 ('registration_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='registration.registration')), | ||||
|                 ('student_class', models.IntegerField(choices=[(12, '12th grade'), (11, '11th grade'), (10, '10th grade or lower')], verbose_name='student class')), | ||||
|                 ('school', models.CharField(max_length=255, verbose_name='school')), | ||||
|                 ('photo_authorization', models.FileField(blank=True, default='', upload_to=registration.models.get_random_filename, verbose_name='photo authorization')), | ||||
|                 ('team', models.ForeignKey(default=None, blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, related_name='students', to='participation.team', verbose_name='team')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'student registration', | ||||
|                 'verbose_name_plural': 'student registrations', | ||||
|             }, | ||||
|             bases=('registration.registration',), | ||||
|         ), | ||||
|         migrations.CreateModel( | ||||
|             name='CoachRegistration', | ||||
|             fields=[ | ||||
|                 ('registration_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='registration.registration')), | ||||
|                 ('professional_activity', models.TextField(verbose_name='professional activity')), | ||||
|                 ('team', models.ForeignKey(default=None, blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, related_name='coachs', to='participation.team', verbose_name='team')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'coach registration', | ||||
|                 'verbose_name_plural': 'coach registrations', | ||||
|             }, | ||||
|             bases=('registration.registration',), | ||||
|         ), | ||||
|     ] | ||||
							
								
								
									
										2
									
								
								apps/registration/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								apps/registration/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
							
								
								
									
										201
									
								
								apps/registration/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										201
									
								
								apps/registration/models.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,201 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from corres2math.tokens import email_validation_token | ||||
| from django.contrib.sites.models import Site | ||||
| from django.db import models | ||||
| from django.template import loader | ||||
| from django.urls import reverse_lazy | ||||
| from django.utils.crypto import get_random_string | ||||
| from django.utils.encoding import force_bytes | ||||
| from django.utils.http import urlsafe_base64_encode | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from polymorphic.models import PolymorphicModel | ||||
|  | ||||
|  | ||||
| class Registration(PolymorphicModel): | ||||
|     """ | ||||
|     Registrations store extra content that are not asked in the User Model. | ||||
|     This is specific to the role of the user, see StudentRegistration, | ||||
|     ClassRegistration or AdminRegistration.. | ||||
|     """ | ||||
|     user = models.OneToOneField( | ||||
|         "auth.User", | ||||
|         on_delete=models.CASCADE, | ||||
|         verbose_name=_("user"), | ||||
|     ) | ||||
|  | ||||
|     give_contact_to_animath = models.BooleanField( | ||||
|         default=False, | ||||
|         verbose_name=_("Grant Animath to contact me in the future about other actions"), | ||||
|     ) | ||||
|  | ||||
|     email_confirmed = models.BooleanField( | ||||
|         default=False, | ||||
|         verbose_name=_("email confirmed"), | ||||
|     ) | ||||
|  | ||||
|     def send_email_validation_link(self): | ||||
|         """ | ||||
|         The account got created or the email got changed. | ||||
|         Send an email that contains a link to validate the address. | ||||
|         """ | ||||
|         subject = "[Corres2math] " + str(_("Activate your Correspondances account")) | ||||
|         token = email_validation_token.make_token(self.user) | ||||
|         uid = urlsafe_base64_encode(force_bytes(self.user.pk)) | ||||
|         site = Site.objects.first() | ||||
|         message = loader.render_to_string('registration/mails/email_validation_email.txt', | ||||
|                                           { | ||||
|                                               'user': self.user, | ||||
|                                               'domain': site.domain, | ||||
|                                               'token': token, | ||||
|                                               'uid': uid, | ||||
|                                           }) | ||||
|         html = loader.render_to_string('registration/mails/email_validation_email.html', | ||||
|                                        { | ||||
|                                            'user': self.user, | ||||
|                                            'domain': site.domain, | ||||
|                                            'token': token, | ||||
|                                            'uid': uid, | ||||
|                                        }) | ||||
|         self.user.email_user(subject, message, html_message=html) | ||||
|  | ||||
|     @property | ||||
|     def type(self):  # pragma: no cover | ||||
|         raise NotImplementedError | ||||
|  | ||||
|     @property | ||||
|     def form_class(self):  # pragma: no cover | ||||
|         raise NotImplementedError | ||||
|  | ||||
|     @property | ||||
|     def participates(self): | ||||
|         return isinstance(self, StudentRegistration) or isinstance(self, CoachRegistration) | ||||
|  | ||||
|     @property | ||||
|     def is_admin(self): | ||||
|         return isinstance(self, AdminRegistration) or self.user.is_superuser | ||||
|  | ||||
|     @property | ||||
|     def matrix_username(self): | ||||
|         return f"corres2math_{self.user.pk}" | ||||
|  | ||||
|     def get_absolute_url(self): | ||||
|         return reverse_lazy("registration:user_detail", args=(self.user_id,)) | ||||
|  | ||||
|     def __str__(self): | ||||
|         return f"{self.user.first_name} {self.user.last_name}" | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("registration") | ||||
|         verbose_name_plural = _("registrations") | ||||
|  | ||||
|  | ||||
| def get_random_filename(instance, filename): | ||||
|     return "authorization/photo/" + get_random_string(64) | ||||
|  | ||||
|  | ||||
| class StudentRegistration(Registration): | ||||
|     """ | ||||
|     Specific registration for students. | ||||
|     They have a team, a student class and a school. | ||||
|     """ | ||||
|     team = models.ForeignKey( | ||||
|         "participation.Team", | ||||
|         related_name="students", | ||||
|         on_delete=models.PROTECT, | ||||
|         null=True, | ||||
|         blank=True, | ||||
|         default=None, | ||||
|         verbose_name=_("team"), | ||||
|     ) | ||||
|  | ||||
|     student_class = models.IntegerField( | ||||
|         choices=[ | ||||
|             (12, _("12th grade")), | ||||
|             (11, _("11th grade")), | ||||
|             (10, _("10th grade or lower")), | ||||
|         ], | ||||
|         verbose_name=_("student class"), | ||||
|     ) | ||||
|  | ||||
|     school = models.CharField( | ||||
|         max_length=255, | ||||
|         verbose_name=_("school"), | ||||
|     ) | ||||
|  | ||||
|     photo_authorization = models.FileField( | ||||
|         verbose_name=_("photo authorization"), | ||||
|         upload_to=get_random_filename, | ||||
|         blank=True, | ||||
|         default="", | ||||
|     ) | ||||
|  | ||||
|     @property | ||||
|     def type(self): | ||||
|         return _("student") | ||||
|  | ||||
|     @property | ||||
|     def form_class(self): | ||||
|         from registration.forms import StudentRegistrationForm | ||||
|         return StudentRegistrationForm | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("student registration") | ||||
|         verbose_name_plural = _("student registrations") | ||||
|  | ||||
|  | ||||
| class CoachRegistration(Registration): | ||||
|     """ | ||||
|     Specific registration for coaches. | ||||
|     They have a team and a professional activity. | ||||
|     """ | ||||
|     team = models.ForeignKey( | ||||
|         "participation.Team", | ||||
|         related_name="coachs", | ||||
|         on_delete=models.PROTECT, | ||||
|         null=True, | ||||
|         blank=True, | ||||
|         default=None, | ||||
|         verbose_name=_("team"), | ||||
|     ) | ||||
|  | ||||
|     professional_activity = models.TextField( | ||||
|         verbose_name=_("professional activity"), | ||||
|     ) | ||||
|  | ||||
|     @property | ||||
|     def type(self): | ||||
|         return _("coach") | ||||
|  | ||||
|     @property | ||||
|     def form_class(self): | ||||
|         from registration.forms import CoachRegistrationForm | ||||
|         return CoachRegistrationForm | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("coach registration") | ||||
|         verbose_name_plural = _("coach registrations") | ||||
|  | ||||
|  | ||||
| class AdminRegistration(Registration): | ||||
|     """ | ||||
|     Specific registration for admins. | ||||
|     They have a field to justify they status. | ||||
|     """ | ||||
|     role = models.TextField( | ||||
|         verbose_name=_("role of the administrator"), | ||||
|     ) | ||||
|  | ||||
|     @property | ||||
|     def type(self): | ||||
|         return _("admin") | ||||
|  | ||||
|     @property | ||||
|     def form_class(self): | ||||
|         from registration.forms import AdminRegistrationForm | ||||
|         return AdminRegistrationForm | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("admin registration") | ||||
|         verbose_name_plural = _("admin registrations") | ||||
							
								
								
									
										16
									
								
								apps/registration/search_indexes.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								apps/registration/search_indexes.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from haystack import indexes | ||||
|  | ||||
| from .models import Registration | ||||
|  | ||||
|  | ||||
| class RegistrationIndex(indexes.ModelSearchIndex, indexes.Indexable): | ||||
|     """ | ||||
|     Registrations are indexed by the user detail. | ||||
|     """ | ||||
|     text = indexes.NgramField(document=True, use_template=True) | ||||
|  | ||||
|     class Meta: | ||||
|         model = Registration | ||||
							
								
								
									
										56
									
								
								apps/registration/signals.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								apps/registration/signals.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from corres2math.lists import get_sympa_client | ||||
| from corres2math.matrix import Matrix | ||||
| from django.contrib.auth.models import User | ||||
|  | ||||
| from .models import AdminRegistration, Registration | ||||
|  | ||||
|  | ||||
| def set_username(instance, **_): | ||||
|     """ | ||||
|     Ensure that the user username is always equal to the user email address. | ||||
|     """ | ||||
|     instance.username = instance.email | ||||
|  | ||||
|  | ||||
| def send_email_link(instance, **_): | ||||
|     """ | ||||
|     If the email address got changed, send a new validation link | ||||
|      and update the registration status in the team mailing list. | ||||
|     """ | ||||
|     if instance.pk: | ||||
|         old_instance = User.objects.get(pk=instance.pk) | ||||
|         if old_instance.email != instance.email: | ||||
|             registration = Registration.objects.get(user=instance) | ||||
|             registration.email_confirmed = False | ||||
|             registration.save() | ||||
|             registration.user = instance | ||||
|             registration.send_email_validation_link() | ||||
|  | ||||
|             if registration.participates and registration.team: | ||||
|                 get_sympa_client().unsubscribe(old_instance.email, f"equipe-{registration.team.trigram.lower()}", False) | ||||
|                 get_sympa_client().subscribe(instance.email, f"equipe-{registration.team.trigram.lower()}", False, | ||||
|                                              f"{instance.first_name} {instance.last_name}") | ||||
|  | ||||
|  | ||||
| def create_admin_registration(instance, **_): | ||||
|     """ | ||||
|     When a super user got created through console, | ||||
|     ensure that an admin registration is created. | ||||
|     """ | ||||
|     if instance.is_superuser: | ||||
|         AdminRegistration.objects.get_or_create(user=instance) | ||||
|  | ||||
|  | ||||
| def invite_to_public_rooms(instance: Registration, created: bool, **_): | ||||
|     """ | ||||
|     When a user got registered, automatically invite the Matrix user into public rooms. | ||||
|     """ | ||||
|     if not created: | ||||
|         Matrix.invite("#annonces:correspondances-maths.fr", f"@{instance.matrix_username}:correspondances-maths.fr") | ||||
|         Matrix.invite("#faq:correspondances-maths.fr", f"@{instance.matrix_username}:correspondances-maths.fr") | ||||
|         Matrix.invite("#je-cherche-une-equip:correspondances-maths.fr", | ||||
|                       f"@{instance.matrix_username}:correspondances-maths.fr") | ||||
|         Matrix.invite("#flood:correspondances-maths.fr", f"@{instance.matrix_username}:correspondances-maths.fr") | ||||
							
								
								
									
										27
									
								
								apps/registration/tables.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								apps/registration/tables.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| import django_tables2 as tables | ||||
|  | ||||
| from .models import Registration | ||||
|  | ||||
|  | ||||
| class RegistrationTable(tables.Table): | ||||
|     """ | ||||
|     Table of all registrations. | ||||
|     """ | ||||
|     last_name = tables.LinkColumn( | ||||
|         'registration:user_detail', | ||||
|         args=[tables.A("user_id")], | ||||
|         verbose_name=lambda: _("last name").capitalize(), | ||||
|         accessor="user__last_name", | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         attrs = { | ||||
|             'class': 'table table condensed table-striped', | ||||
|         } | ||||
|         model = Registration | ||||
|         fields = ('last_name', 'user__first_name', 'user__email', 'type',) | ||||
|         template_name = 'django_tables2/bootstrap4.html' | ||||
| @@ -0,0 +1,31 @@ | ||||
| {% extends "base.html" %} | ||||
| {% comment %} | ||||
| SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% endcomment %} | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
| <div class="card bg-light"> | ||||
|     <h3 class="card-header text-center"> | ||||
|         {{ title }} | ||||
|     </h3> | ||||
|     <div class="card-body"> | ||||
|         {% if validlink %} | ||||
|         <p> | ||||
|             {% trans "Your email have successfully been validated." %} | ||||
|         </p> | ||||
|         <p> | ||||
|             {% blocktrans %}You can now <a href="{{ login_url }}">log in</a>.{% endblocktrans %} | ||||
|         </p> | ||||
|         {% else %} | ||||
|         <p> | ||||
|             {% if user.is_authenticated and user.registration.email_confirmed %} | ||||
|                 {% trans "The link was invalid. The token may have expired, or your account is already activated. However, your account seems to be already valid." %} | ||||
|             {% else %} | ||||
|                 {% trans "The link was invalid. The token may have expired, or your account is already activated. Please send us an email to activate your account." %} | ||||
|             {% endif %} | ||||
|         </p> | ||||
|         {% endif %} | ||||
|      </div> | ||||
| </div> | ||||
| {% endblock %} | ||||
| @@ -0,0 +1,18 @@ | ||||
| {% extends "base.html" %} | ||||
| {% comment %} | ||||
| SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% endcomment %} | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
| <div class="card bg-light"> | ||||
|     <h3 class="card-header text-center"> | ||||
|         {% trans "Account activation" %} | ||||
|     </h3> | ||||
|     <div class="card-body"> | ||||
|         <p> | ||||
|             {% trans "An email has been sent. Please click on the link to activate your account." %} | ||||
|         </p> | ||||
|     </div> | ||||
| </div> | ||||
| {% endblock %} | ||||
| @@ -0,0 +1,36 @@ | ||||
| {% load i18n %} | ||||
|  | ||||
| <!DOCTYPE html> | ||||
| <html lang="fr"> | ||||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <title></title> | ||||
| </head> | ||||
| <body> | ||||
|  | ||||
| <p> | ||||
|     {% trans "Hi" %} {{ user.username }}, | ||||
| </p> | ||||
|  | ||||
| <p> | ||||
|     {% trans "You recently registered on the Correspondances platform. Please click on the link below to confirm your registration." %} | ||||
| </p> | ||||
|  | ||||
| <p> | ||||
|     <a href="https://{{ domain }}{% url 'registration:email_validation' uidb64=uid token=token %}"> | ||||
|         https://{{ domain }}{% url 'registration:email_validation' uidb64=uid token=token %} | ||||
|     </a> | ||||
| </p> | ||||
|  | ||||
| <p> | ||||
|     {% trans "This link is only valid for a couple of days, after that you will need to contact us to validate your email." %} | ||||
| </p> | ||||
|  | ||||
| <p> | ||||
|     {% trans "Thanks" %}, | ||||
| </p> | ||||
|  | ||||
| -- | ||||
| <p> | ||||
|     {% trans "The Correspondances team." %}<br> | ||||
| </p> | ||||
| @@ -0,0 +1,13 @@ | ||||
| {% load i18n %} | ||||
|  | ||||
| {% trans "Hi" %} {{ user.username }}, | ||||
|  | ||||
| {% trans "You recently registered on the Correspondances platform. Please click on the link below to confirm your registration." %} | ||||
|  | ||||
| https://{{ domain }}{% url 'registration:email_validation' uidb64=uid token=token %} | ||||
|  | ||||
| {% trans "This link is only valid for a couple of days, after that you will need to contact us to validate your email." %} | ||||
|  | ||||
| {% trans "Thanks" %}, | ||||
|  | ||||
| {% trans "The Correspondances team." %} | ||||
| @@ -0,0 +1,9 @@ | ||||
| {% extends "base.html" %} | ||||
| {% comment %} | ||||
| SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% endcomment %} | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <p>{% trans 'Your password was changed.' %}</p> | ||||
| {% endblock %} | ||||
| @@ -0,0 +1,13 @@ | ||||
| {% extends "base.html" %} | ||||
| {% comment %} | ||||
| SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% endcomment %} | ||||
| {% load i18n crispy_forms_tags %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post">{% csrf_token %} | ||||
|         <p>{% trans "Please enter your old password, for security's sake, and then enter your new password twice so we can verify you typed it in correctly." %}</p> | ||||
|         {{ form | crispy }} | ||||
|         <input class="btn btn-primary" type="submit" value="{% trans 'Change my password' %}"> | ||||
|     </form> | ||||
| {% endblock %} | ||||
| @@ -0,0 +1,12 @@ | ||||
| {% extends "base.html" %} | ||||
| {% comment %} | ||||
| SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% endcomment %} | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <p>{% trans "Your password has been set. You may go ahead and log in now." %}</p> | ||||
|     <p> | ||||
|         <a href="{{ login_url }}" class="btn btn-success">{% trans 'Log in' %}</a> | ||||
|     </p> | ||||
| {% endblock %} | ||||
| @@ -0,0 +1,17 @@ | ||||
| {% extends "base.html" %} | ||||
| {% comment %} | ||||
| SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% endcomment %} | ||||
| {% load i18n crispy_forms_tags %} | ||||
|  | ||||
| {% block content %} | ||||
|     {% if validlink %} | ||||
|         <p>{% trans "Please enter your new password twice so we can verify you typed it in correctly." %}</p> | ||||
|         <form method="post">{% csrf_token %} | ||||
|             {{ form | crispy }} | ||||
|             <input class="btn btn-primary" type="submit" value="{% trans 'Change my password' %}"> | ||||
|         </form> | ||||
|     {% else %} | ||||
|         <p>{% trans "The password reset link was invalid, possibly because it has already been used. Please request a new password reset." %}</p> | ||||
|     {% endif %} | ||||
| {% endblock %} | ||||
| @@ -0,0 +1,10 @@ | ||||
| {% extends "base.html" %} | ||||
| {% comment %} | ||||
| SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% endcomment %} | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <p>{% trans "We've emailed you instructions for setting your password, if an account exists with the email you entered. You should receive them shortly." %}</p> | ||||
|     <p>{% trans "If you don't receive an email, please make sure you've entered the address you registered with, and check your spam folder." %}</p> | ||||
| {% endblock %} | ||||
| @@ -0,0 +1,13 @@ | ||||
| {% extends "base.html" %} | ||||
| {% comment %} | ||||
| SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% endcomment %} | ||||
| {% load i18n crispy_forms_tags %} | ||||
|  | ||||
| {% block content %} | ||||
|     <p>{% trans "Forgotten your password? Enter your email address below, and we'll email instructions for setting a new one." %}</p> | ||||
|     <form method="post">{% csrf_token %} | ||||
|         {{ form | crispy }} | ||||
|         <input class="btn btn-primary" type="submit" value="{% trans 'Reset my password' %}"> | ||||
|     </form> | ||||
| {% endblock %} | ||||
							
								
								
									
										44
									
								
								apps/registration/templates/registration/signup.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								apps/registration/templates/registration/signup.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| <!-- templates/signup.html --> | ||||
| {% extends 'base.html' %} | ||||
| {% load crispy_forms_filters %} | ||||
| {% load i18n %} | ||||
| {% block title %}{% trans "Sign up" %}{% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|     <h2>{% trans "Sign up" %}</h2> | ||||
|  | ||||
|     <form method="post"> | ||||
|         {% csrf_token %} | ||||
|         {{ form|crispy }} | ||||
|         <div id="student_registration_form"> | ||||
|             {{ student_registration_form|crispy }} | ||||
|         </div> | ||||
|         <div id="coach_registration_form" class="d-none"> | ||||
|             {{ coach_registration_form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-success" type="submit"> | ||||
|             {% trans "Sign up" %} | ||||
|         </button> | ||||
|     </form> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block extrajavascript %} | ||||
|     <script> | ||||
|     $(document).ready(function() { | ||||
|         $("#id_role").change(function() { | ||||
|             let selected_role = $("#id_role :selected"); | ||||
|             if (selected_role.val() === "participant") { | ||||
|                 $("#student_registration_form").removeClass("d-none"); | ||||
|                 $("#coach_registration_form").addClass("d-none"); | ||||
|             } | ||||
|             else { | ||||
|                 $("#student_registration_form").addClass("d-none"); | ||||
|                 $("#coach_registration_form").removeClass("d-none"); | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         $("#student_registration_form :input").removeAttr("required"); | ||||
|         $("#coach_registration_form :input").removeAttr("required"); | ||||
|     }); | ||||
|     </script> | ||||
| {% endblock %} | ||||
							
								
								
									
										15
									
								
								apps/registration/templates/registration/update_user.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								apps/registration/templates/registration/update_user.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load crispy_forms_filters i18n %} | ||||
|  | ||||
| {% block content %} | ||||
|     <form method="post"> | ||||
|         <div id="form-content"> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|             {{ registration_form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-success" type="submit">{% trans "Update" %}</button> | ||||
|     </form> | ||||
| {% endblock content %} | ||||
|  | ||||
| @@ -0,0 +1,20 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load i18n static crispy_forms_filters %} | ||||
|  | ||||
| {% block content %} | ||||
|     <a class="btn btn-info" href="{% url "registration:user_detail" pk=object.user.pk %}"><i class="fas fa-arrow-left"></i> {% trans "Back to the user detail" %}</a> | ||||
|     <hr> | ||||
|     <form method="post" enctype="multipart/form-data"> | ||||
|         <div id="form-content"> | ||||
|             <div class="alert alert-info"> | ||||
|                 {% trans "Authorzation templates:" %} | ||||
|                 <a class="alert-link" href="{% static "Autorisation de droit à l'image - majeur.pdf" %}">{% trans "Adult" %}</a> — | ||||
|                 <a class="alert-link" href="{% static "Autorisation de droit à l'image - mineur.pdf" %}">{% trans "Child" %}</a> | ||||
|             </div> | ||||
|             {% csrf_token %} | ||||
|             {{ form|crispy }} | ||||
|         </div> | ||||
|         <button class="btn btn-success" type="submit">{% trans "Upload" %}</button> | ||||
|     </form> | ||||
| {% endblock %} | ||||
							
								
								
									
										98
									
								
								apps/registration/templates/registration/user_detail.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								apps/registration/templates/registration/user_detail.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,98 @@ | ||||
| {% extends "base.html" %} | ||||
|  | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
| {% trans "any" as any %} | ||||
|  | ||||
|     <div class="card bg-light shadow"> | ||||
|         <div class="card-header text-center"> | ||||
|             <h4>{{ user_object.first_name }} {{ user_object.last_name }}</h4> | ||||
|         </div> | ||||
|     <div class="card-body"> | ||||
|         <dl class="row"> | ||||
|             <dt class="col-sm-6 text-right">{% trans "Last name:" %}</dt> | ||||
|             <dd class="col-sm-6">{{ user_object.last_name }}</dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "First name:" %}</dt> | ||||
|             <dd class="col-sm-6">{{ user_object.first_name }}</dd> | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Email:" %}</dt> | ||||
|             <dd class="col-sm-6"><a href="mailto:{{ user_object.email }}">{{ user_object.email }}</a> | ||||
|                 {% if not user_object.registration.email_confirmed %} (<em>{% trans "Not confirmed" %}, <a href="{% url "registration:email_validation_resend" pk=user_object.pk %}">{% trans "resend the validation link" %}</a></em>){% endif %}</dd> | ||||
|  | ||||
|             {% if user_object.registration.participates or True %} | ||||
|                 <dt class="col-sm-6 text-right">{% trans "Team:" %}</dt> | ||||
|                 {% trans "any" as any %} | ||||
|                 <dd class="col-sm-6"> | ||||
|                     <a href="{% if user_object.registration.team %}{% url "participation:team_detail" pk=user_object.registration.team.pk %}{% else %}#{% endif %}"> | ||||
|                         {{ user_object.registration.team|default:any }} | ||||
|                     </a> | ||||
|                 </dd> | ||||
|             {% endif %} | ||||
|  | ||||
|             {% if user_object.registration.studentregistration %} | ||||
|                 <dt class="col-sm-6 text-right">{% trans "Student class:" %}</dt> | ||||
|                 <dd class="col-sm-6">{{ user_object.registration.get_student_class_display }}</dd> | ||||
|  | ||||
|                 <dt class="col-sm-6 text-right">{% trans "School:" %}</dt> | ||||
|                 <dd class="col-sm-6">{{ user_object.registration.school }}</dd> | ||||
|  | ||||
|                 <dt class="col-sm-6 text-right">{% trans "Photo authorization:" %}</dt> | ||||
|                 <dd class="col-sm-6"> | ||||
|                     {% if user_object.registration.photo_authorization %} | ||||
|                         <a href="{{ user_object.registration.photo_authorization.url }}" data-turbolinks="false">{% trans "Download" %}</a> | ||||
|                     {% endif %} | ||||
|                     {% if user_object.pk == user.pk %} | ||||
|                         <button class="btn btn-primary" data-toggle="modal" data-target="#uploadPhotoAuthorizationModal">{% trans "Replace" %}</button> | ||||
|                     {% endif %} | ||||
|                 </dd> | ||||
|             {% elif user_object.registration.coachregistration %} | ||||
|                 <dt class="col-sm-6 text-right">{% trans "Profesional activity:" %}</dt> | ||||
|                 <dd class="col-sm-6">{{ user_object.registration.professional_activity }}</dd> | ||||
|             {% elif user_object.registration.adminregistration %} | ||||
|                 <dt class="col-sm-6 text-right">{% trans "Role:" %}</dt> | ||||
|                 <dd class="col-sm-6">{{ user_object.registration.role }}</dd> | ||||
|             {% endif %} | ||||
|  | ||||
|             <dt class="col-sm-6 text-right">{% trans "Grant Animath to contact me in the future about other actions:" %}</dt> | ||||
|             <dd class="col-sm-6">{{ user_object.registration.give_contact_to_animath|yesno }}</dd> | ||||
|         </dl> | ||||
|     </div> | ||||
|     {% if user.pk == user_object.pk or user.registration.is_admin %} | ||||
|         <div class="card-footer text-center"> | ||||
|             <button class="btn btn-primary" data-toggle="modal" data-target="#updateUserModal">{% trans "Update" %}</button> | ||||
|             {% if user.registration.is_admin %} | ||||
|                 <a class="btn btn-info" href="{% url "registration:user_impersonate" pk=user_object.pk %}">{% trans "Impersonate" %}</a> | ||||
|             {% endif %} | ||||
|         </div> | ||||
|     {% endif %} | ||||
|     </div> | ||||
|  | ||||
|     {% trans "Update user" as modal_title %} | ||||
|     {% trans "Update" as modal_button %} | ||||
|     {% url "registration:update_user" pk=user_object.pk as modal_action %} | ||||
|     {% include "base_modal.html" with modal_id="updateUser" %} | ||||
|  | ||||
|     {% trans "Upload photo authorization" as modal_title %} | ||||
|     {% trans "Upload" as modal_button %} | ||||
|     {% url "registration:upload_user_photo_authorization" pk=user_object.registration.pk as modal_action %} | ||||
|     {% include "base_modal.html" with modal_id="uploadPhotoAuthorization" modal_enctype="multipart/form-data" %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block extrajavascript %} | ||||
|     <script> | ||||
|         $(document).ready(function() { | ||||
|             $('button[data-target="#updateUserModal"]').click(function() { | ||||
|                 let modalBody = $("#updateUserModal div.modal-body"); | ||||
|                 if (!modalBody.html().trim()) | ||||
|                     modalBody.load("{% url "registration:update_user" pk=user_object.pk %} #form-content"); | ||||
|             }); | ||||
|             $('button[data-target="#uploadPhotoAuthorizationModal"]').click(function() { | ||||
|                 let modalBody = $("#uploadPhotoAuthorizationModal div.modal-body"); | ||||
|                 if (!modalBody.html().trim()) | ||||
|                     modalBody.load("{% url "registration:upload_user_photo_authorization" pk=user_object.registration.pk %} #form-content"); | ||||
|             }); | ||||
|         }); | ||||
|     </script> | ||||
| {% endblock %} | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user