-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathPaloFirewall.yml
320 lines (279 loc) · 16.1 KB
/
PaloFirewall.yml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
commonfields:
id: PaloFirewall
version: -1
name: PaloFirewall
display: Palo Firewall
category: Network Security
image: 
description: This integration works with the Palo Alto Next Gen Firewall API
detaileddescription: |-
## Get a FireWall API Key
To use this integration, you need a Firewall API key.
1. Log in to your Palo Alto NGFW console.
2. Add a role to Device --> Admin Roles
3. Enable XML/REST API features
4. Add a user to Administrators
5. Choose the new role you created in profile
## Configure Demisto
1. Enter your newly created username and password to your Integration instance
2. Go to the command line of XSOAR and use the ##!panos-get-api-key## command
3. Enter the key generated in the playground into your Integration instance
4. Now you can use the Test Button
configuration:
- display: Server URL (e.g. https://example.net)
name: url
defaultvalue: https://firewall
type: 0
required: true
- display: Username
name: credentials
defaultvalue: ""
type: 9
required: true
- display: Fetch incidents
name: isFetch
defaultvalue: ""
type: 8
required: false
- display: Incident type
name: incidentType
defaultvalue: ""
type: 13
required: false
- display: Require Valide Certificate
name: secure
defaultvalue: "true"
type: 8
required: false
- display: Use system proxy settings
name: proxy
defaultvalue: ""
type: 8
required: false
- display: API Key
name: apiKey
defaultvalue: ""
type: 4
required: false
additionalinfo: This key is generated by !panos-get-api-key
script:
script: |-
import urllib3
urllib3.disable_warnings()
PROXY = demisto.params().get('proxy')
SECURE = demisto.params().get('secure')
BASE_URL = demisto.params().get('url')
API_KEY = demisto.params().get('apikey')
URL_SUFFIX = 'api'
if not demisto.params().get('proxy', False):
del os.environ['HTTP_PROXY']
del os.environ['HTTPS_PROXY']
del os.environ['http_proxy']
del os.environ['https_proxy']
'''HELPER FUNCTIONS'''
def http_request(method, params):
r = requests.Response
if method is 'GET':
r = requests.get(BASE_URL + "/" + URL_SUFFIX, params=params, verify=SECURE)
elif method is 'POST':
if not API_KEY:
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
else:
headers = {
'Content-Type': 'application/json',
'Accept': 'application/json',
'X-FunTranslations-Api-Secret': API_KEY
}
r = requests.post(BASE_URL + "/" + URL_SUFFIX, params=params, headers=headers, verify=SECURE)
if r.status_code is not 200:
return_error('Error in API call [%d] - %s' % (r.status_code, r.reason))
json_result = json.loads(xml2json(str(r.text)))
# handle non success
if json_result['response']['@status'] != 'success':
if 'msg' in json_result['response'] and 'line' in json_result['response']['msg']:
# catch non existing object error and display a meaningful message
if json_result['response']['msg']['line'] == 'No such node':
raise Exception(
'Object was not found, verify that the name is correct and that the instance was committed.')
# catch urlfiltering error and display a meaningful message
elif str(json_result['response']['msg']['line']).find('test -> url') != -1:
raise Exception('The URL filtering license is either expired or not active.'
' Please contact your PAN-OS representative.')
# catch non valid jobID errors and display a meaningful message
elif isinstance(json_result['response']['msg']['line'], str) and \
json_result['response']['msg']['line'].find('job') != -1 and \
(json_result['response']['msg']['line'].find('not found') != -1
or json_result['response']['msg']['line'].find('No such query job')) != -1:
raise Exception('Invalid Job ID error: ' + json_result['response']['msg']['line'])
# catch already at the top/bottom error for rules and return this as an entry.note
elif str(json_result['response']['msg']['line']).find('already at the') != -1:
demisto.results('Rule ' + str(json_result['response']['msg']['line']))
sys.exit(0)
# catch already registered ip tags and return this as an entry.note
elif str(json_result['response']['msg']['line']).find('already exists, ignore') != -1:
if isinstance(json_result['response']['msg']['line']['uid-response']['payload']['register']['entry'],
list):
ips = [o['@ip'] for o in
json_result['response']['msg']['line']['uid-response']['payload']['register']['entry']]
else:
ips = json_result['response']['msg']['line']['uid-response']['payload']['register']['entry']['@ip']
demisto.results(
'IP ' + str(ips) + ' already exist in the tag. All submitted IPs were not registered to the tag.')
sys.exit(0)
# catch timed out log queries and return this as an entry.note
elif str(json_result['response']['msg']['line']).find('Query timed out') != -1:
demisto.results(str(json_result['response']['msg']['line']) + '. Rerun the query.')
sys.exit(0)
if '@code' in json_result['response']:
raise Exception(
'Request Failed.\nStatus code: ' + str(json_result['response']['@code']) + '\nWith message: ' + str(
json_result['response']['msg']['line']))
else:
raise Exception('Request Failed.\n' + str(json_result['response']))
# handle @code
if 'response' in json_result and '@code' in json_result['response']:
if json_result['response']['@code'] not in ['19', '20']:
# error code non exist in dict and not of success
if 'msg' in json_result['response']:
raise Exception(
'Request Failed.\nStatus code: ' + str(json_result['response']['@code']) + '\nWith message: ' + str(
json_result['response']['msg']))
else:
raise Exception('Request Failed.\n' + str(json_result['response']))
return json_result['response']['result']
""" MAIN FUNCTIONS """
def panos_get_api_key():
params = {
'type': 'keygen',
'user': demisto.params().get('credentials').get('identifier'),
'password': demisto.params().get('credentials').get('password')
}
r = http_request('GET', params)
return r
def panos_get_current_users():
params = {
'type': 'op',
'cmd': '<show><global-protect-gateway><current-user/></global-protect-gateway></show>',
'key': demisto.params().get('apiKey')
}
r = http_request('GET', params)
return r
def panos_get_gateways():
params = {
'type': 'op',
'cmd': '<show><global-protect-gateway><gateway/></global-protect-gateway></show>',
'key': demisto.params().get('apiKey')
}
r = http_request('GET', params)
return r
def panos_disconnect_current_user():
args = demisto.args()
command_string = '<request><global-protect-gateway><client-logout><gateway>' + args.get('gateway') + "-N"
command_string = command_string + '</gateway><user>' + args.get('user')
command_string = command_string + '</user><reason>' + args.get('reason')
command_string = command_string + '</reason><computer>' + args.get('computer')
command_string = command_string + '</computer></client-logout></global-protect-gateway></request>'
params = {
'type': 'op',
'cmd': command_string,
'key': demisto.params().get('apiKey')
}
r = http_request('GET', params)
return r
def panos_test():
"""
test module
"""
params = {
'type': 'op',
'cmd': '<show><system><info></info></system></show>',
'key': demisto.params().get('apiKey')
}
http_request('GET', params)
demisto.results('ok')
''' EXECUTION '''
def main():
LOG('command is %s' % (demisto.command(),))
try:
if demisto.command() == 'panos-get-api-key':
demisto.results(panos_get_api_key())
if demisto.command() == 'panos-get-current-users':
response_from_api = panos_get_current_users()
raw_response = response_from_api
command_results = CommandResults(
outputs_prefix='GlobalProtect.CurrentUsers',
outputs_key_field='public-ip',
outputs=response_from_api,
readable_output=raw_response,
raw_response=raw_response
)
return_results(command_results)
if demisto.command() == 'panos-get-gateways':
response_from_api = panos_get_gateways()
raw_response = response_from_api
command_results = CommandResults(
outputs_prefix='GlobalProtect.Gateways',
outputs_key_field='gateway-name',
outputs=response_from_api,
readable_output=raw_response,
raw_response=raw_response
)
return_results(command_results)
if demisto.command() == 'panos-disconnect-current-user':
response_from_api = panos_disconnect_current_user()
raw_response = response_from_api
command_results = CommandResults(
outputs_prefix='GlobalProtect.Disconnected',
outputs_key_field='user',
outputs=response_from_api,
readable_output=raw_response,
raw_response=raw_response
)
return_results(command_results)
elif demisto.command() == 'test-module':
panos_test()
except Exception as e:
logging.exception(e)
if __name__ == 'builtins':
main()
type: python
commands:
- name: panos-get-api-key
arguments: []
description: Get the API Key from the firewall.
- name: panos-get-current-users
arguments: []
outputs:
- contextPath: GlobalProtect.CurrentUsers
description: Global Protect Users
type: String
description: Gets the current Global Protect Users
- name: panos-disconnect-current-user
arguments:
- name: gateway
required: true
- name: user
required: true
- name: reason
required: true
auto: PREDEFINED
predefined:
- force-logout
- admin
- initiated
- logout
description: Reason for logout
defaultValue: force-logout
- name: computer
required: true
description: Disconnects the a current users from Global Protect
- name: panos-get-gateways
arguments: []
description: List the Global Protect Gateway
dockerimage: demisto/python3:3.7.5.4583
isfetch: true
runonce: false
subtype: python3