verejny kontaktnicek #71
							
								
								
									
										10
									
								
								Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,10 @@ | |||
| FROM python:3.11.10-bookworm | ||||
| WORKDIR /usr/src/app | ||||
| 
 | ||||
| RUN pip install --upgrade pip | ||||
| COPY ./requirements.txt . | ||||
| RUN pip install -r requirements.txt | ||||
| 
 | ||||
| COPY . . | ||||
| 
 | ||||
| CMD ["python3", "manage.py", "runserver", "0.0.0.0:8000"] | ||||
							
								
								
									
										12
									
								
								docker-compose.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,12 @@ | |||
| version: '3' | ||||
| 
 | ||||
| services: | ||||
|   django_app: | ||||
|     build: . | ||||
|     volumes: | ||||
|       - this_vol:/usr/src/app/ | ||||
|     ports: | ||||
|       - 8000:8000 | ||||
| 
 | ||||
| volumes: | ||||
|   this_vol: | ||||
|  | @ -342,6 +342,7 @@ SEMINAR_KONFERY_DIR = os.path.join('konfery') | |||
| KOREKTURY_PDF_DIR = os.path.join('korektury', 'pdf') | ||||
| KOREKTURY_IMG_DIR = os.path.join('korektury', 'img') | ||||
| CISLO_IMG_DIR = os.path.join('cislo', 'img') | ||||
| SOUSTREDENI_KONTAKTNICKY_DIR = os.path.join('soustredeni', 'kontaktnicky') | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,8 +1,8 @@ | |||
| -c constraints.txt | ||||
| # -c constraints.txt | ||||
| setuptools # django-polymorphic má rozbité dependencies | ||||
| 
 | ||||
| # basic libs | ||||
| psycopg2 # PostgreSQL adaptér | ||||
| # psycopg2 # PostgreSQL adaptér | ||||
| ipython # Interaktivní shell | ||||
| Unidecode # Přepisuje unicode do ASCII (např. soubory nebo e-maily) | ||||
| Pillow | ||||
|  |  | |||
|  | @ -18,9 +18,11 @@ from django.utils import timezone | |||
| def generate_filename_kontaktnicek(self, filename): | ||||
| 	# generate random string | ||||
| 	length = 32 | ||||
| 	fname = "kontaktnicky/" + timezone.now().strftime('%Y-%m-%d-%H_%M') | ||||
| 	file_type = filename.split('.')[-1] | ||||
| 	fname = "kontaktnicky/" + timezone.now().strftime('%Y-%m-%d-%H_%M') + "" | ||||
|  | ||||
| 	fname += ''.join(secrets.choice(string.ascii_uppercase + string.digits) for _ in range(length)) | ||||
| 
				
					
						ledoian
						commented  Prosím ne  Prosím ne `KOREKTURY_PDF_DIR`!! 
				
					
						ledoian
						commented  Also: možná je snazší a bezpečnější použít  Also: možná je snazší a bezpečnější použít `pathlib` než `os.path.join`… | ||||
| 	return os.path.join(settings.KOREKTURY_PDF_DIR, fname) | ||||
| 	fname += '.' + file_type | ||||
| 	return os.path.join(settings.SOUSTREDENI_KONTAKTNICKY_DIR, fname) | ||||
| 
 | ||||
| logger = logging.getLogger(__name__) | ||||
| 
 | ||||
|  | @ -76,8 +78,8 @@ class Soustredeni(SeminarModelBase): | |||
| 	exportovat = models.BooleanField('export do AESOPa', db_column='exportovat', default=False, | ||||
| 			help_text='Exportuje se jen podle tohoto flagu (ne veřejnosti)') | ||||
| 	 | ||||
| 
				
					
						zelvuska
						commented  Nemá to být  Nemá to být `null=True`? (Ale já tomu nerozumím…) | ||||
| 	kontaktnicek_vcf = models.FileField('kontaktníček vcf', upload_to=generate_filename_kontaktnicek, blank=True,) | ||||
| 	kontaktnicek_pdf = models.FileField('kontaktníček pdf', upload_to=generate_filename_kontaktnicek, blank=True,) | ||||
| 	kontaktnicek_vcf = models.FileField('kontaktníček vcf', upload_to=generate_filename_kontaktnicek, blank=True, null=True) | ||||
| 	kontaktnicek_pdf = models.FileField('kontaktníček pdf', upload_to=generate_filename_kontaktnicek, blank=True, null=True) | ||||
| 
 | ||||
