198 lines
7.2 KiB
Python
198 lines
7.2 KiB
Python
from csv import DictReader
|
|
import defusedxml.ElementTree as ET
|
|
|
|
import requests
|
|
from requests.auth import HTTPBasicAuth
|
|
|
|
|
|
def parse_eda_xml(xml_string):
|
|
contract_et = ET.fromstring(xml_string)
|
|
handler = EDAXMLHandler(contract_et)
|
|
handler.parse()
|
|
return {
|
|
"clin_0001": handler.clins.get("0001"),
|
|
"clin_0003": handler.clins.get("0003"),
|
|
"clin_1001": handler.clins.get("1001"),
|
|
"clin_1003": handler.clins.get("1003"),
|
|
"clin_2001": handler.clins.get("2001"),
|
|
"clin_2003": handler.clins.get("2003"),
|
|
}
|
|
|
|
|
|
class EDAXMLHandler:
|
|
def __init__(self, element_tree):
|
|
self.element_tree = element_tree
|
|
self.clins = {}
|
|
|
|
@property
|
|
def _line_items(self):
|
|
return self.element_tree.findall(".//LineItem[LineItemType='CLIN']/../../..")
|
|
|
|
def parse(self):
|
|
for line_item in self._line_items:
|
|
number_el = line_item.find(".//LineItemBase")
|
|
amount_details = line_item.find(
|
|
".//ItemOtherAmounts[AmountDescription='Not to Exceed Amount (Funding)']/Amount"
|
|
)
|
|
if number_el is not None and amount_details is not None:
|
|
try:
|
|
self.clins[number_el.text] = float(amount_details.text)
|
|
except ValueError:
|
|
continue
|
|
|
|
|
|
class EDAClientBase(object):
|
|
def list_contracts(
|
|
self,
|
|
contract_number=None,
|
|
delivery_order=None,
|
|
cage_code=None,
|
|
duns_number=None,
|
|
):
|
|
"""
|
|
Get a list of all contracts matching the given filters.
|
|
"""
|
|
raise NotImplementedError()
|
|
|
|
def get_contract(self, contract_number, status):
|
|
"""
|
|
Get details for a contract.
|
|
"""
|
|
raise NotImplementedError()
|
|
|
|
|
|
class MockEDAClient(EDAClientBase):
|
|
def __init__(self, *args, **kwargs):
|
|
pass
|
|
|
|
def list_contracts(
|
|
self,
|
|
contract_number=None,
|
|
delivery_order=None,
|
|
cage_code=None,
|
|
duns_number=None,
|
|
):
|
|
return [
|
|
{
|
|
"aco_mod": "01",
|
|
"admin_dodaac": None,
|
|
"cage_code": "1U305",
|
|
"contract_no": "DCA10096D0052",
|
|
"delivery_order": "0084",
|
|
"duns_number": None,
|
|
"issue_date": "20000228",
|
|
"issue_dodaac": None,
|
|
"location": "https://docsrv1.nit.disa.mil:443/eda/enforcer/C0414345.PDF?ver=1.4&loc=Y29udHJhY3RzL29nZGVuL3ZlbmRvci8xOTk4LzA5LzE0L0MwNDE0MzQ1LlBERg==&sourceurl=aHR0cHM6Ly9lZGE0Lm5pdC5kaXNhLm1pbC9wbHMvdXNlci9uZXdfYXBwLkdldF9Eb2M_cFRhYmxlX0lEPTImcFJlY29yZF9LZXk9OEE2ODExNjM2RUY5NkU2M0UwMzQwMDYwQjBCMjgyNkM=&uid=6CFC2B2322E86FD5E054002264936E3C&qid=19344159&signed=G&qdate=20180529194407GMT&token=6xQICrrrfIMciEJSpXmfsAYrToM=",
|
|
"pay_dodaac": None,
|
|
"pco_mod": "02",
|
|
},
|
|
{
|
|
"aco_mod": "01",
|
|
"admin_dodaac": None,
|
|
"cage_code": "1U305",
|
|
"contract_no": "DCA10096D0052",
|
|
"delivery_order": "0084",
|
|
"duns_number": None,
|
|
"issue_date": "20000228",
|
|
"issue_dodaac": None,
|
|
"location": "https://docsrv1.nit.disa.mil:443/eda/enforcer/C0414345.PDF?ver=1.4&loc=Y29udHJhY3RzL29nZGVuL3ZlbmRvci8xOTk4LzA5LzE0L0MwNDE0MzQ1LlBERg==&sourceurl=aHR0cHM6Ly9lZGE0Lm5pdC5kaXNhLm1pbC9wbHMvdXNlci9uZXdfYXBwLkdldF9Eb2M_cFRhYmxlX0lEPTImcFJlY29yZF9LZXk9OEE2ODExNjM2RUY5NkU2M0UwMzQwMDYwQjBCMjgyNkM=&uid=6CFC2B2322E86FD5E054002264936E3C&qid=19344159&signed=G&qdate=20180529194407GMT&token=6xQICrrrfIMciEJSpXmfsAYrToM=",
|
|
"pay_dodaac": None,
|
|
"pco_mod": "02",
|
|
},
|
|
{
|
|
"aco_mod": "01",
|
|
"admin_dodaac": None,
|
|
"cage_code": "1U305",
|
|
"contract_no": "DCA10096D0052",
|
|
"delivery_order": "0084",
|
|
"duns_number": None,
|
|
"issue_date": "20000228",
|
|
"issue_dodaac": None,
|
|
"location": "https://docsrv1.nit.disa.mil:443/eda/enforcer/C0414345.PDF?ver=1.4&loc=Y29udHJhY3RzL29nZGVuL3ZlbmRvci8xOTk4LzA5LzE0L0MwNDE0MzQ1LlBERg==&sourceurl=aHR0cHM6Ly9lZGE0Lm5pdC5kaXNhLm1pbC9wbHMvdXNlci9uZXdfYXBwLkdldF9Eb2M_cFRhYmxlX0lEPTImcFJlY29yZF9LZXk9OEE2ODExNjM2RUY5NkU2M0UwMzQwMDYwQjBCMjgyNkM=&uid=6CFC2B2322E86FD5E054002264936E3C&qid=19344159&signed=G&qdate=20180529194407GMT&token=6xQICrrrfIMciEJSpXmfsAYrToM=",
|
|
"pay_dodaac": None,
|
|
"pco_mod": "02",
|
|
},
|
|
]
|
|
|
|
MOCK_CONTRACT_NUMBER = "DCA10096D0052"
|
|
|
|
def get_contract(self, contract_number, status):
|
|
if contract_number == self.MOCK_CONTRACT_NUMBER and status == "y":
|
|
return {
|
|
"number": "DCA10096D0052",
|
|
"clin_0001": 500,
|
|
"clin_0003": 600,
|
|
"clin_1001": 700,
|
|
"clin_1003": 800,
|
|
"clin_2001": 900,
|
|
"clin_2003": 1000,
|
|
}
|
|
else:
|
|
return None
|
|
|
|
|
|
class EDAClient(EDAClientBase):
|
|
def __init__(self, base_url, user_name, user_role, auth_name, auth_pass):
|
|
self.base_url = base_url
|
|
self.user_name = user_name
|
|
self.user_role = user_role
|
|
self.auth = HTTPBasicAuth(auth_name, auth_pass)
|
|
|
|
def _make_url(self, method, **kwargs):
|
|
query_args = dict(kwargs)
|
|
query_string = "&".join(
|
|
["{}={}".format(key, value) for key, value in query_args.items()]
|
|
)
|
|
return "{base_url}/{method}?{query_string}".format(
|
|
base_url=self.base_url, method=method, query_string=query_string
|
|
)
|
|
|
|
def _get(self, method, **kwargs):
|
|
url = self._make_url(method, **kwargs)
|
|
return requests.get(url, auth=self.auth, verify="ssl/server-certs/eda.pem")
|
|
|
|
def list_contracts(
|
|
self,
|
|
contract_number=None,
|
|
delivery_order=None,
|
|
cage_code=None,
|
|
duns_number=None,
|
|
):
|
|
response = self._get(
|
|
"wawf_interface.returnContractList",
|
|
pContract=contract_number,
|
|
pDelivery_Order=delivery_order,
|
|
pCage_Code=cage_code,
|
|
pDuns_Number=duns_number,
|
|
pUserName=self.user_name,
|
|
pUser_Role=self.user_role,
|
|
)
|
|
lines = response.text.replace("<br />", "").split("\n")
|
|
return list(DictReader(lines))
|
|
|
|
def get_contract(self, contract_number, status):
|
|
response = self._get(
|
|
"pds_contract_interface.get_xml_doc",
|
|
pContract=contract_number,
|
|
pStatus=status,
|
|
)
|
|
if response.text.startswith("No data found"):
|
|
return None
|
|
|
|
eda_data = {"number": contract_number}
|
|
eda_data.update(parse_eda_xml(response.text))
|
|
return eda_data
|
|
|
|
def get_clins(self, record_key, clins, cage_code="", duns_number=""):
|
|
response = self._get(
|
|
"wawf_interface.returnclinXML",
|
|
pCage_Code=cage_code,
|
|
pDuns_Number=duns_number,
|
|
pUserName=self.user_name,
|
|
pUser_Role=self.user_role,
|
|
pRecord_key=record_key,
|
|
pClins=clins,
|
|
)
|
|
# TODO: Parse XML, similar to `get_contract`
|
|
return response
|