from ietf.doc.models import * from ietf.person.models import Email from ietf.utils.proxy import TranslatingManager from ietf.name.proxy import * from django.conf import settings import glob, os class InternetDraft(Document): objects = TranslatingManager(dict(filename="name", filename__contains="name__contains", id_document_tag="pk", status=lambda v: ("states__slug", { 1: 'active', 2: 'expired', 3: 'rfc', 4: 'auth-rm', 5: 'repl', 6: 'ietf-rm'}[v], "states__type", "draft"), job_owner="ad", rfc_number=lambda v: ("docalias__name", "rfc%s" % v), cur_state=lambda v: ("states__order", v, 'states__type', 'draft-iesg'), idinternal__primary_flag=None, idinternal__cur_state__state=lambda v: ("states__name", v, 'states__type', 'draft-iesg'), ), always_filter=dict(type="draft")) DAYS_TO_EXPIRE=185 # things from InternetDraft #id_document_tag = models.AutoField(primary_key=True) @property def id_document_tag(self): return self.name # Will only work for some use cases #title = models.CharField(max_length=255, db_column='id_document_name') # same name #id_document_key = models.CharField(max_length=255, editable=False) @property def id_document_key(self): return self.title.upper() #group = models.ForeignKey(Acronym, db_column='group_acronym_id') @property def group(self): from group.proxy import Acronym as AcronymProxy g = super(InternetDraft, self).group return AcronymProxy().from_object(g) if g else None #filename = models.CharField(max_length=255, unique=True) @property def filename(self): return self.name #revision = models.CharField(max_length=2) @property def revision(self): return self.rev #revision_date = models.DateField() @property def revision_date(self): if hasattr(self, "new_revision"): e = self.new_revision else: e = self.latest_event(type="new_revision") return e.time.date() if e else None # helper function def get_file_type_matches_from(self, glob_path): possible_types = [".txt", ".pdf", ".xml", ".ps"] res = [] for m in glob.glob(glob_path): for t in possible_types: if m.endswith(t): res.append(t) return ",".join(res) #file_type = models.CharField(max_length=20) @property def file_type(self): return self.get_file_type_matches_from(os.path.join(settings.INTERNET_DRAFT_PATH, self.name + "-" + self.rev + ".*")) or ".txt" #txt_page_count = models.IntegerField() @property def txt_page_count(self): return self.pages #local_path = models.CharField(max_length=255, blank=True) # unused #start_date = models.DateField() @property def start_date(self): e = NewRevisionDocEvent.objects.filter(doc=self).order_by("time")[:1] return e[0].time.date() if e else None #expiration_date = models.DateField() @property def expiration_date(self): e = self.latest_event(type__in=('expired_document', 'new_revision', "completed_resurrect")) return e.time.date() if e and e.type == "expired_document" else None #abstract = models.TextField() # same name #dunn_sent_date = models.DateField(null=True, blank=True) # unused #extension_date = models.DateField(null=True, blank=True) # unused #status = models.ForeignKey(IDStatus) @property def status(self): s = self.get_state() return IDStatus().from_object(s) if s else None @property def status_id(self): return { 'active': 1, 'repl': 5, 'expired': 2, 'rfc': 3, 'auth-rm': 4, 'ietf-rm': 6 }[self.get_state_slug()] #intended_status = models.ForeignKey(IDIntendedStatus) @property def intended_status(self): return self.intended_std_level #lc_sent_date = models.DateField(null=True, blank=True) @property def lc_sent_date(self): e = self.latest_event(type="sent_last_call") return e.time.date() if e else None #lc_changes = models.CharField(max_length=3) # used in DB, unused in Django code? #lc_expiration_date = models.DateField(null=True, blank=True) @property def lc_expiration_date(self): e = self.latest_event(LastCallDocEvent, type="sent_last_call") return e.expires.date() if e else None #b_sent_date = models.DateField(null=True, blank=True) @property def b_sent_date(self): e = self.latest_event(type="sent_ballot_announcement") return e.time.date() if e else None #b_discussion_date = models.DateField(null=True, blank=True) # unused #b_approve_date = models.DateField(null=True, blank=True) @property def b_approve_date(self): e = self.latest_event(type="iesg_approved") return e.time.date() if e else None #wgreturn_date = models.DateField(null=True, blank=True) # unused #rfc_number = models.IntegerField(null=True, blank=True, db_index=True) @property def rfc_number(self): n = self.canonical_name() return int(n[3:]) if n.startswith("rfc") else None #comments = models.TextField(blank=True) # unused #last_modified_date = models.DateField() @property def last_modified_date(self): return self.time.date() #replaced_by = models.ForeignKey('self', db_column='replaced_by', blank=True, null=True, related_name='replaces_set') @property def replaced_by(self): r = InternetDraft.objects.filter(relateddocument__target__document=self, relateddocument__relationship="replaces") return r[0] if r else None @property def replaced_by_id(self): r = self.replaced_by return r.id_document_tag if r else None #replaces = FKAsOneToOne('replaces', reverse=True) @property def replaces(self): r = self.replaces_set return r[0] if r else None @property def replaces_set(self): return InternetDraft.objects.filter(docalias__relateddocument__source=self, docalias__relateddocument__relationship="replaces") #review_by_rfc_editor = models.BooleanField() @property def review_by_rfc_editor(self): return bool(self.tags.filter(slug='rfc-rev')) #expired_tombstone = models.BooleanField() @property def expired_tombstone(self): return False def calc_process_start_end(self): import datetime start, end = datetime.datetime.min, datetime.datetime.max e = self.latest_event(type="started_iesg_process") if e: start = e.time if self.get_state_slug() == "rfc" and self.name.startswith("draft") and not hasattr(self, "viewing_as_rfc"): previous_process = self.latest_event(type="started_iesg_process", time__lt=e.time) if previous_process: start = previous_process.time end = e.time self._process_start = start self._process_end = end @property def process_start(self): if not hasattr(self, "_process_start"): self.calc_process_start_end() return self._process_start @property def process_end(self): if not hasattr(self, "_process_end"): self.calc_process_start_end() return self._process_end #shepherd = BrokenForeignKey('PersonOrOrgInfo', null=True, blank=True, null_values=(0, )) # same name #idinternal = FKAsOneToOne('idinternal', reverse=True, query=models.Q(rfc_flag = 0)) @property def idinternal(self): # since IDInternal is now merged into the document, we try to # guess here if hasattr(self, "changed_ballot_position"): e = self.changed_ballot_position else: e = self.latest_event(type="changed_ballot_position") return self if e or self.get_state("draft-iesg") else None # reverse relationship @property def authors(self): return IDAuthor.objects.filter(document=self) @property def protowriteup_set(self): from ietf.wgchairs.models import ProtoWriteUpProxy return ProtoWriteUpProxy.objects.filter(doc=self, type="changed_protocol_writeup") # methods from InternetDraft def displayname(self): return self.name def file_tag(self): return "<%s>" % self.filename_with_rev() def filename_with_rev(self): return "%s-%s.txt" % (self.filename, self.revision_display()) def group_acronym(self): return super(Document, self).group.acronym def group_ml_archive(self): return self.group.list_archive def idstate(self): return self.docstate() def revision_display(self): return self.rev def expiration(self): return self.expires.date() def can_expire(self): # Copying the logic from expire-ids-1 without thinking # much about it. if self.review_by_rfc_editor: return False idinternal = self.idinternal if idinternal: cur_state_id = idinternal.cur_state_id # 42 is "AD is Watching"; this matches what's in the # expire-ids-1 perl script. # A better way might be to add a column to the table # saying whether or not a document is prevented from # expiring. if cur_state_id < 42: return False return True def clean_abstract(self): # Cleaning based on what "id-abstracts-text" script does import re a = self.abstract a = re.sub(" *\r\n *", "\n", a) # get rid of DOS line endings a = re.sub(" *\r *", "\n", a) # get rid of MAC line endings a = re.sub("(\n *){3,}", "\n\n", a) # get rid of excessive vertical whitespace a = re.sub("\f[\n ]*[^\n]*\n", "", a) # get rid of page headers # Get rid of 'key words' boilerplate and anything which follows it: # (No way that is part of the abstract...) a = re.sub("(?s)(Conventions [Uu]sed in this [Dd]ocument|Requirements [Ll]anguage)?[\n ]*The key words \"MUST\", \"MUST NOT\",.*$", "", a) # Get rid of status/copyright boilerplate a = re.sub("(?s)\nStatus of [tT]his Memo\n.*$", "", a) # wrap long lines without messing up formatting of Ok paragraphs: while re.match("([^\n]{72,}?) +", a): a = re.sub("([^\n]{72,}?) +([^\n ]*)(\n|$)", "\\1\n\\2 ", a) # Remove leading and trailing whitespace a = a.strip() return a # things from IDInternal #draft = models.ForeignKey(InternetDraft, primary_key=True, unique=True, db_column='id_document_tag') @property def draft(self): return self @property def draft_id(self): return self.name #rfc_flag = models.IntegerField(null=True) @property def rfc_flag(self): return self.get_state_slug() == "rfc" #ballot = models.ForeignKey(BallotInfo, related_name='drafts', db_column="ballot_id") @property def ballot(self): if not self.idinternal: raise BallotInfo.DoesNotExist() return self @property def ballot_id(self): return self.ballot.name #primary_flag = models.IntegerField(blank=True, null=True) @property def primary_flag(self): # left-over from multi-ballot documents which we don't really # support anymore, just pretend we're always primary return True #group_flag = models.IntegerField(blank=True, default=0) # not used anymore, contained the group acronym_id once upon a time (so it wasn't a flag) #token_name = models.CharField(blank=True, max_length=25) @property def token_name(self): return self.ad.name #token_email = models.CharField(blank=True, max_length=255) @property def token_email(self): return self.ad.role_email("ad") #note = models.TextField(blank=True) # same name #status_date = models.DateField(blank=True,null=True) @property def status_date(self): return self.time.date() #email_display = models.CharField(blank=True, max_length=50) # unused #agenda = models.IntegerField(null=True, blank=True) @property def agenda(self): e = self.latest_event(TelechatDocEvent, type="scheduled_for_telechat") return bool(e and e.telechat_date) #cur_state = models.ForeignKey(IDState, db_column='cur_state', related_name='docs') @property def cur_state(self): s = self.get_state("draft-iesg") return IDState().from_object(s) if s else None @property def cur_state_id(self): s = self.get_state("draft-iesg") return s.order if s else None #prev_state = models.ForeignKey(IDState, db_column='prev_state', related_name='docs_prev') @property def prev_state(self): ds = self.history_set.exclude(states=self.get_state("draft-iesg")).order_by('-time')[:1] if ds: s = ds[0].get_state("draft-iesg") if s: return IDState().from_object(s) if ds else None return None #assigned_to = models.CharField(blank=True, max_length=25) # unused #mark_by = models.ForeignKey(IESGLogin, db_column='mark_by', related_name='marked') @property def mark_by(self): e = self.latest_event() from person.proxy import IESGLogin as IESGLoginProxy return IESGLoginProxy().from_object(e.by) if e else None # job_owner = models.ForeignKey(IESGLogin, db_column='job_owner', related_name='documents') @property def job_owner(self): from person.proxy import IESGLogin as IESGLoginProxy return IESGLoginProxy().from_object(self.ad) if self.ad else None #event_date = models.DateField(null=True) @property def event_date(self): e = self.latest_event() return e.time if e else None #area_acronym = models.ForeignKey(Area) @property def area_acronym(self): from group.proxy import Area g = super(InternetDraft, self).group # be careful with group which is proxied if g and g.type_id != "individ": return Area().from_object(g.parent) elif self.ad: # return area for AD try: area = Group.objects.get(role__name="ad", role__person=self.ad, state="active") return Area().from_object(area) except Group.DoesNotExist: return None else: return None #cur_sub_state = BrokenForeignKey(IDSubState, related_name='docs', null=True, blank=True, null_values=(0, -1)) @property def cur_sub_state(self): s = self.tags.filter(slug__in=['extpty', 'need-rev', 'ad-f-up', 'point']) return IDSubState().from_object(s[0]) if s else None @property def cur_sub_state_id(self): s = self.cur_sub_state return s.order if s else None #prev_sub_state = BrokenForeignKey(IDSubState, related_name='docs_prev', null=True, blank=True, null_values=(0, -1)) @property def prev_sub_state(self): ds = self.history_set.all().order_by('-time')[:1] substates = ds[0].tags.filter(slug__in=['extpty', 'need-rev', 'ad-f-up', 'point']) if ds else None return IDSubState().from_object(substates[0]) if substates else None @property def prev_sub_state_id(self): s = self.prev_sub_state return s.order if s else None #returning_item = models.IntegerField(null=True, blank=True) @property def returning_item(self): e = self.latest_event(TelechatDocEvent, type="scheduled_for_telechat") return e.returning_item if e else None #telechat_date = models.DateField(null=True, blank=True) @property def telechat_date(self): e = self.latest_event(TelechatDocEvent, type="scheduled_for_telechat") return e.telechat_date if e else None #via_rfc_editor = models.IntegerField(null=True, blank=True) @property def via_rfc_editor(self): return bool(self.tags.filter(slug='via-rfc')) #state_change_notice_to = models.CharField(blank=True, max_length=255) @property def state_change_notice_to(self): return self.notify #dnp = models.IntegerField(null=True, blank=True) @property def dnp(self): e = self.latest_event(type__in=("iesg_disapproved", "iesg_approved")) return e != None and e.type == "iesg_disapproved" #dnp_date = models.DateField(null=True, blank=True) @property def dnp_date(self): e = self.latest_event(type__in=("iesg_disapproved", "iesg_approved")) return e.time.date() if e != None and e.type == "iesg_disapproved" else None #noproblem = models.IntegerField(null=True, blank=True) @property def noproblem(self): e = self.latest_event(type__in=("iesg_disapproved", "iesg_approved")) return e != None and e.type == "iesg_approved" #resurrect_requested_by = BrokenForeignKey(IESGLogin, db_column='resurrect_requested_by', related_name='docsresurrected', null=True, blank=True) @property def resurrect_requested_by(self): e = self.latest_event(type__in=("requested_resurrect", "completed_resurrect")) from person.proxy import IESGLogin as IESGLoginProxy return IESGLoginProxy().from_object(e.by) if e and e.type == "requested_resurrect" else None #approved_in_minute = models.IntegerField(null=True, blank=True) @property def approved_in_minute(self): return self.latest_event(type="approved_in_minute") def get_absolute_url(self): if self.rfc_flag and self.rfc_number: return "/doc/rfc%d/" % self.rfc_number else: return "/doc/%s/" % self.name def document(self): return self def comments(self): return DocumentComment.objects.filter(doc=self).order_by('-time') def public_comments(self): return self.comments() def ballot_set(self): return [self] def ballot_primary(self): return [self] def ballot_others(self): return [] def docstate(self): s = self.get_state("draft-iesg") if s: return s.name else: return "I-D Exists" # things from BallotInfo #active = models.BooleanField() @property def active(self): # taken from BallotWrapper s = self.get_state("draft-iesg") return self.latest_event(type="sent_ballot_announcement") and s and s.name in ['In Last Call', 'Waiting for Writeup', 'Waiting for AD Go-Ahead', 'IESG Evaluation', 'IESG Evaluation - Defer'] and (self.get_state_slug() in ("rfc", "active")) #an_sent = models.BooleanField() @property def an_sent(self): return bool(self.latest_event(type="iesg_approved")) #an_sent_date = models.DateField(null=True, blank=True) @property def an_sent_date(self): e = self.latest_event(type="iesg_approved") return e.time if e else None #an_sent_by = models.ForeignKey(IESGLogin, db_column='an_sent_by', related_name='ansent', null=True) @property def an_sent_by(self): e = self.latest_event(type="iesg_approved") from person.proxy import IESGLogin as IESGLoginProxy return IESGLoginProxy().from_object(e.by) if e else None #defer = models.BooleanField() @property def defer(self): # we're deferred if we're in the deferred state return self.get_state_slug("draft-iesg") == "defer" #defer_by = models.ForeignKey(IESGLogin, db_column='defer_by', related_name='deferred', null=True) @property def defer_by(self): e = self.latest_event(type="changed_document", desc__startswith="State changed to IESG Evaluation - Defer") from person.proxy import IESGLogin as IESGLoginProxy return IESGLoginProxy().from_object(e.by) if e else None #defer_date = models.DateField(null=True, blank=True) @property def defer_date(self): e = self.latest_event(type="changed_document", desc__startswith="State changed to IESG Evaluation - Defer") return e.time.date() if e else None #approval_text = models.TextField(blank=True) @property def approval_text(self): e = self.latest_event(WriteupDocEvent, type="changed_ballot_approval_text") return e.text if e else "" #last_call_text = models.TextField(blank=True) @property def last_call_text(self): e = self.latest_event(WriteupDocEvent, type="changed_last_call_text") return e.text if e else "" #ballot_writeup = models.TextField(blank=True) @property def ballot_writeup(self): e = self.latest_event(WriteupDocEvent, type="changed_ballot_writeup_text") return e.text if e else "" #ballot_issued = models.IntegerField(null=True, blank=True) @property def ballot_issued(self): return bool(self.latest_event(type="sent_ballot_announcement")) # def remarks(self): # apparently not used # remarks = list(self.discusses.all()) + list(self.comments.all()) # return remarks def active_positions(self): """Returns a list of dicts, with AD and Position tuples""" from person.proxy import IESGLogin as IESGLoginProxy from ietf.doc.utils import active_ballot_positions res = [] for ad, pos in active_ballot_positions(self).iteritems(): res.append(dict(ad=IESGLoginProxy().from_object(ad), pos=Position().from_object(pos) if pos else None)) res.sort(key=lambda x: x["ad"].last_name) return res def needed(self, standardsTrack=True): """Returns text answering the question what does this document need to pass?. The return value is only useful if the document is currently in IESG evaluation.""" tmp = self.active_positions() positions = [x["pos"] for x in tmp if x["pos"]] ads = [x["ad"] for x in tmp] yes = noobj = discuss = recuse = 0 for position in positions: p = position.pos_id if p == "yes": yes += 1 if p == "noobj": noobj += 1 if p == "discuss": discuss += 1 if p == "recuse": recuse += 1 answer = '' if yes < 1: answer += "Needs a YES. " if discuss > 0: if discuss == 1: answer += "Has a DISCUSS. " else: answer += "Has %d DISCUSSes. " % discuss if standardsTrack: # For standards-track, need positions from 2/3 of the # non-recused current IESG. needed = int((len(ads) - recuse) * 2 / 3) else: # Info and experimental only need one position. needed = 1 have = yes + noobj + discuss if have < needed: more = needed - have if more == 1: answer += "Needs %d more position. " % more else: answer += "Needs %d more positions. " % more else: answer += "Has enough positions to pass" if discuss: answer += " once DISCUSSes are resolved" answer += ". " return answer.rstrip() # things from RfcIndex #rfc_number = models.IntegerField(primary_key=True) # already taken care of #title = models.CharField(max_length=250) # same name #authors = models.CharField(max_length=250) # exists already #rfc_published_date = models.DateField() @property def rfc_published_date(self): if hasattr(self, 'published_rfc'): e = self.published_rfc else: e = self.latest_event(type="published_rfc") return e.time.date() if e else datetime.date(1990,1,1) #current_status = models.CharField(max_length=50,null=True) @property def current_status(self): return self.std_level.name #updates = models.CharField(max_length=200,blank=True,null=True) @property def updates(self): return ",".join("RFC%s" % n for n in sorted(d.rfc_number for d in InternetDraft.objects.filter(docalias__relateddocument__source=self, docalias__relateddocument__relationship="updates"))) #updated_by = models.CharField(max_length=200,blank=True,null=True) @property def updated_by(self): if not hasattr(self, "updated_by_list"): self.updated_by_list = [d.rfc_number for d in InternetDraft.objects.filter(relateddocument__target__document=self, relateddocument__relationship="updates")] return ",".join("RFC%s" % n for n in sorted(self.updated_by_list)) #obsoletes = models.CharField(max_length=200,blank=True,null=True) @property def obsoletes(self): return ",".join("RFC%s" % n for n in sorted(d.rfc_number for d in InternetDraft.objects.filter(docalias__relateddocument__source=self, docalias__relateddocument__relationship="obs"))) #obsoleted_by = models.CharField(max_length=200,blank=True,null=True) @property def obsoleted_by(self): if not hasattr(self, "obsoleted_by_list"): self.obsoleted_by_list = [d.rfc_number for d in InternetDraft.objects.filter(relateddocument__target__document=self, relateddocument__relationship="obs")] return ",".join("RFC%s" % n for n in sorted(self.obsoleted_by_list)) #also = models.CharField(max_length=50,blank=True,null=True) @property def also(self): aliases = self.docalias_set.filter(models.Q(name__startswith="bcp") | models.Q(name__startswith="std") | models.Q(name__startswith="bcp")) return aliases[0].name.upper() if aliases else None #draft = models.CharField(max_length=200,null=True) # have to ignore this, it's already implemented #has_errata = models.BooleanField() @property def has_errata(self): return bool(self.tags.filter(slug="errata")) #stream = models.CharField(max_length=15,blank=True,null=True) @property def stream(self): s = super(InternetDraft, self).stream return s.name if s else None #wg = models.CharField(max_length=15,blank=True,null=True) @property def wg(self): return self.group.acronym #file_formats = models.CharField(max_length=20,blank=True,null=True) @property def file_formats(self): return self.get_file_type_matches_from(os.path.join(settings.RFC_PATH, "rfc" + str(self.rfc_number) + ".*")).replace(".", "").replace("txt", "ascii") @property def positions(self): res = [] found = set() for pos in Position.objects.filter(doc=self, type="changed_ballot_position").select_related('ad').order_by("-time", "-id"): if pos.ad not in found: found.add(pos.ad) res.append(pos) class Dummy: def all(self): return self.res d = Dummy() d.res = res return d @property def ipr(self): from ipr.models import IprDraftProxy return IprDraftProxy.objects.filter(doc_alias__document=self.pk) class Meta: proxy = True IDInternal = InternetDraft BallotInfo = InternetDraft RfcIndex = InternetDraft Rfc = InternetDraft class IDAuthor(DocumentAuthor): #document = models.ForeignKey(InternetDraft, db_column='id_document_tag', related_name='authors') # same name #person = models.ForeignKey(PersonOrOrgInfo, db_column='person_or_org_tag') @property def person(self): return self.author.person #author_order = models.IntegerField() @property def author_order(self): return self.order def email(self): return None if self.author.address.startswith("unknown-email") else self.author.address def final_author_order(self): return self.order class Meta: proxy = True class DocumentComment(DocEvent): objects = TranslatingManager(dict(comment_text="desc", date="time" )) BALLOT_DISCUSS = 1 BALLOT_COMMENT = 2 BALLOT_CHOICES = ( (BALLOT_DISCUSS, 'discuss'), (BALLOT_COMMENT, 'comment'), ) #document = models.ForeignKey(IDInternal) @property def document(self): return self.doc #rfc_flag = models.IntegerField(null=True, blank=True) #public_flag = models.BooleanField() #unused #date = models.DateField(db_column='comment_date', default=datetime.date.today) @property def date(self): return self.time.date() #time = models.CharField(db_column='comment_time', max_length=20, default=lambda: datetime.datetime.now().strftime("%H:%M:%S")) #version = models.CharField(blank=True, max_length=3) @property def version(self): e = self.doc.latest_event(NewRevisionDocEvent, type="new_revision", time__lte=self.time) return e.rev if e else "0" #comment_text = models.TextField(blank=True) @property def comment_text(self): return self.desc #created_by = BrokenForeignKey(IESGLogin, db_column='created_by', null=True, null_values=(0, 999)) #result_state = BrokenForeignKey(IDState, db_column='result_state', null=True, related_name="comments_leading_to_state", null_values=(0, 99)) #origin_state = models.ForeignKey(IDState, db_column='origin_state', null=True, related_name="comments_coming_from_state") #ballot = models.IntegerField(null=True, choices=BALLOT_CHOICES) def get_absolute_url(self): return "/doc/%s/" % self.doc.name def get_author(self): return self.by.plain_name() def get_username(self): return unicode(self.by) def get_fullname(self): return self.by.plain_name() def datetime(self): return self.time def doc_id(self): return self.doc_id def doc_name(self): return self.doc.name def __str__(self): return "\"%s...\" by %s" % (self.comment_text[:20], self.get_author()) class Meta: proxy = True class Position(BallotPositionDocEvent): def from_object(self, base): for f in base._meta.fields: if not f.name in ('discuss',): # don't overwrite properties setattr(self, f.name, getattr(base, f.name)) return self #ballot = models.ForeignKey(BallotInfo, related_name='positions') @property def ballot(self): return self.doc # FIXME: doesn't emulate old interface # ad = models.ForeignKey(IESGLogin) # same name #yes = models.IntegerField(db_column='yes_col') @property def yes(self): return self.pos_id == "yes" #noobj = models.IntegerField(db_column='no_col') @property def noobj(self): return self.pos_id == "noobj" #abstain = models.IntegerField() @property def abstain(self): return self.pos_id == "abstain" #approve = models.IntegerField(default=0) # unused #discuss = models.IntegerField() # needs special treatment because of clash with attribute on base class def get_discuss(self): return self.pos_id == "discuss" def set_discuss(self, x): pass discuss = property(get_discuss, set_discuss) #recuse = models.IntegerField() @property def recuse(self): return self.pos_id == "recuse" def __str__(self): return "Position for %s on %s" % ( self.ad, self.ballot ) def abstain_ind(self): if self.recuse: return 'R' if self.abstain: return 'X' else: return ' ' def name(self): return self.pos.name if self.pos else "No Record" class Meta: proxy = True class DraftLikeDocAlias(DocAlias): # this class is mostly useful for the IPR part def __str__(self): return str(unicode(self)) def __unicode__(self): if self.name.startswith("rfc"): return "RFC%04d" % int(self.name[3:]) else: return self.name @property def id_document_tag(self): return self.name @property def title(self): return self.document.title @property def filename(self): return self.name @property def ipr(self): from ipr.models import IprDraftProxy return IprDraftProxy.objects.filter(doc_alias=self.pk) class Meta: proxy = True class ObjectHistoryEntryProxy(DocEvent): #date = models.DateTimeField(_('Date'), auto_now_add=True) @property def date(self): return self.time #comment = models.TextField(_('Comment')) @property def comment(self): return "" #person = models.ForeignKey(PersonOrOrgInfo) @property def person(self): return self.by def get_real_instance(self): return self def describe_change(self): return u"
%s
" % self.desc class Meta: proxy = True class IDStatus(State): def from_object(self, base): for f in base._meta.fields: setattr(self, f.name, getattr(base, f.name)) return self #status_id = models.AutoField(primary_key=True) #status = models.CharField(max_length=25, db_column='status_value') @property def status(self): return self.name def __unicode__(self): return super(self.__class__, self).__unicode__() class Meta: proxy = True class IDState(State): PUBLICATION_REQUESTED = 10 LAST_CALL_REQUESTED = 15 IN_LAST_CALL = 16 WAITING_FOR_WRITEUP = 18 WAITING_FOR_AD_GO_AHEAD = 19 IESG_EVALUATION = 20 IESG_EVALUATION_DEFER = 21 APPROVED_ANNOUNCEMENT_SENT = 30 AD_WATCHING = 42 DEAD = 99 DO_NOT_PUBLISH_STATES = (33, 34) objects = TranslatingManager(dict(pk=lambda v: ("order", v, "type", "draft-iesg"), document_state_id=lambda v: ("order", v, "type", "draft-iesg"), document_state_id__in=lambda v: ("order__in", v, "type", "draft-iesg"))) def from_object(self, base): for f in base._meta.fields: setattr(self, f.name, getattr(base, f.name)) return self #document_state_id = models.AutoField(primary_key=True) @property def document_state_id(self): return self.order #state = models.CharField(max_length=50, db_column='document_state_val') @property def state(self): return self.name #equiv_group_flag = models.IntegerField(null=True, blank=True) # unused #description = models.TextField(blank=True, db_column='document_desc') @property def description(self): return self.desc @property def nextstate(self): # simulate related queryset return IDState.objects.filter(pk__in=[x.pk for x in self.next_states]) @property def next_state(self): # simulate IDNextState return self def __str__(self): return self.state @staticmethod def choices(): return [(state.pk, state.name) for state in IDState.objects.all()] class Meta: proxy = True