brynq-sdk-nmbrs 2.2.1__tar.gz → 2.3.0__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/PKG-INFO +1 -1
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/__init__.py +36 -8
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/absence.py +4 -11
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/companies.py +1 -4
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/employees.py +19 -1
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/salary_tables.py +3 -12
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/employees.py +10 -1
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/wage_tax.py +42 -1
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/social_insurance.py +1 -4
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/wage_tax.py +48 -8
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs.egg-info/PKG-INFO +1 -1
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/setup.py +1 -1
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/address.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/bank.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/children.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/contract.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/costcenter.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/costunit.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/days.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/debtors.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/department.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/document.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/employment.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/function.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/hours.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/leave.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/manager.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/salaries.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schedules.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/__init__.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/absence.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/address.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/bank.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/contracts.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/costcenter.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/costunit.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/days.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/debtor.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/department.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/employment.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/function.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/hours.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/leave.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/manager.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/salary.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/schedules.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/social_insurance.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/schemas/wagecomponents.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs/wagecomponents.py +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs.egg-info/SOURCES.txt +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs.egg-info/dependency_links.txt +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs.egg-info/not-zip-safe +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs.egg-info/requires.txt +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs.egg-info/top_level.txt +0 -0
- {brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/setup.cfg +0 -0
|
@@ -67,6 +67,16 @@ class Nmbrs(BrynQ):
|
|
|
67
67
|
'https://api.nmbrs.nl/soap/v3/EmployeeService.asmx?wsdl',
|
|
68
68
|
settings=self.soap_settings
|
|
69
69
|
)
|
|
70
|
+
self.soap_auth_header = self._get_soap_auth_header()
|
|
71
|
+
# Following methods can only be used if the SOAP authentication header is set (optional, this is not always in scope for all integrations)
|
|
72
|
+
if self.soap_auth_header is not None:
|
|
73
|
+
self.soap_company_ids = self.companies.get_soap_ids()
|
|
74
|
+
self.salary_tables = SalaryTables(self)
|
|
75
|
+
self.salary_scales = SalaryScales(self)
|
|
76
|
+
self.salary_steps = SalarySteps(self)
|
|
77
|
+
self.wage_tax = WageTax(self)
|
|
78
|
+
self.absence = Absence(self)
|
|
79
|
+
self.children = Children(self)
|
|
70
80
|
|
|
71
81
|
self.address = Address(self)
|
|
72
82
|
self.bank = Bank(self)
|
|
@@ -78,7 +88,6 @@ class Nmbrs(BrynQ):
|
|
|
78
88
|
debtors, _ = self.debtor.get()
|
|
79
89
|
self.debtor_ids = debtors['debtor_id'].to_list()
|
|
80
90
|
self.company_ids = self.companies.get()['companyId'].to_list()
|
|
81
|
-
self.soap_company_ids = self.companies.get_soap_ids()
|
|
82
91
|
self.employees = Employees(self)
|
|
83
92
|
self.employment = Employment(self)
|
|
84
93
|
self.function = EmployeeFunction(self)
|
|
@@ -92,12 +101,6 @@ class Nmbrs(BrynQ):
|
|
|
92
101
|
self.schedule = Schedule(self)
|
|
93
102
|
self.fixed_wagecomponents = EmployeeFixedWageComponents(self)
|
|
94
103
|
self.variable_wagecomponents = EmployeeVariableWageComponents(self)
|
|
95
|
-
self.children = Children(self)
|
|
96
|
-
self.salary_tables = SalaryTables(self)
|
|
97
|
-
self.salary_scales = SalaryScales(self)
|
|
98
|
-
self.salary_steps = SalarySteps(self)
|
|
99
|
-
self.wage_tax = WageTax(self)
|
|
100
|
-
self.absence = Absence(self)
|
|
101
104
|
self.current_period = self.companies.get_current_period()
|
|
102
105
|
|
|
103
106
|
def _get_request_headers(self):
|
|
@@ -115,6 +118,31 @@ class Nmbrs(BrynQ):
|
|
|
115
118
|
"""
|
|
116
119
|
Creates the SOAP authentication header using credentials from initial_credentials.
|
|
117
120
|
|
|
121
|
+
Returns:
|
|
122
|
+
AuthHeaderWithDomainType: The authentication header for SOAP requests
|
|
123
|
+
"""
|
|
124
|
+
initial_credentials = self.interfaces.credentials.get(system='nmbrs', system_type=self.system_type)
|
|
125
|
+
config = initial_credentials.get("custom_data", {})
|
|
126
|
+
|
|
127
|
+
if 'soap_api_token' not in config.keys():
|
|
128
|
+
return None
|
|
129
|
+
else:
|
|
130
|
+
# Get the AuthHeaderWithDomain type from the WSDL
|
|
131
|
+
AuthHeaderWithDomainType = self.soap_client_companies.get_element('ns0:AuthHeaderWithDomain')
|
|
132
|
+
|
|
133
|
+
# Create the auth header using credentials from config
|
|
134
|
+
auth_header = AuthHeaderWithDomainType(
|
|
135
|
+
Username=config.get("soap_api_username"),
|
|
136
|
+
Token=config.get("soap_api_token"),
|
|
137
|
+
Domain=config.get("soap_api_domain")
|
|
138
|
+
)
|
|
139
|
+
|
|
140
|
+
return auth_header
|
|
141
|
+
|
|
142
|
+
def _get_soap_auth_header_employees(self):
|
|
143
|
+
"""
|
|
144
|
+
Creates the SOAP authentication header using credentials from initial_credentials.
|
|
145
|
+
|
|
118
146
|
Returns:
|
|
119
147
|
AuthHeaderWithDomainType: The authentication header for SOAP requests
|
|
120
148
|
"""
|
|
@@ -122,7 +150,7 @@ class Nmbrs(BrynQ):
|
|
|
122
150
|
config = initial_credentials.get("config", {})
|
|
123
151
|
|
|
124
152
|
# Get the AuthHeaderWithDomain type from the WSDL
|
|
125
|
-
AuthHeaderWithDomainType = self.
|
|
153
|
+
AuthHeaderWithDomainType = self.soap_client_employees.get_element('ns0:AuthHeaderWithDomain')
|
|
126
154
|
|
|
127
155
|
# Create the auth header using credentials from config
|
|
128
156
|
auth_header = AuthHeaderWithDomainType(
|
|
@@ -40,7 +40,7 @@ class Absence:
|
|
|
40
40
|
|
|
41
41
|
def _get(self, company_id: int, employee_id: int = None) -> pd.DataFrame:
|
|
42
42
|
"""
|
|
43
|
-
Get
|
|
43
|
+
Get all absences for a specific company, period and year.
|
|
44
44
|
|
|
45
45
|
Args:
|
|
46
46
|
company_id (int): The ID of the company
|
|
@@ -51,20 +51,16 @@ class Absence:
|
|
|
51
51
|
pd.DataFrame: DataFrame containing the salary tables
|
|
52
52
|
"""
|
|
53
53
|
try:
|
|
54
|
-
|
|
55
|
-
# Get the auth header using the centralized method
|
|
56
|
-
auth_header = self.nmbrs._get_soap_auth_header()
|
|
57
|
-
|
|
58
54
|
if employee_id is None:
|
|
59
55
|
response = self.soap_client_employees.service.Absence_GetAll_AllEmployeesByCompany(
|
|
60
56
|
CompanyId=company_id,
|
|
61
|
-
_soapheaders=[
|
|
57
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
62
58
|
)
|
|
63
59
|
else:
|
|
64
60
|
# Make SOAP request with the proper header structure
|
|
65
61
|
response = self.soap_client_employees.service.Absence_GetList(
|
|
66
62
|
EmployeeId=employee_id,
|
|
67
|
-
_soapheaders=[
|
|
63
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
68
64
|
)
|
|
69
65
|
|
|
70
66
|
# Convert response to DataFrame
|
|
@@ -95,9 +91,6 @@ class Absence:
|
|
|
95
91
|
if self.nmbrs.mock_mode:
|
|
96
92
|
return absence_model
|
|
97
93
|
|
|
98
|
-
# Get the auth header using the centralized method
|
|
99
|
-
auth_header = self.nmbrs._get_soap_auth_header()
|
|
100
|
-
|
|
101
94
|
# Use the model's built-in SOAP conversion method
|
|
102
95
|
absence_settings = absence_model.to_soap_settings(self.nmbrs.soap_client_employees)
|
|
103
96
|
|
|
@@ -106,7 +99,7 @@ class Absence:
|
|
|
106
99
|
EmployeeId=absence_model.employee_id,
|
|
107
100
|
Dossier=absence_model.new_dossier,
|
|
108
101
|
Absence=absence_settings,
|
|
109
|
-
_soapheaders=[
|
|
102
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
110
103
|
)
|
|
111
104
|
|
|
112
105
|
# Convert response to DataFrame
|
|
@@ -51,12 +51,9 @@ class Companies:
|
|
|
51
51
|
pd.DataFrame: DataFrame containing all companies
|
|
52
52
|
"""
|
|
53
53
|
try:
|
|
54
|
-
# Get the auth header using the centralized method
|
|
55
|
-
auth_header = self.nmbrs._get_soap_auth_header()
|
|
56
|
-
|
|
57
54
|
# Make SOAP request with the proper header structure
|
|
58
55
|
response = self.soap_client_companies.service.List_GetAll(
|
|
59
|
-
_soapheaders=[
|
|
56
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
60
57
|
)
|
|
61
58
|
|
|
62
59
|
# Convert response to DataFrame
|
|
@@ -22,7 +22,7 @@ from .days import VariableDays
|
|
|
22
22
|
from .wagecomponents import EmployeeVariableWageComponents, EmployeeFixedWageComponents
|
|
23
23
|
from .leave import Leave, LeaveBalance
|
|
24
24
|
from .schemas.employees import (
|
|
25
|
-
EmployeeGet, EmployeeCreate, EmployeeUpdate, EmployeeDelete,
|
|
25
|
+
BsnGet, EmployeeGet, EmployeeCreate, EmployeeUpdate, EmployeeDelete,
|
|
26
26
|
BasicInfo, BirthInfo, ContactInfo, PartnerInfo, Period, AdditionalEmployeeInfo,
|
|
27
27
|
CreateEmployeePersonalInfo
|
|
28
28
|
)
|
|
@@ -83,6 +83,24 @@ class Employees:
|
|
|
83
83
|
|
|
84
84
|
return df
|
|
85
85
|
|
|
86
|
+
def get_private_info(self) -> str:
|
|
87
|
+
combined = []
|
|
88
|
+
for company_id in self.nmbrs.company_ids:
|
|
89
|
+
request = requests.Request(method='GET',
|
|
90
|
+
url=f"{self.nmbrs.base_url}companies/{company_id}/employees/privateInfos")
|
|
91
|
+
data = self.nmbrs.get_paginated_result(request)
|
|
92
|
+
df = pd.json_normalize(
|
|
93
|
+
data,
|
|
94
|
+
record_path='privateInfos',
|
|
95
|
+
meta=['employeeId']
|
|
96
|
+
)
|
|
97
|
+
df['companyId'] = company_id
|
|
98
|
+
combined.append(df)
|
|
99
|
+
df = pd.concat(combined, ignore_index=True)
|
|
100
|
+
valid_bsn, invalid_bsn = Functions.validate_data(df, BsnGet)
|
|
101
|
+
|
|
102
|
+
return valid_bsn, invalid_bsn
|
|
103
|
+
|
|
86
104
|
def get_default_templates(self) -> pd.DataFrame:
|
|
87
105
|
default_templates = pd.DataFrame()
|
|
88
106
|
for company in self.nmbrs.company_ids:
|
|
@@ -50,15 +50,12 @@ class SalaryTables:
|
|
|
50
50
|
pd.DataFrame: DataFrame containing the salary tables
|
|
51
51
|
"""
|
|
52
52
|
try:
|
|
53
|
-
# Get the auth header using the centralized method
|
|
54
|
-
auth_header = self.nmbrs._get_soap_auth_header()
|
|
55
|
-
|
|
56
53
|
# Make SOAP request with the proper header structure
|
|
57
54
|
response = self.soap_client_companies.service.SalaryTable2_Get(
|
|
58
55
|
CompanyId=company_id,
|
|
59
56
|
Period=period,
|
|
60
57
|
Year=year,
|
|
61
|
-
_soapheaders=[
|
|
58
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
62
59
|
)
|
|
63
60
|
|
|
64
61
|
# Convert response to DataFrame
|
|
@@ -123,15 +120,12 @@ class SalaryScales:
|
|
|
123
120
|
pd.DataFrame: DataFrame containing the salary scales
|
|
124
121
|
"""
|
|
125
122
|
try:
|
|
126
|
-
# Get the auth header using the centralized method
|
|
127
|
-
auth_header = self.nmbrs._get_soap_auth_header()
|
|
128
|
-
|
|
129
123
|
# Make SOAP request with the proper header structure
|
|
130
124
|
response = self.soap_client_companies.service.SalaryTable2_GetScales(
|
|
131
125
|
CompanyId=company_id,
|
|
132
126
|
Period=period,
|
|
133
127
|
Year=year,
|
|
134
|
-
_soapheaders=[
|
|
128
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
135
129
|
)
|
|
136
130
|
|
|
137
131
|
# Convert response to DataFrame
|
|
@@ -208,9 +202,6 @@ class SalarySteps:
|
|
|
208
202
|
pd.DataFrame: DataFrame containing the salary steps
|
|
209
203
|
"""
|
|
210
204
|
try:
|
|
211
|
-
# Get the auth header using the centralized method
|
|
212
|
-
auth_header = self.nmbrs._get_soap_auth_header()
|
|
213
|
-
|
|
214
205
|
# Make SOAP request with the proper header structure
|
|
215
206
|
response = self.soap_client_companies.service.SalaryTable2_GetSteps(
|
|
216
207
|
CompanyId=company_id,
|
|
@@ -223,7 +214,7 @@ class SalarySteps:
|
|
|
223
214
|
'ScalePercentageMax': scale.get('scale_percentage_max', 0),
|
|
224
215
|
'ScalePercentageMin': scale.get('scale_percentage_min', 0)
|
|
225
216
|
},
|
|
226
|
-
_soapheaders=[
|
|
217
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
227
218
|
)
|
|
228
219
|
|
|
229
220
|
# Convert response to DataFrame
|
|
@@ -84,7 +84,7 @@ class BirthInfo(BaseModel):
|
|
|
84
84
|
StringConstraints(
|
|
85
85
|
pattern=r'^[A-Za-z]+$',
|
|
86
86
|
strip_whitespace=True,
|
|
87
|
-
min_length=2,
|
|
87
|
+
min_length=2,
|
|
88
88
|
max_length=3
|
|
89
89
|
)
|
|
90
90
|
]] = Field(None, example="PT", description="Nationality Code ISO", alias="nationalityCodeISO")
|
|
@@ -140,3 +140,12 @@ class EmployeeUpdate(BaseModel):
|
|
|
140
140
|
|
|
141
141
|
class EmployeeDelete(BaseModel):
|
|
142
142
|
employee_id: str = Field(..., example="3054d4cf-b449-489d-8d2e-5dd30e5ab994", description="Employee ID", alias="employeeId")
|
|
143
|
+
|
|
144
|
+
class BsnGet(BrynQPanderaDataFrameModel):
|
|
145
|
+
employee_id: Series[String] = pa.Field(coerce=True, description="Employee ID", alias="employeeId")
|
|
146
|
+
social_security_number: Series[String] = pa.Field(coerce=True, description="Social Security Number", alias="BSN")
|
|
147
|
+
company_id: Series[String] = pa.Field(coerce=True, description="Company ID", alias="companyId")
|
|
148
|
+
created_at: Series[DateTime] = pa.Field(coerce=True, description="Created At", alias="createdAt")
|
|
149
|
+
|
|
150
|
+
class _Annotation:
|
|
151
|
+
primary_key = "employee_id"
|
|
@@ -10,7 +10,7 @@ from pydantic import BaseModel, Field, StringConstraints
|
|
|
10
10
|
from datetime import datetime
|
|
11
11
|
|
|
12
12
|
|
|
13
|
-
class
|
|
13
|
+
class WageTaxSettingsGet(BrynQPanderaDataFrameModel):
|
|
14
14
|
wagetax_id: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Loonaangifte ID", alias="LoonaangifteID")
|
|
15
15
|
serial_number: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Serial Number", alias="SerialNumber")
|
|
16
16
|
payment_reference: Series[String] = pa.Field(coerce=True, description="Payment Reference", alias="PaymentReference")
|
|
@@ -24,6 +24,47 @@ class WageTaxGet(BrynQPanderaDataFrameModel):
|
|
|
24
24
|
correction_period_start: Series[DateTime] = pa.Field(nullable=True, coerce=True, description="Correction Tijdvak Start", alias="CorrectionTijdvakStart")
|
|
25
25
|
correction_period_end: Series[DateTime] = pa.Field(nullable=True, coerce=True, description="Correction Tijdvak End", alias="CorrectionTijdvakEnd")
|
|
26
26
|
|
|
27
|
+
# <soap:Envelope xmlns:soap="http://www.w3.org/2003/05/soap-envelope" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
|
|
28
|
+
# <soap:Body>
|
|
29
|
+
# <WageTax_GetListResponse xmlns="https://api.nmbrs.nl/soap/v3/EmployeeService">
|
|
30
|
+
# <WageTax_GetListResult>
|
|
31
|
+
# <WageTaxSettings>
|
|
32
|
+
# <Id>2998190</Id>
|
|
33
|
+
# <JaarloonBT>12144.00</JaarloonBT>
|
|
34
|
+
# <AfwBijzTariefLH>0</AfwBijzTariefLH>
|
|
35
|
+
# <AutoKleineBanenRegeling>true</AutoKleineBanenRegeling>
|
|
36
|
+
# <Loonheffingkorting>true</Loonheffingkorting>
|
|
37
|
+
# <Voordeelreg>true</Voordeelreg>
|
|
38
|
+
# <Loonheffing>true</Loonheffing>
|
|
39
|
+
# <CodeAfdrachtvermindering>0</CodeAfdrachtvermindering>
|
|
40
|
+
# <KleurTabel>1</KleurTabel>
|
|
41
|
+
# <SoortInkomen>15</SoortInkomen>
|
|
42
|
+
# <SpecialeTabel>0</SpecialeTabel>
|
|
43
|
+
# <TijdvakTabel>2</TijdvakTabel>
|
|
44
|
+
# <VakantieBonnen>0</VakantieBonnen>
|
|
45
|
+
# <CodeCalc30PercRule>0</CodeCalc30PercRule>
|
|
46
|
+
# </WageTaxSettings>
|
|
47
|
+
# </WageTax_GetListResult>
|
|
48
|
+
# </WageTax_GetListResponse>
|
|
49
|
+
# </soap:Body>
|
|
50
|
+
# </soap:Envelope>
|
|
51
|
+
|
|
52
|
+
class WageTaxGet(BrynQPanderaDataFrameModel):
|
|
53
|
+
wage_tax_id: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Wage Tax ID", alias="Id")
|
|
54
|
+
employee_id: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Employee ID", alias="EmployeeId")
|
|
55
|
+
yearly_salary: Series[pd.Float64Dtype] = pa.Field(coerce=True, description="Yearly Salary", alias="JaarloonBT")
|
|
56
|
+
deviation_special_rate_payroll_tax_deduction: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Afw Bijz Tarief LH", alias="AfwBijzTariefLH")
|
|
57
|
+
auto_small_jobs: Series[Bool] = pa.Field(coerce=True, description="Auto Kleine Banen Regeling", alias="AutoKleineBanenRegeling")
|
|
58
|
+
payroll_tax_deduction: Series[Bool] = pa.Field(coerce=True, description="Loonheffingkorting", alias="Loonheffingkorting")
|
|
59
|
+
benefit_scheme: Series[Bool] = pa.Field(coerce=True, description="Voordeelreg", alias="Voordeelreg")
|
|
60
|
+
payroll_tax: Series[Bool] = pa.Field(coerce=True, description="Loonheffing", alias="Loonheffing")
|
|
61
|
+
code_tax_reduction: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Code Afdrachtvermindering", alias="CodeAfdrachtvermindering")
|
|
62
|
+
color_table: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Kleur Tabel", alias="KleurTabel")
|
|
63
|
+
type_of_income: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Soort Inkomen", alias="SoortInkomen")
|
|
64
|
+
special_table: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Speciale Tabel", alias="SpecialeTabel")
|
|
65
|
+
period_table: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Tijdvak Tabel", alias="TijdvakTabel")
|
|
66
|
+
holiday_vouchers: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Vakantie Bonnen", alias="VakantieBonnen")
|
|
67
|
+
code_calculate_30_percent_rule: Series[pd.Int64Dtype] = pa.Field(coerce=True, description="Code Calc 30% Rule", alias="CodeCalc30PercRule")
|
|
27
68
|
|
|
28
69
|
class WageTaxUpdate(BaseModel):
|
|
29
70
|
employee_id: Optional[int] = Field(None, example="1234567890", description="Employee ID", alias="EmployeeId")
|
|
@@ -20,9 +20,6 @@ class SocialInsurance:
|
|
|
20
20
|
if self.nmbrs.mock_mode:
|
|
21
21
|
return social_insurance_model
|
|
22
22
|
|
|
23
|
-
# Get the auth header using the centralized method
|
|
24
|
-
auth_header = self.nmbrs._get_soap_auth_header()
|
|
25
|
-
|
|
26
23
|
# Use the model's built-in SOAP conversion method
|
|
27
24
|
social_insurance_settings = social_insurance_model.to_soap_settings(self.nmbrs.soap_client_employees)
|
|
28
25
|
|
|
@@ -30,7 +27,7 @@ class SocialInsurance:
|
|
|
30
27
|
response = self.nmbrs.soap_client_employees.service.SVW_UpdateCurrent(
|
|
31
28
|
EmployeeId=social_insurance_model.employee_id,
|
|
32
29
|
SVWSettings=social_insurance_settings,
|
|
33
|
-
_soapheaders=[
|
|
30
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
34
31
|
)
|
|
35
32
|
|
|
36
33
|
# Convert response to DataFrame
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
from typing import Any, Dict, List, Union, Tuple
|
|
2
2
|
import pandas as pd
|
|
3
|
-
from .schemas.wage_tax import WageTaxGet, WageTaxUpdate
|
|
3
|
+
from .schemas.wage_tax import WageTaxGet, WageTaxUpdate, WageTaxSettingsGet
|
|
4
4
|
from zeep.exceptions import Fault
|
|
5
5
|
from zeep.ns import WSDL, SOAP_ENV_11
|
|
6
6
|
from zeep.xsd import ComplexType, Element, String
|
|
@@ -33,7 +33,7 @@ class WageTax:
|
|
|
33
33
|
wagetax_settings_temp['companyId'] = company['number']
|
|
34
34
|
wagetax_settings = pd.concat([wagetax_settings, wagetax_settings_temp])
|
|
35
35
|
|
|
36
|
-
valid_wagetax_settings, invalid_wagetax_settings = Functions.validate_data(df=wagetax_settings, schema=
|
|
36
|
+
valid_wagetax_settings, invalid_wagetax_settings = Functions.validate_data(df=wagetax_settings, schema=WageTaxSettingsGet, debug=True)
|
|
37
37
|
|
|
38
38
|
# No validation schema for now, but could be added later
|
|
39
39
|
return valid_wagetax_settings, invalid_wagetax_settings
|
|
@@ -51,14 +51,11 @@ class WageTax:
|
|
|
51
51
|
pd.DataFrame: DataFrame containing the salary tables
|
|
52
52
|
"""
|
|
53
53
|
try:
|
|
54
|
-
# Get the auth header using the centralized method
|
|
55
|
-
auth_header = self.nmbrs._get_soap_auth_header()
|
|
56
|
-
|
|
57
54
|
# Make SOAP request with the proper header structure
|
|
58
55
|
response = self.soap_client_companies.service.WageTax_GetList(
|
|
59
56
|
CompanyId=company_id,
|
|
60
57
|
intYear=year,
|
|
61
|
-
_soapheaders=[
|
|
58
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
62
59
|
)
|
|
63
60
|
|
|
64
61
|
# Convert response to DataFrame
|
|
@@ -82,6 +79,49 @@ class WageTax:
|
|
|
82
79
|
except Exception as e:
|
|
83
80
|
raise Exception(f"Failed to get salary tables: {str(e)}")
|
|
84
81
|
|
|
82
|
+
def get(self, employee_id: int) -> Tuple[pd.DataFrame, pd.DataFrame]:
|
|
83
|
+
"""
|
|
84
|
+
Get wage tax settings for a specific employee.
|
|
85
|
+
|
|
86
|
+
Args:
|
|
87
|
+
employee_id (int): The ID of the employee
|
|
88
|
+
|
|
89
|
+
Returns:
|
|
90
|
+
pd.DataFrame: DataFrame containing the wage tax settings
|
|
91
|
+
"""
|
|
92
|
+
try:
|
|
93
|
+
# Get the auth header using the centralized method
|
|
94
|
+
auth_header = self.nmbrs._get_soap_auth_header_employees()
|
|
95
|
+
|
|
96
|
+
# Make SOAP request with the proper header structure
|
|
97
|
+
response = self.soap_client_employees.service.WageTax_GetCurrent(
|
|
98
|
+
EmployeeId=employee_id,
|
|
99
|
+
_soapheaders=[auth_header]
|
|
100
|
+
)
|
|
101
|
+
|
|
102
|
+
# Convert response to DataFrame
|
|
103
|
+
if response:
|
|
104
|
+
# Convert Zeep objects to Python dictionaries
|
|
105
|
+
serialized_response = serialize_object(response)
|
|
106
|
+
|
|
107
|
+
# Convert to list if it's not already
|
|
108
|
+
if not isinstance(serialized_response, list):
|
|
109
|
+
serialized_response = [serialized_response]
|
|
110
|
+
|
|
111
|
+
# Convert to DataFrame
|
|
112
|
+
df = pd.DataFrame(serialized_response)
|
|
113
|
+
df['EmployeeId'] = employee_id
|
|
114
|
+
valid_wage_tax, invalid_wage_tax = Functions.validate_data(df=df, schema=WageTaxGet, debug=True)
|
|
115
|
+
|
|
116
|
+
return valid_wage_tax, invalid_wage_tax
|
|
117
|
+
else:
|
|
118
|
+
return pd.DataFrame(), pd.DataFrame()
|
|
119
|
+
|
|
120
|
+
except Fault as e:
|
|
121
|
+
raise Exception(f"SOAP request failed: {str(e)}")
|
|
122
|
+
except Exception as e:
|
|
123
|
+
raise Exception(f"Failed to get wage tax settings: {str(e)}")
|
|
124
|
+
|
|
85
125
|
def update(self, data: Dict[str, Any]) -> pd.DataFrame:
|
|
86
126
|
try:
|
|
87
127
|
wage_tax_model = WageTaxUpdate(**data)
|
|
@@ -90,7 +130,7 @@ class WageTax:
|
|
|
90
130
|
return wage_tax_model
|
|
91
131
|
|
|
92
132
|
# Get the auth header using the centralized method
|
|
93
|
-
auth_header = self.nmbrs.
|
|
133
|
+
auth_header = self.nmbrs._get_soap_auth_header_employees()
|
|
94
134
|
|
|
95
135
|
# Use the model's built-in SOAP conversion method
|
|
96
136
|
wage_tax_settings = wage_tax_model.to_soap_settings(self.nmbrs.soap_client_employees)
|
|
@@ -99,7 +139,7 @@ class WageTax:
|
|
|
99
139
|
response = self.nmbrs.soap_client_employees.service.WageTax_UpdateCurrent(
|
|
100
140
|
EmployeeId=wage_tax_model.employee_id,
|
|
101
141
|
LoonheffingSettings=wage_tax_settings,
|
|
102
|
-
_soapheaders=[
|
|
142
|
+
_soapheaders=[self.nmbrs.soap_auth_header]
|
|
103
143
|
)
|
|
104
144
|
|
|
105
145
|
# Convert response to DataFrame
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{brynq_sdk_nmbrs-2.2.1 → brynq_sdk_nmbrs-2.3.0}/brynq_sdk_nmbrs.egg-info/dependency_links.txt
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|