def complaintPeriod(self): normalized_end = calculate_normalized_date(self.tenderPeriod.endDate, self) return Period( dict(startDate=self.tenderPeriod.startDate, endDate=calculate_business_date(normalized_end, -COMPLAINT_SUBMIT_TIME, self)))
def complaintPeriod(self): if self.tenderPeriod.startDate < COMPLAINT_OLD_SUBMIT_TIME_BEFORE: return Period( dict(startDate=self.tenderPeriod.startDate, endDate=calculate_business_date( self.tenderPeriod.endDate, -COMPLAINT_OLD_SUBMIT_TIME, self))) else: normalized_end = calculate_normalized_date( self.tenderPeriod.endDate, self) return Period( dict(startDate=self.tenderPeriod.startDate, endDate=calculate_business_date(normalized_end, -COMPLAINT_SUBMIT_TIME, self, True)))
def patch(self): """Update of award Example request to change the award: .. sourcecode:: http PATCH /tenders/4879d3f8ee2443169b5fbbc9f89fa607/awards/71b6c23ed8944d688e92a31ec8c3f61a HTTP/1.1 Host: example.com Accept: application/json { "data": { "value": { "amount": 600 } } } And here is the response to be expected: .. sourcecode:: http HTTP/1.0 200 OK Content-Type: application/json { "data": { "id": "4879d3f8ee2443169b5fbbc9f89fa607", "date": "2014-10-28T11:44:17.947Z", "status": "active", "suppliers": [ { "id": { "name": "Державне управління справами", "scheme": "https://ns.openprocurement.org/ua/edrpou", "uid": "00037256", "uri": "http://www.dus.gov.ua/" }, "address": { "countryName": "Україна", "postalCode": "01220", "region": "м. Київ", "locality": "м. Київ", "streetAddress": "вул. Банкова, 11, корпус 1" } } ], "value": { "amount": 600, "currency": "UAH", "valueAddedTaxIncluded": true } } } """ tender = self.request.validated['tender'] if tender.status != 'active': self.request.errors.add( 'body', 'data', 'Can\'t update award in current ({}) tender status'.format( tender.status)) self.request.errors.status = 403 return award = self.request.context if not self.validate_lot_cancellation('update'): return award_status = award.status apply_patch(self.request, save=False, src=self.request.context.serialize()) if award.status == "active" and not award.qualified: self.request.errors.add( 'body', 'data', 'Can\'t update award to active status with not qualified') self.request.errors.status = 403 return if award.lotID and \ [aw.lotID for aw in tender.awards if aw.status in['pending', 'active']].count(award.lotID) > 1: self.request.errors.add( 'body', 'lotID', 'Another award is already using this lotID.') self.request.errors.status = 403 return if award_status == 'pending' and award.status == 'active': normalized_end = calculate_normalized_date(get_now(), tender, True) award.complaintPeriod.endDate = calculate_business_date( normalized_end, self.stand_still_delta, tender) tender.contracts.append( type(tender).contracts.model_class({ 'awardID': award.id, 'suppliers': award.suppliers, 'date': get_now(), 'value': award.value, 'items': [i for i in tender.items if i.relatedLot == award.lotID], 'contractID': '{}-{}{}'.format(tender.tenderID, self.server_id, len(tender.contracts) + 1) })) # add_next_award(self.request) elif award_status == 'active' and award.status == 'cancelled' and any( [i.status == 'satisfied' for i in award.complaints]): now = get_now() cancelled_awards = [] for i in tender.awards: if i.lotID != award.lotID: continue if not i.complaintPeriod.endDate or i.complaintPeriod.endDate > now: i.complaintPeriod.endDate = now i.status = 'cancelled' cancelled_awards.append(i.id) for i in tender.contracts: if i.awardID in cancelled_awards: i.status = 'cancelled' elif award_status == 'active' and award.status == 'cancelled': now = get_now() if award.complaintPeriod.endDate > now: award.complaintPeriod.endDate = now for i in tender.contracts: if i.awardID == award.id: i.status = 'cancelled' # add_next_award(self.request) elif award_status == 'pending' and award.status == 'unsuccessful': award.complaintPeriod.endDate = get_now() # add_next_award(self.request) elif award_status == 'unsuccessful' and award.status == 'cancelled' and any( [i.status == 'satisfied' for i in award.complaints]): now = get_now() cancelled_awards = [] for i in tender.awards: if i.lotID != award.lotID: continue if not i.complaintPeriod.endDate or i.complaintPeriod.endDate > now: i.complaintPeriod.endDate = now i.status = 'cancelled' cancelled_awards.append(i.id) for i in tender.contracts: if i.awardID in cancelled_awards: i.status = 'cancelled' elif award_status != award.status: self.request.errors.add( 'body', 'data', 'Can\'t update award in current ({}) status'.format( award_status)) self.request.errors.status = 403 return elif self.request.authenticated_role != 'Administrator' and award_status != 'pending': self.request.errors.add( 'body', 'data', 'Can\'t update award in current ({}) status'.format( award_status)) self.request.errors.status = 403 return if save_tender(self.request): self.LOGGER.info( 'Updated tender award {}'.format(self.request.context.id), extra=context_unpack(self.request, {'MESSAGE_ID': 'tender_award_patch'}, {'TENDER_REV': tender.rev})) return {'data': award.serialize("view")}
def patch(self): """Tender Edit (partial) For example here is how procuring entity can change number of items to be procured and total Value of a tender: .. sourcecode:: http PATCH /tenders/4879d3f8ee2443169b5fbbc9f89fa607 HTTP/1.1 Host: example.com Accept: application/json { "data": { "value": { "amount": 600 }, "itemsToBeProcured": [ { "quantity": 6 } ] } } And here is the response to be expected: .. sourcecode:: http HTTP/1.0 200 OK Content-Type: application/json { "data": { "id": "4879d3f8ee2443169b5fbbc9f89fa607", "tenderID": "UA-64e93250be76435397e8c992ed4214d1", "dateModified": "2014-10-27T08:12:34.956Z", "value": { "amount": 600 }, "itemsToBeProcured": [ { "quantity": 6 } ] } } """ tender = self.context if self.request.authenticated_role != 'Administrator' and tender.status in ['complete', 'unsuccessful', 'cancelled']: self.request.errors.add('body', 'data', 'Can\'t update tender in current ({}) status'.format(tender.status)) self.request.errors.status = 403 return data = self.request.validated['data'] if self.request.authenticated_role == 'tender_owner' and 'status' in data and data['status'] not in ['active.pre-qualification.stand-still', tender.status]: self.request.errors.add('body', 'data', 'Can\'t update tender status') self.request.errors.status = 403 return if self.request.authenticated_role == 'tender_owner' and self.request.validated['tender_status'] == 'active.tendering': if 'tenderPeriod' in data and 'endDate' in data['tenderPeriod']: self.request.validated['tender'].tenderPeriod.import_data(data['tenderPeriod']) if calculate_business_date(get_now(), TENDERING_EXTRA_PERIOD, self.request.validated['tender']) > self.request.validated['tender'].tenderPeriod.endDate: self.request.errors.add('body', 'data', 'tenderPeriod should be extended by {0.days} days'.format(TENDERING_EXTRA_PERIOD)) self.request.errors.status = 403 return self.request.validated['tender'].initialize() self.request.validated['data']["enquiryPeriod"] = self.request.validated['tender'].enquiryPeriod.serialize() apply_patch(self.request, save=False, src=self.request.validated['tender_src']) if self.request.authenticated_role == 'chronograph': check_status(self.request) elif self.request.authenticated_role == 'tender_owner' and tender.status == 'active.tendering': tender.invalidate_bids_data() elif self.request.authenticated_role == 'tender_owner' and self.request.validated['tender_status'] == 'active.pre-qualification' and tender.status == "active.pre-qualification.stand-still": if all_bids_are_reviewed(self.request): normalized_date = calculate_normalized_date(get_now(), tender, True) tender.qualificationPeriod.endDate = calculate_business_date(normalized_date, COMPLAINT_STAND_STILL, self.request.validated['tender']) tender.check_auction_time() else: self.request.errors.add('body', 'data', 'Can\'t switch to \'active.pre-qualification.stand-still\' while not all bids are qualified') self.request.errors.status = 403 return save_tender(self.request) self.LOGGER.info('Updated tender {}'.format(tender.id), extra=context_unpack(self.request, {'MESSAGE_ID': 'tender_patch'})) return {'data': tender.serialize(tender.status)}
def patch(self): """Update of award Example request to change the award: .. sourcecode:: http PATCH /tenders/4879d3f8ee2443169b5fbbc9f89fa607/awards/71b6c23ed8944d688e92a31ec8c3f61a HTTP/1.1 Host: example.com Accept: application/json { "data": { "value": { "amount": 600 } } } And here is the response to be expected: .. sourcecode:: http HTTP/1.0 200 OK Content-Type: application/json { "data": { "id": "4879d3f8ee2443169b5fbbc9f89fa607", "date": "2014-10-28T11:44:17.947Z", "status": "active", "suppliers": [ { "id": { "name": "Державне управління справами", "scheme": "https://ns.openprocurement.org/ua/edrpou", "uid": "00037256", "uri": "http://www.dus.gov.ua/" }, "address": { "countryName": "Україна", "postalCode": "01220", "region": "м. Київ", "locality": "м. Київ", "streetAddress": "вул. Банкова, 11, корпус 1" } } ], "value": { "amount": 600, "currency": "UAH", "valueAddedTaxIncluded": true } } } """ tender = self.request.validated['tender'] if tender.status not in ['active.qualification', 'active.awarded']: self.request.errors.add('body', 'data', 'Can\'t update award in current ({}) tender status'.format(tender.status)) self.request.errors.status = 403 return award = self.request.context if any([i.status != 'active' for i in tender.lots if i.id == award.lotID]): self.request.errors.add('body', 'data', 'Can update award only in active lot status') self.request.errors.status = 403 return if any([any([c.status == 'accepted' for c in i.complaints]) for i in tender.awards if i.lotID == award.lotID]): self.request.errors.add('body', 'data', 'Can\'t update award with accepted complaint') self.request.errors.status = 403 return award_status = award.status apply_patch(self.request, save=False, src=self.request.context.serialize()) if award_status == 'pending' and award.status == 'active': normalized_end = calculate_normalized_date(get_now(), tender, True) award.complaintPeriod.endDate = calculate_business_date(normalized_end, STAND_STILL_TIME, tender) tender.contracts.append(type(tender).contracts.model_class({ 'awardID': award.id, 'suppliers': award.suppliers, 'value': award.value, 'items': [i for i in tender.items if i.relatedLot == award.lotID ], 'contractID': '{}-{}{}'.format(tender.tenderID, self.server_id, len(tender.contracts) +1) })) add_next_award(self.request) elif award_status == 'active' and award.status == 'cancelled' and any([i.status == 'satisfied' for i in award.complaints]): now = get_now() cancelled_awards = [] for i in tender.awards: if i.lotID != award.lotID: continue if not i.complaintPeriod.endDate or i.complaintPeriod.endDate > now: i.complaintPeriod.endDate = now i.status = 'cancelled' cancelled_awards.append(i.id) for i in tender.contracts: if i.awardID in cancelled_awards: i.status = 'cancelled' add_next_award(self.request) elif award_status == 'active' and award.status == 'cancelled': now = get_now() if award.complaintPeriod.endDate > now: award.complaintPeriod.endDate = now for i in tender.contracts: if i.awardID == award.id: i.status = 'cancelled' add_next_award(self.request) elif award_status == 'pending' and award.status == 'unsuccessful': normalized_end = calculate_normalized_date(get_now(), tender, True) award.complaintPeriod.endDate = calculate_business_date(normalized_end, STAND_STILL_TIME, tender) add_next_award(self.request) elif award_status == 'unsuccessful' and award.status == 'cancelled' and any([i.status == 'satisfied' for i in award.complaints]): if tender.status == 'active.awarded': tender.status = 'active.qualification' tender.awardPeriod.endDate = None now = get_now() award.complaintPeriod.endDate = now cancelled_awards = [] for i in tender.awards: if i.lotID != award.lotID: continue i.complaintPeriod.endDate = now i.status = 'cancelled' cancelled_awards.append(i.id) for i in tender.contracts: if i.awardID in cancelled_awards: i.status = 'cancelled' add_next_award(self.request) elif self.request.authenticated_role != 'Administrator' and not(award_status == 'pending' and award.status == 'pending'): self.request.errors.add('body', 'data', 'Can\'t update award in current ({}) status'.format(award_status)) self.request.errors.status = 403 return if save_tender(self.request): self.LOGGER.info('Updated tender award {}'.format(self.request.context.id), extra=context_unpack(self.request, {'MESSAGE_ID': 'tender_award_patch'})) return {'data': award.serialize("view")}
def patch(self): """Tender Edit (partial) For example here is how procuring entity can change number of items to be procured and total Value of a tender: .. sourcecode:: http PATCH /tenders/4879d3f8ee2443169b5fbbc9f89fa607 HTTP/1.1 Host: example.com Accept: application/json { "data": { "value": { "amount": 600 }, "itemsToBeProcured": [ { "quantity": 6 } ] } } And here is the response to be expected: .. sourcecode:: http HTTP/1.0 200 OK Content-Type: application/json { "data": { "id": "4879d3f8ee2443169b5fbbc9f89fa607", "tenderID": "UA-64e93250be76435397e8c992ed4214d1", "dateModified": "2014-10-27T08:12:34.956Z", "value": { "amount": 600 }, "itemsToBeProcured": [ { "quantity": 6 } ] } } """ tender = self.context if self.request.authenticated_role != "Administrator" and tender.status in [ "complete", "unsuccessful", "cancelled", ]: self.request.errors.add("body", "data", "Can't update tender in current ({}) status".format(tender.status)) self.request.errors.status = 403 return data = self.request.validated["data"] if ( self.request.authenticated_role == "tender_owner" and "status" in data and data["status"] not in ["active.pre-qualification.stand-still", tender.status] ): self.request.errors.add("body", "data", "Can't update tender status") self.request.errors.status = 403 return if ( self.request.authenticated_role == "tender_owner" and self.request.validated["tender_status"] == "active.tendering" ): if "tenderPeriod" in data and "endDate" in data["tenderPeriod"]: self.request.validated["tender"].tenderPeriod.import_data(data["tenderPeriod"]) if ( calculate_business_date(get_now(), TENDERING_EXTRA_PERIOD, self.request.validated["tender"]) > self.request.validated["tender"].tenderPeriod.endDate ): self.request.errors.add( "body", "data", "tenderPeriod should be extended by {0.days} days".format(TENDERING_EXTRA_PERIOD), ) self.request.errors.status = 403 return self.request.validated["tender"].initialize() self.request.validated["data"]["enquiryPeriod"] = self.request.validated[ "tender" ].enquiryPeriod.serialize() apply_patch(self.request, save=False, src=self.request.validated["tender_src"]) if self.request.authenticated_role == "chronograph": check_status(self.request) elif self.request.authenticated_role == "tender_owner" and tender.status == "active.tendering": tender.invalidate_bids_data() elif ( self.request.authenticated_role == "tender_owner" and self.request.validated["tender_status"] == "active.pre-qualification" and tender.status == "active.pre-qualification.stand-still" ): if any( [ i["status"] in self.request.validated["tender"].block_complaint_status for q in self.request.validated["tender"]["qualifications"] for i in q["complaints"] ] ): self.request.errors.add( "body", "data", "Can't switch to 'active.pre-qualification.stand-still' before resolve all complaints", ) self.request.errors.status = 403 return if all_bids_are_reviewed(self.request): normalized_date = calculate_normalized_date(get_now(), tender, True) tender.qualificationPeriod.endDate = calculate_business_date( normalized_date, COMPLAINT_STAND_STILL, self.request.validated["tender"] ) tender.check_auction_time() else: self.request.errors.add( "body", "data", "Can't switch to 'active.pre-qualification.stand-still' while not all bids are qualified", ) self.request.errors.status = 403 return save_tender(self.request) self.LOGGER.info( "Updated tender {}".format(tender.id), extra=context_unpack(self.request, {"MESSAGE_ID": "tender_patch"}) ) return {"data": tender.serialize(tender.status)}
def patch(self): """Tender Edit (partial) For example here is how procuring entity can change number of items to be procured and total Value of a tender: .. sourcecode:: http PATCH /tenders/4879d3f8ee2443169b5fbbc9f89fa607 HTTP/1.1 Host: example.com Accept: application/json { "data": { "value": { "amount": 600 }, "itemsToBeProcured": [ { "quantity": 6 } ] } } And here is the response to be expected: .. sourcecode:: http HTTP/1.0 200 OK Content-Type: application/json { "data": { "id": "4879d3f8ee2443169b5fbbc9f89fa607", "tenderID": "UA-64e93250be76435397e8c992ed4214d1", "dateModified": "2014-10-27T08:12:34.956Z", "value": { "amount": 600 }, "itemsToBeProcured": [ { "quantity": 6 } ] } } """ tender = self.context if self.request.authenticated_role != 'Administrator' and tender.status in [ 'complete', 'unsuccessful', 'cancelled' ]: self.request.errors.add( 'body', 'data', 'Can\'t update tender in current ({}) status'.format( tender.status)) self.request.errors.status = 403 return data = self.request.validated['data'] if self.request.authenticated_role == 'tender_owner' and 'status' in data and data[ 'status'] not in [ 'active.pre-qualification.stand-still', tender.status ]: self.request.errors.add('body', 'data', 'Can\'t update tender status') self.request.errors.status = 403 return if self.request.authenticated_role == 'tender_owner' and self.request.validated[ 'tender_status'] == 'active.tendering': if 'tenderPeriod' in data and 'endDate' in data['tenderPeriod']: self.request.validated['tender'].tenderPeriod.import_data( data['tenderPeriod']) if calculate_business_date( get_now(), TENDERING_EXTRA_PERIOD, self.request.validated['tender'] ) > self.request.validated['tender'].tenderPeriod.endDate: self.request.errors.add( 'body', 'data', 'tenderPeriod should be extended by {0.days} days'. format(TENDERING_EXTRA_PERIOD)) self.request.errors.status = 403 return self.request.validated['tender'].initialize() self.request.validated['data'][ "enquiryPeriod"] = self.request.validated[ 'tender'].enquiryPeriod.serialize() apply_patch(self.request, save=False, src=self.request.validated['tender_src']) if self.request.authenticated_role == 'chronograph': check_status(self.request) elif self.request.authenticated_role == 'tender_owner' and tender.status == 'active.tendering': tender.invalidate_bids_data() elif self.request.authenticated_role == 'tender_owner' and self.request.validated[ 'tender_status'] == 'active.pre-qualification' and tender.status == "active.pre-qualification.stand-still": if any([ i['status'] in self.request.validated['tender'].block_complaint_status for q in self.request.validated['tender']['qualifications'] for i in q['complaints'] ]): self.request.errors.add( 'body', 'data', 'Can\'t switch to \'active.pre-qualification.stand-still\' before resolve all complaints' ) self.request.errors.status = 403 return if all_bids_are_reviewed(self.request): normalized_date = calculate_normalized_date( get_now(), tender, True) tender.qualificationPeriod.endDate = calculate_business_date( normalized_date, COMPLAINT_STAND_STILL, self.request.validated['tender']) tender.check_auction_time() else: self.request.errors.add( 'body', 'data', 'Can\'t switch to \'active.pre-qualification.stand-still\' while not all bids are qualified' ) self.request.errors.status = 403 return save_tender(self.request) self.LOGGER.info('Updated tender {}'.format(tender.id), extra=context_unpack(self.request, {'MESSAGE_ID': 'tender_patch'})) return {'data': tender.serialize(tender.status)}
def complaintPeriod(self): normalized_end = calculate_normalized_date(self.tenderPeriod.endDate, self) return Period(dict(startDate=self.tenderPeriod.startDate, endDate=calculate_business_date(normalized_end, -COMPLAINT_SUBMIT_TIME, self)))