| 	def __str__(self): | ||||
| 		return '{} ({})'.format(self.misto, self.datum_zacatku) | ||||
|  |  | |||
|  | @ -47,12 +47,12 @@ urlpatterns = [ | |||
| 				path( | ||||
| 					'kontaktnicek_pdf', | ||||
| 					views.soustredeniKontaktnicekPdfView, | ||||
| 					name='soustredeni_kontaktnicek' | ||||
| 					name='soustredeni_kontaktnicek_pdf' | ||||
| 				), | ||||
| 				path( | ||||
| 					'kontaktnicek_vcf', | ||||
| 					views.soustredeniKontaktnicekVcfView, | ||||
| 					name='soustredeni_kontaktnicek' | ||||
| 					name='soustredeni_kontaktnicek_vcf' | ||||
| 
				
					
						ledoian
						commented  Stejný  Stejný `name` pro obě URL nevypadá moc správně :-( | ||||
| 				), | ||||
| 
				
				zelvuska marked this conversation as resolved
				
			 
				
					
						zelvuska
						commented  Tady je nějaké divné odsazení… Tady je nějaké divné odsazení… | ||||
| 				path( | ||||
| 					'fotogalerie/', | ||||
|  |  | |||
|  | @ -3,6 +3,7 @@ from django.http import HttpResponse | |||
| from django.views import generic | ||||
| from django.contrib.staticfiles.finders import find | ||||
| from django.http import Http404 | ||||
| from django.core.exceptions import PermissionDenied | ||||
| 
 | ||||
| import csv | ||||
| import tempfile | ||||
|  | @ -108,7 +109,7 @@ class SoustredeniAbstraktyView(generic.DetailView): | |||
| 	template_name = 'soustredeni/export_do_abstraktu.html' | ||||
| 	pk_url_kwarg = 'soustredeni' # v url bude <int:soustredeni> místo defaultně požadovaného <int:pk> | ||||
| 
 | ||||
| 
				
				zelvuska marked this conversation as resolved
				
					
					
						Outdated
					
				
			 
				
					
						ledoian
						commented  Diakritika maybe? Diakritika maybe? | ||||
| # kontaktnicek | ||||
| # Kontaktníčky | ||||
| def soustredeniKontaktnicekPdfView(request, soustredeni): | ||||
| 	return soustredeniKontaktnicekView(request, soustredeni, "pdf") | ||||
| 
 | ||||
|  | @ -117,22 +118,24 @@ def soustredeniKontaktnicekVcfView(request, soustredeni): | |||
| 
 | ||||
| def soustredeniKontaktnicekView(request, soustredeni, typ): | ||||
| 	soustredeni = get_object_or_404(Soustredeni, id=soustredeni) | ||||
| 
 | ||||
| 	if (not request.user in [u.osoba.user for u in soustredeni.ucastnici.all()]): | ||||
| 		if not request.user.je_org: # nebyl jsi tam, nebo nejsi org | ||||
| 			raise Http404() | ||||
| 	if not soustredeni.kontaktnicek_pdf and typ == "pdf": # není k dispozici | ||||
| 		raise Http404() | ||||
| 	elif not soustredeni.kontaktnicek_vcf and typ == "vcf": # není k dispozici | ||||
|  	# nebyl jsi tam, nebo nejsi org | ||||
| 	if (not request.user in [u.osoba.user for u in soustredeni.ucastnici.all()]) and not request.user.je_org: | ||||
| 
				
					
						zelvuska
						commented  Asi bych napsal popisnější hlášku, vynechal bych tu část z orgem, a naopak bych tam připsal, že nejsi přihlášený (to se stane asi častěji než že bych se dostal na špatný kontaktníček). Také si nejsem jistý, jestli HttpResponse je to správné. (Jakože vrátí to něco, co moc nevypadá jako M&M stránky, ne?) Asi bych napsal popisnější hlášku, vynechal bych tu část z orgem, a naopak bych tam připsal, že nejsi přihlášený (to se stane asi častěji než že bych se dostal na špatný kontaktníček).
Také si nejsem jistý, jestli HttpResponse je to správné. (Jakože vrátí to něco, co moc nevypadá jako M&M stránky, ne?) 
				
					
						ledoian
						commented  Uh, je to ekvivalentní s  Uh, je to ekvivalentní s `if request.user not in […] and not request.user.is_org:`? | ||||
| 		raise PermissionDenied() | ||||
| 	# není k dispozici | ||||
| 	if (not soustredeni.kontaktnicek_pdf and typ == "pdf") or (not soustredeni.kontaktnicek_vcf and typ == "vcf"): | ||||
| 
				
					
						ledoian
						commented  404 není správný návratový kód (kontaktníček zjevně existuje, ale nemám k němu přístup), má být 403 ( 404 není správný návratový kód (kontaktníček zjevně existuje, ale nemám k němu přístup), má být 403 (`PermissionDenied` se myslím jmenuje ta Djangová výjimka). | ||||
| 		raise Http404() | ||||
| 
				
					
						ledoian
						commented  Poněkud se mi nelíbí místní duplikace kódu, když se tam mění dohromady tři věci (až přidáme další, tak někde něco zapomeneme přepsat a bude to bug…). Co třeba: Poněkud se mi nelíbí místní duplikace kódu, když se tam mění dohromady tři věci (až přidáme další, tak někde něco zapomeneme přepsat a bude to bug…). Co třeba:
```python3
# typ -> (field, mime_type, otevreni)
kontaktnicky = {
|   'pdf': (soustredeni.kontaktnicek_pdf, 'applcation/pdf', 'rb'),
|   'vcf': (soustredeni.kontaktnicek_vcf, 'text/vcard', 'r'), # vcf je texťák, nevím, jestli je potřeba ho otevítat binárně.
}
try:
	field, mime, otevreni = kontaktnicky[typ]
except KeyError as e:
	raise ValueError("Neznámý typ kontaktníčku") from e
if not field:
|   ...
with open(field.path, otevreni) as soubor:
|   return HttpResponse(soubor.read(), content_type=mime)
``` 
				
					
						ledoian
						commented  Uh, ten  Uh, ten `if not field: ...` byl náznak toho, že není potřeba výčet typů v `if (not soustredeni.kontaktnicek_pdf and typ == "pdf") or (not soustredeni.kontaktnicek_vcf and typ == "vcf"):`, sorry za neexplicitnost. | ||||
| 	 | ||||
| 	if typ == "pdf": | ||||
| 		with open(soustredeni.kontaktnicek_pdf.path, 'rb') as pdf: | ||||
| 			response = HttpResponse(pdf.read(), content_type='application/pdf') | ||||
| 		return response | ||||
| 	elif typ == "vcf": | ||||
| 		with open(soustredeni.kontaktnicek_vcf.path, 'rb') as vcf: | ||||
| 			response = HttpResponse(vcf.read(), content_type='text/vcard') | ||||
| 		return response | ||||
| 	else: | ||||
| 		raise ValueError("Nepodporovaný typ kontaktníčku") | ||||
| 	kontaktnicky = { | ||||
| 		'pdf': (soustredeni.kontaktnicek_pdf, 'applcation/pdf', 'rb'), | ||||
| 		'vcf': (soustredeni.kontaktnicek_vcf, 'text/vcard', 'rb'), # vcf je texťák, nevím, jestli je potřeba ho otevítat binárně. | ||||
| 	} | ||||
| 	 | ||||
| 
				
					
						ledoian
						commented  Možno rovnou: Možno rovnou:
```python3
with open(…) as whatever:
	return HttpResponse(…)
``` | ||||
| 	try: | ||||
| 		field, mime, otevreni = kontaktnicky[typ] | ||||
| 	except KeyError as e: | ||||
| 		raise ValueError("Neznámý typ kontaktníčku") from e | ||||
| 	 | ||||
| 	with open(field.path, otevreni) as kontaktnicek: | ||||
| 		response = HttpResponse(kontaktnicek.read(), content_type=mime) | ||||
| 		response['Content-Disposition'] = 'attachment; filename="kontaktnicek.{}"'.format(typ) | ||||
| 	return response | ||||
|  |  | |||
Tohle nezachvá příponu, což je spíš bug – když se dívám na soubory na Gimlim, tak chci vědět, co jsou zač.