You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

408 lines
22 KiB

4 years ago
  1. from datetime import date
  2. from django.utils.http import urlencode
  3. from django.db import models
  4. from django.utils.html import format_html
  5. import urllib
  6. from django.utils.safestring import mark_safe
  7. from .settings import ACCOUNTS
  8. EMAIL_STATES = {'NONE': 'noch keine Mail versendet',
  9. 'INF': 'die Benachrichtigung zur Projektabschlussmail wurde versendet',
  10. 'CLOSE': 'die Projektabschlussmail wurde versendet',
  11. 'END': 'alle automatischen Mails, auch surveyMail, wurden versendet'}
  12. class Volunteer(models.Model):
  13. realname = models.CharField(max_length=200, null=True, verbose_name="Realname",
  14. help_text="Bitte gib deinen Vornamen und deinen Nachnamen ein.", default='oi')
  15. email = models.EmailField(max_length=200, null=True, verbose_name='E-Mail-Adresse',
  16. help_text=format_html('Bitte gib deine E-Mail-Adresse ein, damit dich<br>Wikimedia Deutschland bei Rückfragen oder für<br>die Zusage kontaktieren kann.'))
  17. # the following Fields are not supposed to be edited by users
  18. granted = models.BooleanField(null=True, verbose_name='bewilligt')
  19. granted_date = models.DateField(null=True, verbose_name='bewilligt am')
  20. survey_mail_date = models.DateField(verbose_name='Umfragemail wurde verschickt am', null=True, blank=True)
  21. mail_state = models.CharField(max_length=6, choices=EMAIL_STATES.items(), default='NONE')
  22. survey_mail_send = models.BooleanField(default=False, verbose_name='Keine Umfragemail schicken')
  23. @classmethod
  24. def set_granted(cl, key, b):
  25. obj = cl.objects.get(pk=key)
  26. obj.granted = b
  27. obj.granted_date = date.today()
  28. obj.save()
  29. class Meta:
  30. abstract = True
  31. class Extern(Volunteer):
  32. ''' abstract basis class for all data entered by extern volunteers '''
  33. username = models.CharField(max_length=200, null=True, verbose_name='Benutzer_innenname',
  34. help_text=format_html("Wikimedia Benutzer_innenname"))
  35. # the following Fields are not supposed to be edited by users
  36. service_id = models.CharField(max_length=15, null=True, blank=True)
  37. def save(self,*args,**kwargs):
  38. # we don't call save with args/kwargs to avoid UNIQUE CONSTRAINT errors
  39. # but maybe there is a better solution?
  40. super().save()
  41. self.service_id = type(self).__name__ + str(self.pk)
  42. super().save()
  43. class Meta:
  44. abstract = True
  45. class ConcreteExtern(Extern):
  46. ''' needed because we can't initiate abstract base classes in the view'''
  47. pass
  48. class Account(models.Model):
  49. code = models.CharField('Kostenstelle', max_length=5, default="DEF",
  50. null=False, primary_key = True)
  51. description = models.CharField('Beschreibung', max_length=60, default='NO DESCRIPTION')
  52. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name="interne Anmerkungen")
  53. def __str__(self):
  54. return f"{self.code} {self.description}"
  55. class Project(Volunteer):
  56. end_mail_send = models.BooleanField(default=False, verbose_name='Keine Projektabschlussmail schicken')
  57. name = models.CharField(max_length=200, verbose_name='Name des Projekts')
  58. description = models.CharField(max_length=500, verbose_name="Kurzbeschreibung", null=True)
  59. start = models.DateField('Startdatum', null=True)
  60. end = models.DateField('Erwartetes Projektende', null=True)
  61. otrs = models.URLField(max_length=300, null=True, verbose_name='OTRS-Link')
  62. plan = models.URLField(max_length=2000, null=True, blank=True, verbose_name="Link zum Förderplan")
  63. page = models.URLField(max_length=2000, null=True, blank=True, verbose_name="Link zur Projektseite")
  64. urls = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Weitere Links")
  65. group = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Mitorganisierende")
  66. location = models.CharField(max_length=2000, null=True, blank=True, verbose_name="Ort/Adresse/Location")
  67. participants_estimated = models.IntegerField(blank=True, null=True, verbose_name='Teilnehmende angefragt')
  68. participants_real = models.IntegerField(blank=True, null=True, verbose_name='Teilnehmende ausgezählt')
  69. insurance = models.BooleanField(default=False, verbose_name='Haftpflichtversicherung')
  70. insurance_technic = models.BooleanField(default=False, verbose_name='Technikversicherung Ausland')
  71. support = models.CharField(max_length=300, blank=True, null=True, verbose_name='Betreuungsperson und Vertretung')
  72. cost = models.IntegerField(blank=True, null=True, verbose_name='Kosten')
  73. account = models.ForeignKey('Account', on_delete=models.CASCADE, null=True, to_field='code', db_constraint = False)
  74. granted_from = models.CharField(max_length=100,null=True,verbose_name='Bewilligt von')
  75. notes = models.TextField(max_length=1000,null=True,blank=True,verbose_name='Anmerkungen')
  76. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name="interne Anmerkungen")
  77. # the following Fields are not supposed to be edited by users
  78. pid = models.CharField(max_length=15, null=True, blank=True)
  79. status = models.CharField(max_length=3,choices=(('RUN', 'läuft'),('END','beendet'),('NOT','nicht stattgefunden')),default='RUN')
  80. finance_id = models.CharField(max_length=15, null= True, blank=True)
  81. project_of_year = models.IntegerField(default=0)
  82. end_quartal = models.CharField(max_length=15, null=True, blank=True, verbose_name="Quartal Projekt Ende")
  83. def some_otrs(self):
  84. """ This returns a HTML anchor (hyperlink) to somewhere """
  85. return u'<a href="%s">Link</a>' % self.otrs
  86. some_otrs.allow_tags = True
  87. def save(self,*args,**kwargs):
  88. '''we generate the autogenerated fields here'''
  89. # we don't call save with args/kwargs to avoid UNIQUE CONSTRAINT errors
  90. # but maybe there is a better solution?
  91. preotrs = self.otrs
  92. #postotrs = ''
  93. #for n in range(len(preotrs)):
  94. # if preotrs[n] == ';':
  95. # postotrs += '\;'
  96. # else:
  97. # postotrs += preotrs[n]
  98. #print(self.otrs)
  99. #print(preotrs)
  100. #print(postotrs)
  101. postotrs = urllib.parse.quote(preotrs, safe=':;/=?&')
  102. self.otrs = mark_safe(urllib.parse.unquote(postotrs))
  103. startyear_tmp = 'NONE'
  104. if self.pid:
  105. print('self pid last four', self.pid[:4], self.start.year)
  106. if int(self.pid[:4]) != int(self.start.year):
  107. startyear_tmp = self.start.year
  108. print('the startyear_tmp is as follows ', startyear_tmp)
  109. super().save()
  110. if startyear_tmp == 'NONE':
  111. self.pid = str(self.start.year) + '-' + str(self.account.code) + str(self.pk).zfill(3)
  112. # self.pid = str(self.account.code) + str(self.pk).zfill(3)
  113. # generation of field quartals
  114. if self.end.month in [1, 2, 3]:
  115. self.end_quartal = 'Q1'
  116. if self.end.month in [4, 5, 6]:
  117. self.end_quartal = 'Q2'
  118. if self.end.month in [7, 8, 9]:
  119. self.end_quartal = 'Q3'
  120. if self.end.month in [10, 11, 12]:
  121. self.end_quartal = 'Q4'
  122. # generation of pid and financeID
  123. # project of year is true if entry gets updated with changes.. but year can change!!!!!!!!
  124. if self.project_of_year:
  125. print('oi oi oi oi oi')
  126. print(self.pid)
  127. # project of year is false if entry gets saved as new
  128. if not self.project_of_year or startyear_tmp != 'NONE':
  129. print('AAA')
  130. print('self projekt of year', self.project_of_year, self.start.year)
  131. # we need to determine if this is a new year with its first new project...
  132. year = self.start.year
  133. #print(year)
  134. projects = Project.objects.filter(start__year=year)
  135. print('projects after filter of startyear of project',projects)
  136. if not projects:
  137. #print('BBB')
  138. self.project_of_year = 1
  139. self.pid = str(self.start.year) + '-' + str(self.account.code) + str(self.project_of_year).zfill(3)
  140. else:
  141. #print('CCC')
  142. # get the project of year number of latest entry
  143. projects = projects.order_by("-project_of_year")[0]
  144. # add one to value of latest entry
  145. self.project_of_year = int(projects.project_of_year) + 1
  146. self.pid = str(self.start.year) + '-' + str(self.account.code) + str(self.project_of_year).zfill(3)
  147. if str(self.account.code) == '21111':
  148. self.finance_id = str(self.account.code) + str(self.project_of_year).zfill(3)
  149. else:
  150. self.finance_id = str(self.account.code)
  151. super().save()
  152. def __str__(self):
  153. return f"{self.pid} {self.name}"
  154. class Intern(Volunteer):
  155. '''abstract base class for data entry from /intern (except Project)'''
  156. request_url = models.URLField(max_length=2000, verbose_name='Antrag (URL)')
  157. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name='interne Anmerkungen')
  158. class Meta:
  159. abstract = True
  160. class ConcreteVolunteer(Volunteer):
  161. ''' needed because we can't initiate abstract base classes in the view'''
  162. pass
  163. class HonoraryCertificate(Intern):
  164. ''' this class is also used for accreditations '''
  165. project = models.ForeignKey(Project, null=True, blank=True, on_delete=models.SET_NULL)
  166. def __str__(self):
  167. return "Certificate for " + self.realname
  168. TRANSPORT_CHOICES = {'BAHN': 'Bahn',
  169. 'NONE': 'Keine Fahrtkosten',
  170. 'OTHER': 'Sonstiges (mit Begründung)'}
  171. PAYEDBY_CHOICES = {'WMDE': 'WMDE',
  172. 'REQU': 'Antragstellender Mensch'}
  173. HOTEL_CHOICES = {'TRUE': format_html('Hotelzimmer benötigt'),
  174. 'FALSE': format_html('Kein Hotelzimmer benötigt')
  175. }
  176. from django.contrib.contenttypes.models import ContentType
  177. class Travel(Extern):
  178. # project variable is now null true and blank true, which means it can be saved without project id to be later on filled out by admins
  179. project = models.ForeignKey(Project, on_delete=models.CASCADE, null=True, blank=True)
  180. project_name = models.CharField(max_length=50, null=True, blank=True, verbose_name='Projektname:')
  181. transport = models.CharField(max_length=5, choices=TRANSPORT_CHOICES.items(), default='BAHN', verbose_name='Transportmittel:')
  182. other_transport = models.CharField(max_length=200, null=True, blank=True, verbose_name='Sonstige Transportmittel (mit Begründung)')
  183. travelcost = models.CharField(max_length=10, default="0", verbose_name='Fahrtkosten')
  184. checkin = models.DateField(blank=True, null=True, verbose_name='Anreise')
  185. checkout = models.DateField(blank=True, null=True, verbose_name='Abreise')
  186. payed_for_hotel_by = models.CharField(max_length=4, choices=PAYEDBY_CHOICES.items(), blank=True, null=True, verbose_name='Kostenauslage Hotel durch')
  187. payed_for_travel_by = models.CharField(max_length=4, choices=PAYEDBY_CHOICES.items(), blank=True, null=True, verbose_name='Kostenauslage Fahrt durch')
  188. hotel = models.CharField(max_length=10, choices=HOTEL_CHOICES.items(), verbose_name='Hotelzimmer benötigt:')
  189. notes = models.TextField(max_length=1000, blank=True, verbose_name='Anmerkungen')
  190. request_url = models.URLField(max_length=2000, verbose_name='Antrag (URL)')
  191. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name='interne Anmerkungen')
  192. project_end = models.DateField(blank=True, null=True, verbose_name='Projektende')
  193. # use content type model to get the end date for the project foreign key
  194. project_end_quartal = models.CharField(max_length=15, null=True, blank=True, verbose_name="Quartal Projekt Ende")
  195. from django.db.models.signals import pre_save
  196. from django.dispatch import receiver
  197. @receiver(pre_save, sender=Travel, dispatch_uid="get_project_end")
  198. def getProjectEnd(sender, instance, **kwargs):
  199. #instance.project_end = instance.project.end
  200. if instance.project:
  201. instance.project_end = instance.project.end
  202. instance.project_end_quartal = instance.project.end_quartal
  203. # using pre save instead
  204. # def save(self,*args,**kwargs):
  205. # '''we generate the autogenerated fields here'''
  206. # # we don't call save with args/kwargs to avoid UNIQUE CONSTRAINT errors
  207. # # but maybe there is a better solution?
  208. # intern_notes
  209. # project_end = self.checkout
  210. # super(Travel, self).save(*args,**kwargs)
  211. #abstract base class for Library and IFG
  212. class Grant(Extern):
  213. cost = models.CharField(max_length=10, verbose_name='Kosten',
  214. help_text="Bitte gib die ungefähr zu erwartenden Kosten in Euro an.")
  215. notes = models.TextField(max_length=1000, blank=True, verbose_name='Anmerkungen',
  216. help_text="Bitte gib an wofür Du das Stipendium verwenden willst.")
  217. class Meta:
  218. abstract = True
  219. TYPE_CHOICES = {'BIB': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#Bibliotheksstipendium" target="_blank" rel="noopener">Bibliotheksstipendium</a>'),
  220. 'ELIT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#eLiteraturstipendium" target="_blank" rel="noopener">eLiteraturstipendium</a>'),
  221. 'MAIL': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#E-Mail-Adressen" target="_blank" rel="noopener">E-Mail-Adresse</a>'),
  222. 'IFG': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Gebührenerstattungen_für_Behördenanfragen" target="_blank" rel="noopener">Kostenübernahme IFG-Anfrage</a>'),
  223. 'LIT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Zugang_zu_Fachliteratur#Literaturstipendium" target="_blank" rel="noopener">Literaturstipendium</a>'),
  224. 'LIST': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#Mailinglisten" target="_blank" rel="noopener">Mailingliste</a>'),
  225. 'TRAV': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:F%C3%B6rderung/Reisekostenerstattungen" target="_blank" rel="noopener">Reisekosten</a>'),
  226. 'SOFT': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/Software-Stipendien" target="_blank" rel="noopener">Softwarestipendium</a>'),
  227. 'VIS': format_html('<a href="https://de.wikipedia.org/wiki/Wikipedia:Förderung/E-Mail-Adressen_und_Visitenkarten#Visitenkarten" target="_blank" rel="noopener">Visitenkarten</a>'),
  228. }
  229. # same model is used for Library, ELitStip and Software!
  230. class Library(Grant):
  231. type = models.CharField(
  232. max_length=4,
  233. choices=TYPE_CHOICES.items(), #attention: actually only BIB, ELIT, SOFT should be used here
  234. default='BIB',
  235. )
  236. library = models.CharField(max_length=200)
  237. duration = models.CharField(max_length=100, verbose_name="Dauer")
  238. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name="interne Anmerkungen")
  239. def __str__(self):
  240. return self.library
  241. SELFBUY_CHOICES = {'TRUE': format_html('Ich möchte das Werk selbst kaufen und per Kostenerstattung bei Wikimedia Deutschland abrechnen.'),
  242. 'FALSE': format_html('Ich möchte, dass Wikimedia Deutschland das Werk für mich kauft'),
  243. }
  244. class Literature(Grant):
  245. info = models.CharField(max_length=500, verbose_name='Informationen zum Werk',
  246. help_text=format_html("Bitte gib alle Informationen zum benötigten Werk an,<br>\
  247. die eine eindeutige Identifizierung ermöglichen (Autor, Titel, Verlag, ISBN, ...)"))
  248. source = models.CharField(max_length=200, verbose_name='Bezugsquelle',
  249. help_text="Bitte gib an, wo du das Werk kaufen möchtest.")
  250. selfbuy = models.CharField( max_length=10, verbose_name='Selbstkauf?', choices=SELFBUY_CHOICES.items(), default='TRUE')
  251. selfbuy_give_data = models.BooleanField(verbose_name=format_html('Datenweitergabe erlauben'), help_text=format_html('Ich stimme der Weitergabe meiner Daten (Name, Postadresse) an den von mir angegebenen Anbieter/Dienstleister zu.'))
  252. selfbuy_data = models.TextField(max_length=1000, verbose_name='Persönliche Daten sowie Adresse', default='',\
  253. help_text=format_html("Bitte gib hier alle persönlichen Daten an, die wir benötigen, um das Werk<br>\
  254. für dich zu kaufen und es dir anschließend zu schicken (z.B. Vorname Nachname, Anschrift, <br>\
  255. Telefonnummer, E-Mail-Adresse usw.). Trenne die einzelnen Angaben durch Zeilenumbrüche."))
  256. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name="interne Anmerkungen")
  257. class IFG(Grant):
  258. url = models.URLField(max_length=2000, verbose_name="URL",
  259. help_text="Bitte gib den Link zu deiner Anfrage bei Frag den Staat an.")
  260. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name="interne Anmerkungen")
  261. def __str__(self):
  262. return "IFG-Anfrage von " + self.realname
  263. DOMAIN_CHOICES = {'PEDIA': '@wikipedia.de',
  264. 'BOOKS': '@wikibooks.de',
  265. 'QUOTE': '@wikiquote.de',
  266. 'SOURCE': '@wikisource.de',
  267. 'VERSITY': '@wikiversity.de',}
  268. class Domain(Extern):
  269. domain = models.CharField(max_length=10,
  270. choices=DOMAIN_CHOICES.items(),
  271. default='PEDIA')
  272. class Meta:
  273. abstract = True
  274. MAIL_CHOICES = {'REALNAME': 'Vorname.Nachname',
  275. 'USERNAME': 'Username',
  276. 'OTHER': 'Sonstiges:'}
  277. ADULT_CHOICES = {'TRUE': format_html('Ich bin volljährig.'),
  278. 'FALSE': format_html('Ich bin noch nicht volljährig.')
  279. }
  280. class Email(Domain):
  281. address = models.CharField(max_length=50,
  282. choices=MAIL_CHOICES.items(),
  283. default='USERNAME', verbose_name='Adressbestandteil',
  284. help_text=format_html("Bitte gib hier den gewünschten Adressbestandteil an,<br>der sich vor der Domain befinden soll."))
  285. other = models.CharField(max_length=50,blank=True,null=True, verbose_name="Sonstiges")
  286. adult = models.CharField( max_length=10, verbose_name='Volljährigkeit', choices=ADULT_CHOICES.items(), default='FALSE')
  287. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name="interne Anmerkungen")
  288. class List(Domain):
  289. address = models.CharField(max_length=50, default='NO_ADDRESS',
  290. verbose_name="Adressbestandteil für Projektmailingliste",
  291. help_text=format_html("Bitte gib hier den gewünschten Adressbestandteil an,<br>der sich vor der Domain befinden soll."))
  292. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name="interne Anmerkungen")
  293. PROJECT_CHOICE = {'PEDIA': 'Wikipedia',
  294. 'SOURCE': 'Wikisource',
  295. 'BOOKS': 'Wikibooks',
  296. 'QUOTE': 'Wikiquote',
  297. 'VERSITY': 'Wikiversity',
  298. 'VOYAGE': 'Wikivoyage',
  299. 'DATA': 'Wikidata',
  300. 'NEWS': 'Wikinews',
  301. 'COMMONS': 'Wikimedia Commons'}
  302. BC_VARIANT = {'PIC': 'mit Bild',
  303. 'NOPIC': 'ohne Bild'}
  304. class BusinessCard(Extern):
  305. project = models.CharField(max_length=20, choices=PROJECT_CHOICE.items(),
  306. default='PEDIA', verbose_name='Wikimedia-Projekt',
  307. help_text='Für welches Wikimedia-Projekt möchtest Du Visitenkarten?')
  308. data = models.TextField(max_length=1000, verbose_name='Persönliche Daten für die Visitenkarten', default='',
  309. help_text=format_html("Bitte gib hier alle persönlichen Daten an, und zwar genau so,<br>\
  310. wie sie (auch in der entsprechenden Reihenfolge) auf den Visitenkarten stehen sollen<br>\
  311. (z.B. Vorname Nachname, Benutzer:/Benutzerin:, Benutzer-/-innenname, Anschrift,<br>\
  312. Telefonnummer, E-Mail-Adresse usw.). Trenne die einzelnen Angaben durch Zeilenumbrüche.<br>\
  313. Hinweis: Telefonnummern bilden wir üblicherweise im internationalen Format gemäß<br>\
  314. DIN 5008 ab. Als anzugebende E-Mail-Adresse empfehlen wir dir eine Wikimedia-Projekt-<br>\
  315. Adresse, die du ebenfalls beantragen kannst, sofern du nicht bereits eine besitzt."))
  316. variant = models.CharField(max_length=5, choices=BC_VARIANT.items(),
  317. default='NOPIC', verbose_name='Variante',
  318. help_text=format_html('so sehen die Varianten aus: <a href="https://upload.wikimedia.org/wikipedia/commons/c/cd/Muster_Visitenkarten_WMDE_2018.jpg">\
  319. mit Bild</a> <a href="https://upload.wikimedia.org/wikipedia/commons/d/d3/Muster_Visitenkarte_WMDE.png">ohne Bild</a>' ))
  320. url_of_pic = models.CharField(max_length=200, verbose_name='Url des Bildes', default='', help_text="Bitte gib die Wikimedia-Commons-URL des Bildes an.")
  321. sent_to = models.TextField(max_length=1000, verbose_name='Versandadresse',
  322. default='', help_text="Bitte gib den Namen und die vollständige Adresse ein, an welche die Visitenkarten geschickt werden sollen.")
  323. send_data_to_print = models.BooleanField(default=False, verbose_name=format_html('Datenweitergabe erlauben'), help_text=format_html('Hiermit erlaube ich die Weitergabe meiner Daten (Name, Postadresse) an den von Wikimedia<br> Deutschland ausgewählten Dienstleister (z. B. <a href="wir-machen-druck.de">wir-machen-druck.de</a>) zum Zwecke des direkten <br> Versands der Druckerzeugnisse an mich.'))
  324. intern_notes = models.TextField(max_length=1000, blank=True, verbose_name="interne Anmerkungen")