1
|
|
|
<?php |
2
|
|
|
|
3
|
|
|
/* |
4
|
|
|
* ***************************************************************************** |
5
|
|
|
* Contributions to this work were made on behalf of the GÉANT project, a |
6
|
|
|
* project that has received funding from the European Union’s Framework |
7
|
|
|
* Programme 7 under Grant Agreements No. 238875 (GN3) and No. 605243 (GN3plus), |
8
|
|
|
* Horizon 2020 research and innovation programme under Grant Agreements No. |
9
|
|
|
* 691567 (GN4-1) and No. 731122 (GN4-2). |
10
|
|
|
* On behalf of the aforementioned projects, GEANT Association is the sole owner |
11
|
|
|
* of the copyright in all material which was developed by a member of the GÉANT |
12
|
|
|
* project. GÉANT Vereniging (Association) is registered with the Chamber of |
13
|
|
|
* Commerce in Amsterdam with registration number 40535155 and operates in the |
14
|
|
|
* UK as a branch of GÉANT Vereniging. |
15
|
|
|
* |
16
|
|
|
* Registered office: Hoekenrode 3, 1102BR Amsterdam, The Netherlands. |
17
|
|
|
* UK branch address: City House, 126-130 Hills Road, Cambridge CB2 1PQ, UK |
18
|
|
|
* |
19
|
|
|
* License: see the web/copyright.inc.php file in the file structure or |
20
|
|
|
* <base_url>/copyright.php after deploying the software |
21
|
|
|
*/ |
22
|
|
|
|
23
|
|
|
/** |
24
|
|
|
* This file contains the AbstractProfile class. It contains common methods for |
25
|
|
|
* both RADIUS/EAP profiles and SilverBullet profiles |
26
|
|
|
* |
27
|
|
|
* @author Stefan Winter <[email protected]> |
28
|
|
|
* @author Tomasz Wolniewicz <[email protected]> |
29
|
|
|
* @author Maja Górecka-Wolniewicz <[email protected]> |
30
|
|
|
* |
31
|
|
|
* @package Developer |
32
|
|
|
* |
33
|
|
|
*/ |
34
|
|
|
|
35
|
|
|
namespace core; |
36
|
|
|
|
37
|
|
|
use \Exception; |
|
|
|
|
38
|
|
|
|
39
|
|
|
/** |
40
|
|
|
* This class represents an EAP Profile. |
41
|
|
|
* Profiles can inherit attributes from their IdP, if the IdP has some. Otherwise, |
42
|
|
|
* one can set attribute in the Profile directly. If there is a conflict between |
43
|
|
|
* IdP-wide and Profile-wide attributes, the more specific ones (i.e. Profile) win. |
44
|
|
|
* |
45
|
|
|
* @author Stefan Winter <[email protected]> |
46
|
|
|
* @author Tomasz Wolniewicz <[email protected]> |
47
|
|
|
* |
48
|
|
|
* @license see LICENSE file in root directory |
49
|
|
|
* |
50
|
|
|
* @package Developer |
51
|
|
|
*/ |
52
|
|
|
abstract class AbstractProfile extends EntityWithDBProperties |
53
|
|
|
{ |
54
|
|
|
|
55
|
|
|
const HIDDEN = -1; |
56
|
|
|
const AVAILABLE = 0; |
57
|
|
|
const UNAVAILABLE = 1; |
58
|
|
|
const INCOMPLETE = 2; |
59
|
|
|
const NOTCONFIGURED = 3; |
60
|
|
|
const PROFILETYPE_RADIUS = "RADIUS"; |
61
|
|
|
const PROFILETYPE_SILVERBULLET = "SILVERBULLET"; |
62
|
|
|
public const SERVERNAME_ADDED = 2; |
63
|
|
|
public const CA_ADDED = 3; |
64
|
|
|
public const CA_CLASH_ADDED = 4; |
65
|
|
|
public const SERVER_CERT_ADDED = 5; |
66
|
|
|
public const CA_ROOT_NO_EXT = 6; |
67
|
|
|
|
68
|
|
|
/** |
69
|
|
|
* DB identifier of the parent institution of this profile |
70
|
|
|
* @var integer |
71
|
|
|
*/ |
72
|
|
|
public $institution; |
73
|
|
|
|
74
|
|
|
/** |
75
|
|
|
* name of the parent institution of this profile in the current language |
76
|
|
|
* @var string |
77
|
|
|
*/ |
78
|
|
|
public $instName; |
79
|
|
|
|
80
|
|
|
/** |
81
|
|
|
* realm of this profile (empty string if unset) |
82
|
|
|
* @var string |
83
|
|
|
*/ |
84
|
|
|
public $realm; |
85
|
|
|
|
86
|
|
|
/** |
87
|
|
|
* This array holds the supported EAP types (in object representation). |
88
|
|
|
* |
89
|
|
|
* They are not synced against the DB after instantiation. |
90
|
|
|
* |
91
|
|
|
* @var array |
92
|
|
|
*/ |
93
|
|
|
protected $privEaptypes; |
94
|
|
|
|
95
|
|
|
/** |
96
|
|
|
* number of profiles of the IdP this profile is attached to |
97
|
|
|
* |
98
|
|
|
* @var integer |
99
|
|
|
*/ |
100
|
|
|
protected $idpNumberOfProfiles; |
101
|
|
|
|
102
|
|
|
/** |
103
|
|
|
* IdP-wide attributes of the IdP this profile is attached to |
104
|
|
|
* |
105
|
|
|
* @var array |
106
|
|
|
*/ |
107
|
|
|
protected $idpAttributes; |
108
|
|
|
|
109
|
|
|
/** |
110
|
|
|
* Federation level attributes that this profile is attached to via its IdP |
111
|
|
|
* |
112
|
|
|
* @var array |
113
|
|
|
*/ |
114
|
|
|
protected $fedAttributes; |
115
|
|
|
|
116
|
|
|
/** |
117
|
|
|
* This class also needs to handle frontend operations, so needs its own |
118
|
|
|
* access to the FRONTEND database. This member stores the corresponding |
119
|
|
|
* handle. |
120
|
|
|
* |
121
|
|
|
* @var DBConnection |
122
|
|
|
*/ |
123
|
|
|
protected $frontendHandle; |
124
|
|
|
|
125
|
|
|
/** |
126
|
|
|
* readiness levels for OpenRoaming column in profiles) |
127
|
|
|
*/ |
128
|
|
|
const OVERALL_OPENROAMING_LEVEL_NO = 4; |
129
|
|
|
const OVERALL_OPENROAMING_LEVEL_GOOD = 3; |
130
|
|
|
const OVERALL_OPENROAMING_LEVEL_NOTE = 2; |
131
|
|
|
const OVERALL_OPENROAMING_LEVEL_WARN = 1; |
132
|
|
|
const OVERALL_OPENROAMING_LEVEL_ERROR = 0; |
133
|
|
|
|
134
|
|
|
/** |
135
|
|
|
* constants used for displaying messages |
136
|
|
|
*/ |
137
|
|
|
const OPENROAMING_ALL_GOOD = 24; |
138
|
|
|
const OPENROAMING_NO_REALM = 17; //none |
139
|
|
|
const OPENROAMING_BAD_SRV = 16; //none |
140
|
|
|
const OPENROAMING_BAD_NAPTR = 10; // warning |
141
|
|
|
const OPENROAMING_SOME_BAD_CONNECTIONS = 8; //warning |
142
|
|
|
const OPENROAMING_NO_DNSSEC = 8; //warning |
143
|
|
|
const OPENROAMING_NO_NAPTR = 3; //error |
144
|
|
|
const OPENROAMING_BAD_NAPTR_RESOLVE = 2; //error |
145
|
|
|
const OPENROAMING_BAD_SRV_RESOLVE = 1; //error |
146
|
|
|
const OPENROAMING_BAD_CONNECTION = 0; //error |
147
|
|
|
|
148
|
|
|
|
149
|
|
|
const READINESS_LEVEL_NOTREADY = 0; |
150
|
|
|
const READINESS_LEVEL_SUFFICIENTCONFIG = 1; |
151
|
|
|
const READINESS_LEVEL_SHOWTIME = 2; |
152
|
|
|
|
153
|
|
|
|
154
|
|
|
const CERT_STATUS_NONE = -1; |
155
|
|
|
const CERT_STATUS_OK = 0; |
156
|
|
|
const CERT_STATUS_WARN = 1; |
157
|
|
|
const CERT_STATUS_ERROR = 2; |
158
|
|
|
|
159
|
|
|
const OVERALL_OPENROAMING_INDEX = [ |
160
|
|
|
self::OVERALL_OPENROAMING_LEVEL_NO => 'OVERALL_OPENROAMING_LEVEL_NO', |
161
|
|
|
self::OVERALL_OPENROAMING_LEVEL_GOOD => 'OVERALL_OPENROAMING_LEVEL_GOOD', |
162
|
|
|
self::OVERALL_OPENROAMING_LEVEL_NOTE => 'OVERALL_OPENROAMING_LEVEL_NOTE', |
163
|
|
|
self::OVERALL_OPENROAMING_LEVEL_WARN => 'OVERALL_OPENROAMING_LEVEL_WARN', |
164
|
|
|
self::OVERALL_OPENROAMING_LEVEL_ERROR => 'OVERALL_OPENROAMING_LEVEL_ERROR', |
165
|
|
|
]; |
166
|
|
|
|
167
|
|
|
const OPENROAMING_INDEX = [ |
168
|
|
|
self::OVERALL_OPENROAMING_LEVEL_NO => 'OVERALL_OPENROAMING_LEVEL_NO', |
169
|
|
|
]; |
170
|
|
|
|
171
|
|
|
const CERT_STATUS_INDEX = [ |
172
|
|
|
self::CERT_STATUS_OK => 'CERT_STATUS_OK', |
173
|
|
|
self::CERT_STATUS_WARN => 'CERT_STATUS_WARN', |
174
|
|
|
self::CERT_STATUS_ERROR => 'CERT_STATUS_ERROR', |
175
|
|
|
]; |
176
|
|
|
|
177
|
|
|
/** |
178
|
|
|
* generates a detailed log of which installer was downloaded |
179
|
|
|
* |
180
|
|
|
* @param int $idpIdentifier the IdP identifier |
181
|
|
|
* @param int $profileId the Profile identifier |
182
|
|
|
* @param string $deviceId the Device identifier |
183
|
|
|
* @param string $area the download area (user, silverbullet, admin) |
184
|
|
|
* @param string $lang the language of the installer |
185
|
|
|
* @param int $eapType the EAP type of the installer |
186
|
|
|
* @return void |
187
|
|
|
* @throws Exception |
188
|
|
|
*/ |
189
|
|
|
protected function saveDownloadDetails($idpIdentifier, $profileId, $deviceId, $area, $lang, $eapType, $openRoaming) |
190
|
|
|
{ |
191
|
|
|
if (\config\Master::PATHS['logdir']) { |
192
|
|
|
$file = fopen(\config\Master::PATHS['logdir']."/download_details.log", "a"); |
193
|
|
|
if ($file === FALSE) { |
194
|
|
|
throw new Exception("Unable to open file for append: $file"); |
195
|
|
|
} |
196
|
|
|
fprintf($file, "%-015s;%d;%d;%s;%s;%s;%d;%d\n", microtime(TRUE), $idpIdentifier, $profileId, $deviceId, $area, $lang, $eapType, $openRoaming); |
197
|
|
|
fclose($file); |
198
|
|
|
} |
199
|
|
|
} |
200
|
|
|
|
201
|
|
|
/** |
202
|
|
|
* checks if security-relevant parameters have changed |
203
|
|
|
* |
204
|
|
|
* @param AbstractProfile $old old instantiation of a profile to compare against |
205
|
|
|
* @param AbstractProfile $new new instantiation of a profile |
206
|
|
|
* @return array there are never any user-induced changes in SB |
207
|
|
|
*/ |
208
|
|
|
public static function significantChanges($old, $new) |
209
|
|
|
{ |
210
|
|
|
$retval = []; |
211
|
|
|
// check if a CA was added |
212
|
|
|
$x509 = new common\X509(); |
213
|
|
|
$baselineCA = []; |
214
|
|
|
$baselineCApublicKey = []; |
215
|
|
|
foreach ($old->getAttributes("eap:ca_file") as $oldCA) { |
216
|
|
|
$ca = $x509->processCertificate($oldCA['value']); |
217
|
|
|
$baselineCA[$ca['sha1']] = $ca['name']; |
218
|
|
|
$baselineCApublicKey[$ca['sha1']] = $ca['full_details']['public_key']; |
219
|
|
|
} |
220
|
|
|
// remove the new ones that are identical to the baseline |
221
|
|
|
foreach ($new->getAttributes("eap:ca_file") as $newCA) { |
222
|
|
|
$ca = $x509->processCertificate($newCA['value']); |
223
|
|
|
if (array_key_exists($ca['sha1'], $baselineCA) || $ca['root'] != 1) { |
224
|
|
|
// do nothing; we assume here that SHA1 doesn't clash |
225
|
|
|
continue; |
226
|
|
|
} |
227
|
|
|
// check if a CA with identical DN was added - alert NRO if so |
228
|
|
|
$foundSHA1 = array_search($ca['name'], $baselineCA); |
229
|
|
|
if ($foundSHA1 !== FALSE) { |
230
|
|
|
// but only if the public key does not match |
231
|
|
|
if ($baselineCApublicKey[$foundSHA1] === $ca['full_details']['public_key']) { |
232
|
|
|
continue; |
233
|
|
|
} |
234
|
|
|
$retval[AbstractProfile::CA_CLASH_ADDED] .= "#SHA1 for CA with DN '".$ca['name']."' has SHA1 fingerprints (pre-existing) "./** @scrutinizer ignore-type */ array_search($ca['name'], $baselineCA)." and (added) ".$ca['sha1']; |
235
|
|
|
} else { |
236
|
|
|
$retval[AbstractProfile::CA_ADDED] .= "#CA with DN '"./** @scrutinizer ignore-type */ print_r($ca['name'], TRUE)."' and SHA1 fingerprint ".$ca['sha1']." was added as trust anchor"; |
237
|
|
|
} |
238
|
|
|
} |
239
|
|
|
// check if a servername was added |
240
|
|
|
$baselineNames = []; |
241
|
|
|
foreach ($old->getAttributes("eap:server_name") as $oldName) { |
242
|
|
|
$baselineNames[] = $oldName['value']; |
243
|
|
|
} |
244
|
|
|
foreach ($new->getAttributes("eap:server_name") as $newName) { |
245
|
|
|
if (!in_array($newName['value'], $baselineNames)) { |
246
|
|
|
$retval[AbstractProfile::SERVERNAME_ADDED] .= "#New server name '".$newName['value']."' added"; |
247
|
|
|
} |
248
|
|
|
} |
249
|
|
|
return $retval; |
250
|
|
|
} |
251
|
|
|
|
252
|
|
|
/** |
253
|
|
|
* Tests OpenRoaming aspects of the profile like DNS settings and server reachibility |
254
|
|
|
* |
255
|
|
|
* @return array of arrays of the form [['level' => $level, 'explanation' => $explanation, 'reason' => $reason]]; |
256
|
|
|
*/ |
257
|
|
|
public function openroamingReadinessTest() { |
258
|
|
|
// do OpenRoaming initial diagnostic checks |
259
|
|
|
// numbers correspond to RFC7585Tests::OVERALL_LEVEL |
260
|
|
|
$results = []; |
261
|
|
|
$resultLevel = $this::OVERALL_OPENROAMING_LEVEL_GOOD; // assume all is well, degrade if we have concrete findings to suggest otherwise |
262
|
|
|
$tag = "aaa+auth:radius.tls.tcp"; |
263
|
|
|
// do we know the realm at all? Notice if not. |
264
|
|
|
if (!isset($this->getAttributes("internal:realm")[0]['value'])) { |
265
|
|
|
$explanation = _("The profile information does not include the realm, so no DNS checks for OpenRoaming can be executed."); |
266
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_NOTE; |
267
|
|
|
$reason = $this::OPENROAMING_NO_REALM; |
268
|
|
|
$results[] = ['level' => $level, 'explanation' => $explanation, 'reason' => $reason]; |
269
|
|
|
$resultLevel = min([$resultLevel, $level]); |
270
|
|
|
} else { |
271
|
|
|
$dnsChecks = new \core\diag\RFC7585Tests($this->getAttributes("internal:realm")[0]['value'], $tag); |
272
|
|
|
$relevantNaptrRecords = $dnsChecks->relevantNAPTR(); |
273
|
|
|
if ($relevantNaptrRecords <= 0) { |
274
|
|
|
$explanation = _("There is no relevant DNS NAPTR record ($tag) for this realm. OpenRoaming will not work."); |
275
|
|
|
$reason = $this::OPENROAMING_NO_NAPTR; |
276
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_ERROR; |
277
|
|
|
$results[] = ['level' => $level, 'explanation' => $explanation, 'reason' => $reason]; |
278
|
|
|
$resultLevel = min([$resultLevel, $level]); |
279
|
|
|
} else { |
280
|
|
|
$recordCompliance = $dnsChecks->relevantNAPTRcompliance(); |
281
|
|
|
if ($recordCompliance != \core\diag\AbstractTest::RETVAL_OK) { |
282
|
|
|
$explanation = _("The DNS NAPTR record ($tag) for this realm is not syntax conform. OpenRoaming will likely not work."); |
283
|
|
|
$reason = $this::OPENROAMING_BAD_NAPTR; |
284
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_WARN; |
285
|
|
|
$results[] = ['level' => $level, 'explanation' => $explanation, 'reason' => $reason]; |
286
|
|
|
$resultLevel = min([$resultLevel, $level]); |
287
|
|
|
} |
288
|
|
|
// check if target is the expected one, if set by NRO |
289
|
|
|
foreach ($this->fedAttributes as $attr) { |
290
|
|
|
if ($attr['name'] === 'fed:openroaming_customtarget') { |
291
|
|
|
$customText = $attr['value']; |
292
|
|
|
} else { |
293
|
|
|
$customText = ''; |
294
|
|
|
} |
295
|
|
|
} |
296
|
|
|
if ($customText !== '') { |
|
|
|
|
297
|
|
|
foreach ($dnsChecks->NAPTR_records as $orpointer) { |
298
|
|
|
if ($orpointer["replacement"] != $customText) { |
299
|
|
|
$explanation = _("The SRV target of an OpenRoaming NAPTR record is unexpected."); |
300
|
|
|
$reason = $this::OPENROAMING_BAD_SRV; |
301
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_NOTE; |
302
|
|
|
$results[] = ['level' => $level, 'explanation' => $explanation, 'reason' => $reason]; |
303
|
|
|
$resultLevel = min([$resultLevel, $level]); |
304
|
|
|
} |
305
|
|
|
} |
306
|
|
|
} |
307
|
|
|
$srvResolution = $dnsChecks->relevantNAPTRsrvResolution(); |
308
|
|
|
$hostnameResolution = $dnsChecks->relevantNAPTRhostnameResolution(); |
309
|
|
|
|
310
|
|
|
if ($srvResolution <= 0) { |
311
|
|
|
$explanation = _("The DNS SRV target for NAPTR $tag does not resolve. OpenRoaming will not work."); |
312
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_ERROR; |
313
|
|
|
$reason = $this::OPENROAMING_BAD_NAPTR_RESOLVE; |
314
|
|
|
$results[] = ['level' => $level, 'explanation' => $explanation, 'reason' => $reason]; |
315
|
|
|
$resultLevel = min([$resultLevel, $this::OVERALL_OPENROAMING_LEVEL_ERROR]); |
316
|
|
|
} elseif ($hostnameResolution <= 0) { |
317
|
|
|
$explanation = _("The DNS hostnames in the SRV records do not resolve to actual host IPs. OpenRoaming will not work."); |
318
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_ERROR; |
319
|
|
|
$reason = $this::OPENROAMING_BAD_SRV_RESOLVE; |
320
|
|
|
$results[] = ['level' => $level, 'explanation' => $explanation, 'reason' => $reason]; |
321
|
|
|
$resultLevel = min([$resultLevel, $level]); |
322
|
|
|
} |
323
|
|
|
// connect to all IPs we found and see if they are really an OpenRoaming server |
324
|
|
|
$allHostsOkay = TRUE; |
325
|
|
|
$oneHostOkay = FALSE; |
326
|
|
|
$testCandidates = []; |
327
|
|
|
foreach ($dnsChecks->NAPTR_hostname_records as $oneServer) { |
328
|
|
|
$testCandidates[$oneServer['hostname']][] = ($oneServer['family'] == "IPv4" ? $oneServer['IP'] : "[".$oneServer['IP']."]").":".$oneServer['port']; |
329
|
|
|
} |
330
|
|
|
foreach ($testCandidates as $oneHost => $listOfIPs) { |
331
|
|
|
$connectionTests = new \core\diag\RFC6614Tests(array_values($listOfIPs), $oneHost, "openroaming"); |
332
|
|
|
// for now (no OpenRoaming client certs available) only run server-side tests |
333
|
|
|
foreach ($listOfIPs as $oneIP) { |
334
|
|
|
$connectionResult = $connectionTests->cApathCheck($oneIP); |
335
|
|
|
if ($connectionResult != \core\diag\AbstractTest::RETVAL_OK || ( isset($connectionTests->TLS_CA_checks_result['cert_oddity']) && count($connectionTests->TLS_CA_checks_result['cert_oddity']) > 0)) { |
336
|
|
|
$allHostsOkay = FALSE; |
337
|
|
|
} else { |
338
|
|
|
$oneHostOkay = TRUE; |
339
|
|
|
} |
340
|
|
|
} |
341
|
|
|
} |
342
|
|
|
if (!$allHostsOkay) { |
343
|
|
|
if (!$oneHostOkay) { |
344
|
|
|
$explanation = _("When connecting to the discovered OpenRoaming endpoints, they all had errors. OpenRoaming will likely not work."); |
345
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_ERROR; |
346
|
|
|
$reason = $this::OPENROAMING_BAD_CONNECTION; |
347
|
|
|
$results[] = ['level' => $level, 'explanation' => $explanation, 'reason' => $reason]; |
348
|
|
|
$resultLevel = min([$resultLevel, $level]); |
349
|
|
|
} else { |
350
|
|
|
$explanation = _("When connecting to the discovered OpenRoaming endpoints, only a subset of endpoints had no errors."); |
351
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_WARN; |
352
|
|
|
$reason = $this::OPENROAMING_SOME_BAD_CONNECTIONS; |
353
|
|
|
$results[] = ['level' => $level, 'explanation' => $explanation, 'reason' => $reason]; |
354
|
|
|
$resultLevel = min([$resultLevel, $level]); |
355
|
|
|
} |
356
|
|
|
} |
357
|
|
|
} |
358
|
|
|
} |
359
|
|
|
if (!$dnsChecks->allResponsesSecure) { |
|
|
|
|
360
|
|
|
$explanation = _("At least one DNS response was NOT secured using DNSSEC. OpenRoaming ANPs may refuse to connect to the endpoint."); |
361
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_WARN; |
362
|
|
|
$reason = $this::OPENROAMING_NO_DNSSEC; |
363
|
|
|
$results[] = ['level' => $level, 'explanation' => $explanation, 'reason' => $reason]; |
364
|
|
|
$resultLevel = min([$resultLevel, $level]); |
365
|
|
|
} |
366
|
|
|
if ($resultLevel == $this::OVERALL_OPENROAMING_LEVEL_GOOD) { |
367
|
|
|
$explanation = _("Initial diagnostics regarding the DNS part of OpenRoaming (including DNSSEC) were successful."); |
368
|
|
|
$level = $this::OVERALL_OPENROAMING_LEVEL_GOOD; |
369
|
|
|
$reason = $this::OPENROAMING_ALL_GOOD; |
370
|
|
|
$results = [['level' => $level, 'explanation' => $explanation, 'reason' => $reason]]; |
371
|
|
|
} |
372
|
|
|
$this->setOpenRoamingReadinessInfo($resultLevel); |
373
|
|
|
return $results; |
374
|
|
|
} |
375
|
|
|
|
376
|
|
|
/** |
377
|
|
|
* Takes note of the OpenRoaming participation and conformance level |
378
|
|
|
* |
379
|
|
|
* @param int $level the readiness level, as determined by RFC7585Tests |
380
|
|
|
* @return void |
381
|
|
|
*/ |
382
|
|
|
public function setOpenRoamingReadinessInfo(int $level) |
383
|
|
|
{ |
384
|
|
|
$this->databaseHandle->exec("UPDATE profile SET openroaming = ? WHERE profile_id = ?", "ii", $level, $this->identifier); |
385
|
|
|
} |
386
|
|
|
|
387
|
|
|
/** |
388
|
|
|
* each profile has supported EAP methods, so get this from DB, Silver Bullet has one |
389
|
|
|
* static EAP method. |
390
|
|
|
* |
391
|
|
|
* @return array list of supported EAP methods |
392
|
|
|
*/ |
393
|
|
|
protected function fetchEAPMethods() |
394
|
|
|
{ |
395
|
|
|
$eapMethod = $this->databaseHandle->exec("SELECT eap_method_id |
396
|
|
|
FROM supported_eap supp |
397
|
|
|
WHERE supp.profile_id = $this->identifier |
398
|
|
|
ORDER by preference"); |
399
|
|
|
$eapTypeArray = []; |
400
|
|
|
// SELECTs never return a boolean, it's always a resource |
401
|
|
|
while ($eapQuery = (mysqli_fetch_object(/** @scrutinizer ignore-type */ $eapMethod))) { |
402
|
|
|
$eaptype = new common\EAP($eapQuery->eap_method_id); |
403
|
|
|
$eapTypeArray[] = $eaptype; |
404
|
|
|
} |
405
|
|
|
$this->loggerInstance->debug(4, "This profile supports the following EAP types:\n"./** @scrutinizer ignore-type */ print_r($eapTypeArray, true)); |
406
|
|
|
return $eapTypeArray; |
407
|
|
|
} |
408
|
|
|
|
409
|
|
|
/** |
410
|
|
|
* Class constructor for existing profiles (use IdP::newProfile() to actually create one). Retrieves all attributes and |
411
|
|
|
* supported EAP types from the DB and stores them in the priv_ arrays. |
412
|
|
|
* |
413
|
|
|
* sub-classes need to set the property $realm, $name themselves! |
414
|
|
|
* |
415
|
|
|
* @param int $profileIdRaw identifier of the profile in the DB |
416
|
|
|
* @param IdP $idpObject optionally, the institution to which this Profile belongs. Saves the construction of the IdP instance. If omitted, an extra query and instantiation is executed to find out. |
417
|
|
|
* @throws Exception |
418
|
|
|
*/ |
419
|
|
|
public function __construct($profileIdRaw, $idpObject = NULL) |
420
|
|
|
{ |
421
|
|
|
$this->databaseType = "INST"; |
422
|
|
|
parent::__construct(); // we now have access to our INST database handle and logging |
423
|
|
|
$handle = DBConnection::handle("FRONTEND"); |
424
|
|
|
if ($handle instanceof DBConnection) { |
425
|
|
|
$this->frontendHandle = $handle; |
426
|
|
|
} else { |
427
|
|
|
throw new Exception("This database type is never an array!"); |
428
|
|
|
} |
429
|
|
|
$profile = $this->databaseHandle->exec("SELECT inst_id FROM profile WHERE profile_id = ?", "i", $profileIdRaw); |
430
|
|
|
// SELECT always yields a resource, never a boolean |
431
|
|
|
if ($profile->num_rows == 0) { |
432
|
|
|
$this->loggerInstance->debug(2, "Profile $profileIdRaw not found in database!\n"); |
433
|
|
|
throw new Exception("Profile $profileIdRaw not found in database!"); |
434
|
|
|
} |
435
|
|
|
$this->identifier = $profileIdRaw; |
436
|
|
|
$profileQuery = mysqli_fetch_object(/** @scrutinizer ignore-type */ $profile); |
437
|
|
|
if (!($idpObject instanceof IdP)) { |
438
|
|
|
$this->institution = $profileQuery->inst_id; |
439
|
|
|
$idp = new IdP($this->institution); |
440
|
|
|
} else { |
441
|
|
|
$idp = $idpObject; |
442
|
|
|
$this->institution = $idp->identifier; |
443
|
|
|
} |
444
|
|
|
|
445
|
|
|
$this->instName = $idp->name; |
446
|
|
|
$this->idpNumberOfProfiles = $idp->profileCount(); |
447
|
|
|
$this->idpAttributes = $idp->getAttributes(); |
448
|
|
|
$fedObject = new Federation($idp->federation); |
449
|
|
|
$this->fedAttributes = $fedObject->getAttributes(); |
450
|
|
|
$this->loggerInstance->debug(4, "--- END Constructing new AbstractProfile object ... ---\n"); |
451
|
|
|
} |
452
|
|
|
|
453
|
|
|
/** |
454
|
|
|
* find a profile, given its realm |
455
|
|
|
* |
456
|
|
|
* @param string $realm the realm for which we are trying to find a profile |
457
|
|
|
* @return int|false the profile identifier, if any |
458
|
|
|
*/ |
459
|
|
|
public static function profileFromRealm($realm) |
460
|
|
|
{ |
461
|
|
|
// static, need to create our own handle |
462
|
|
|
$handle = DBConnection::handle("INST"); |
463
|
|
|
$execQuery = $handle->exec("SELECT profile_id FROM profile WHERE realm LIKE '%@$realm'"); |
464
|
|
|
// a SELECT query always yields a resource, not a boolean |
465
|
|
|
if ($profileIdQuery = mysqli_fetch_object(/** @scrutinizer ignore-type */ $execQuery)) { |
466
|
|
|
return $profileIdQuery->profile_id; |
467
|
|
|
} |
468
|
|
|
return FALSE; |
469
|
|
|
} |
470
|
|
|
|
471
|
|
|
/** |
472
|
|
|
* Constructs the outer ID which should be used during realm tests. Obviously |
473
|
|
|
* can only do something useful if the realm is known to the system. |
474
|
|
|
* |
475
|
|
|
* @return string the outer ID to use for realm check operations |
476
|
|
|
* @throws Exception |
477
|
|
|
*/ |
478
|
|
|
public function getRealmCheckOuterUsername() |
479
|
|
|
{ |
480
|
|
|
$realm = $this->getAttributes("internal:realm")[0]['value'] ?? FALSE; |
481
|
|
|
if ($realm == FALSE) { // we can't really return anything useful here |
482
|
|
|
throw new Exception("Unable to construct a realmcheck username if the admin did not tell us the realm. You shouldn't have called this function in this context."); |
483
|
|
|
} |
484
|
|
|
if (count($this->getAttributes("internal:checkuser_outer")) > 0 && $this->getAttributes("internal:checkuser_value")[0]['value'] != NULL) { |
485
|
|
|
// we are supposed to use a specific outer username for checks, |
486
|
|
|
// which is different from the outer username we put into installers |
487
|
|
|
return $this->getAttributes("internal:checkuser_value")[0]['value']."@".$realm; |
488
|
|
|
} |
489
|
|
|
if (count($this->getAttributes("internal:use_anon_outer")) > 0 && $this->getAttributes("internal:anon_local_value")[0]['value'] != NULL ) { |
490
|
|
|
// no special check username, but there is an anon outer ID for |
491
|
|
|
// installers - so let's use that one |
492
|
|
|
return $this->getAttributes("internal:anon_local_value")[0]['value']."@".$realm; |
493
|
|
|
} |
494
|
|
|
// okay, no guidance on outer IDs at all - but we need *something* to |
495
|
|
|
// test with for the RealmChecks. So: |
496
|
|
|
return "@".$realm; |
497
|
|
|
} |
498
|
|
|
|
499
|
|
|
/** |
500
|
|
|
* update the last_changed timestamp for this profile |
501
|
|
|
* |
502
|
|
|
* @return void |
503
|
|
|
*/ |
504
|
|
|
public function updateFreshness() |
505
|
|
|
{ |
506
|
|
|
$this->databaseHandle->exec("UPDATE profile SET last_change = CURRENT_TIMESTAMP WHERE profile_id = $this->identifier"); |
507
|
|
|
} |
508
|
|
|
|
509
|
|
|
/** |
510
|
|
|
* gets the last-modified timestamp (useful for caching "dirty" check) |
511
|
|
|
* |
512
|
|
|
* @return string the date in string form, as returned by SQL |
513
|
|
|
*/ |
514
|
|
|
public function getFreshness() |
515
|
|
|
{ |
516
|
|
|
$execLastChange = $this->databaseHandle->exec("SELECT last_change FROM profile WHERE profile_id = $this->identifier"); |
517
|
|
|
// SELECT always returns a resource, never a boolean |
518
|
|
|
if ($freshnessQuery = mysqli_fetch_object(/** @scrutinizer ignore-type */ $execLastChange)) { |
519
|
|
|
return $freshnessQuery->last_change; |
520
|
|
|
} |
521
|
|
|
} |
522
|
|
|
/** |
523
|
|
|
* tests if the configurator needs to be regenerated |
524
|
|
|
* returns the configurator path or NULL if regeneration is required |
525
|
|
|
*/ |
526
|
|
|
/** |
527
|
|
|
* This function tests if the configurator needs to be regenerated |
528
|
|
|
* (properties of the Profile may have changed since the last configurator |
529
|
|
|
* generation). |
530
|
|
|
* SilverBullet will always return NULL here because all installers are new! |
531
|
|
|
* |
532
|
|
|
* @param string $device device ID to check |
533
|
|
|
* @return mixed a string with the path to the configurator download, or NULL if it needs to be regenerated |
534
|
|
|
*/ |
535
|
|
|
|
536
|
|
|
/** |
537
|
|
|
* This function tests if the configurator needs to be regenerated (properties of the Profile may have changed since the last configurator generation). |
538
|
|
|
* |
539
|
|
|
* @param string $device device ID to check |
540
|
|
|
* @return mixed a string with the path to the configurator download, or NULL if it needs to be regenerated |
541
|
|
|
*/ |
542
|
|
|
public function testCache($device, $openRoaming) |
543
|
|
|
{ |
544
|
|
|
$returnValue = ['cache' => NULL, 'mime' => NULL]; |
545
|
|
|
$lang = $this->languageInstance->getLang(); |
546
|
|
|
$result = $this->frontendHandle->exec("SELECT download_path, mime, UNIX_TIMESTAMP(installer_time) AS tm FROM downloads WHERE profile_id = ? AND device_id = ? AND lang = ? AND openroaming = ?", "issi", $this->identifier, $device, $lang, $openRoaming); |
547
|
|
|
// SELECT queries always return a resource, not a boolean |
548
|
|
|
if ($result && $cache = mysqli_fetch_object(/** @scrutinizer ignore-type */ $result)) { |
549
|
|
|
$execUpdate = $this->databaseHandle->exec("SELECT UNIX_TIMESTAMP(last_change) AS last_change FROM profile WHERE profile_id = $this->identifier"); |
550
|
|
|
// SELECT queries always return a resource, not a boolean |
551
|
|
|
if ($lastChange = mysqli_fetch_object(/** @scrutinizer ignore-type */ $execUpdate)->last_change) { |
552
|
|
|
if ($lastChange < $cache->tm) { |
553
|
|
|
$this->loggerInstance->debug(4, "Installer cached:$cache->download_path\n"); |
554
|
|
|
$returnValue = ['cache' => $cache->download_path, 'mime' => $cache->mime]; |
555
|
|
|
} |
556
|
|
|
} |
557
|
|
|
} |
558
|
|
|
return $returnValue; |
559
|
|
|
} |
560
|
|
|
|
561
|
|
|
/** |
562
|
|
|
* Updates database with new installer location. Actually does stuff when |
563
|
|
|
* caching is possible; is a noop if not |
564
|
|
|
* |
565
|
|
|
* @param string $device the device identifier string |
566
|
|
|
* @param string $path the path where the new installer can be found |
567
|
|
|
* @param string $mime the mime type of the new installer |
568
|
|
|
* @param int $integerEapType the inter-representation of the EAP type that is configured in this installer |
569
|
|
|
* @return void |
570
|
|
|
*/ |
571
|
|
|
abstract public function updateCache($device, $path, $mime, $integerEapType, $openRoaming); |
572
|
|
|
|
573
|
|
|
/** Toggle anonymous outer ID support. |
574
|
|
|
* |
575
|
|
|
* @param boolean $shallwe TRUE to enable outer identities (needs valid $realm), FALSE to disable |
576
|
|
|
* @return void |
577
|
|
|
*/ |
578
|
|
|
abstract public function setAnonymousIDSupport($shallwe); |
579
|
|
|
|
580
|
|
|
/** |
581
|
|
|
* Log a new download for our stats |
582
|
|
|
* |
583
|
|
|
* @param string $device the device id string |
584
|
|
|
* @param string $area either admin or user |
585
|
|
|
* @return boolean TRUE if incrementing worked, FALSE if not |
586
|
|
|
*/ |
587
|
|
|
public function incrementDownloadStats($device, $area, $openRoaming) |
588
|
|
|
{ |
589
|
|
|
if ($area == "admin" || $area == "user" || $area == "silverbullet") { |
590
|
|
|
$lang = $this->languageInstance->getLang(); |
591
|
|
|
$this->frontendHandle->exec("INSERT INTO downloads (profile_id, device_id, lang, openroaming, downloads_$area) VALUES (? ,?, ?, ?, 1) ON DUPLICATE KEY UPDATE downloads_$area = downloads_$area + 1", "issi", $this->identifier, $device, $lang, $openRoaming); |
592
|
|
|
$this->frontendHandle->exec("INSERT INTO downloads_history (profile_id, device_id, downloads_$area, openroaming, stat_date) VALUES (?, ?, 1, ?, DATE_FORMAT(NOW(), '%Y-%m-01')) ON DUPLICATE KEY UPDATE downloads_$area = downloads_$area + 1", "isi", $this->identifier, $device, $openRoaming); |
593
|
|
|
// get eap_type from the downloads table |
594
|
|
|
$eapTypeQuery = $this->frontendHandle->exec("SELECT eap_type FROM downloads WHERE profile_id = ? AND device_id = ? AND lang = ?", "iss", $this->identifier, $device, $lang); |
595
|
|
|
// SELECT queries always return a resource, not a boolean |
596
|
|
|
if (!$eapTypeQuery || !$eapO = mysqli_fetch_object(/** @scrutinizer ignore-type */ $eapTypeQuery)) { |
597
|
|
|
$this->loggerInstance->debug(2, "Error getting EAP_type from the database\n"); |
598
|
|
|
} else { |
599
|
|
|
if ($eapO->eap_type == NULL) { |
600
|
|
|
$this->loggerInstance->debug(2, "EAP_type not set in the database\n"); |
601
|
|
|
} else { |
602
|
|
|
$this->saveDownloadDetails($this->institution, $this->identifier, $device, $area, $this->languageInstance->getLang(), $eapO->eap_type, $openRoaming); |
603
|
|
|
} |
604
|
|
|
} |
605
|
|
|
return TRUE; |
606
|
|
|
} |
607
|
|
|
return FALSE; |
608
|
|
|
} |
609
|
|
|
|
610
|
|
|
/** |
611
|
|
|
* Retrieve current download stats from database, either for one specific device or for all devices |
612
|
|
|
* |
613
|
|
|
* @param string $device the device id string |
614
|
|
|
* @return mixed user downloads of this profile; if device is given, returns the counter as int, otherwise an array with devicename => counter |
615
|
|
|
*/ |
616
|
|
|
public function getUserDownloadStats($device = NULL) |
617
|
|
|
{ |
618
|
|
|
$columnName = "downloads_user"; |
619
|
|
|
if ($this instanceof \core\ProfileSilverbullet) { |
620
|
|
|
$columnName = "downloads_silverbullet"; |
621
|
|
|
} |
622
|
|
|
$returnarray = []; |
623
|
|
|
$numbers = $this->frontendHandle->exec("SELECT device_id, SUM($columnName) AS downloads_user FROM downloads WHERE profile_id = ? GROUP BY device_id", "i", $this->identifier); |
624
|
|
|
// SELECT queries always return a resource, not a boolean |
625
|
|
|
while ($statsQuery = mysqli_fetch_object(/** @scrutinizer ignore-type */ $numbers)) { |
626
|
|
|
$returnarray[$statsQuery->device_id] = $statsQuery->downloads_user; |
627
|
|
|
} |
628
|
|
|
if ($device !== NULL) { |
629
|
|
|
if (isset($returnarray[$device])) { |
630
|
|
|
return $returnarray[$device]; |
631
|
|
|
} |
632
|
|
|
return 0; |
633
|
|
|
} |
634
|
|
|
// we should pretty-print the device names |
635
|
|
|
$finalarray = []; |
636
|
|
|
$devlist = \devices\Devices::listDevices($this->identifier); |
637
|
|
|
foreach ($returnarray as $devId => $count) { |
638
|
|
|
if (isset($devlist[$devId])) { |
639
|
|
|
$finalarray[$devlist[$devId]['display']]['current'] = $count; |
640
|
|
|
} |
641
|
|
|
|
642
|
|
|
} |
643
|
|
|
|
644
|
|
|
$monthlyList = []; |
645
|
|
|
$monthly = $this->frontendHandle->exec("SELECT downloads_user,device_id FROM downloads_history WHERE profile_id=? AND stat_date=DATE_FORMAT(NOW(),'%Y-%m-01')", "i", $this->identifier); |
646
|
|
|
while ($statsQuery = mysqli_fetch_object(/** @scrutinizer ignore-type */ $monthly)) { |
647
|
|
|
$monthlyList[$statsQuery->device_id] = $statsQuery->downloads_user; |
648
|
|
|
} |
649
|
|
|
foreach ($monthlyList as $devId => $count) { |
650
|
|
|
if (isset($devlist[$devId])) { |
651
|
|
|
$finalarray[$devlist[$devId]['display']]['monthly'] = $count; |
652
|
|
|
} |
653
|
|
|
|
654
|
|
|
} |
655
|
|
|
|
656
|
|
|
\core\common\Entity::intoThePotatoes(); |
657
|
|
|
ksort($finalarray, SORT_STRING|SORT_FLAG_CASE); |
658
|
|
|
\core\common\Entity::outOfThePotatoes(); |
659
|
|
|
return $finalarray; |
660
|
|
|
} |
661
|
|
|
|
662
|
|
|
/** |
663
|
|
|
* Deletes the profile from database and uninstantiates itself. |
664
|
|
|
* Works fine also for Silver Bullet; the first query will simply do nothing |
665
|
|
|
* because there are no stored options |
666
|
|
|
* |
667
|
|
|
* @return void |
668
|
|
|
*/ |
669
|
|
|
public function destroy() |
670
|
|
|
{ |
671
|
|
|
$this->databaseHandle->exec("DELETE FROM profile_option WHERE profile_id = $this->identifier"); |
672
|
|
|
$this->databaseHandle->exec("DELETE FROM supported_eap WHERE profile_id = $this->identifier"); |
673
|
|
|
$this->databaseHandle->exec("DELETE FROM profile WHERE profile_id = $this->identifier"); |
674
|
|
|
} |
675
|
|
|
|
676
|
|
|
/** |
677
|
|
|
* Specifies the realm of this profile. |
678
|
|
|
* |
679
|
|
|
* Forcefully type-hinting $realm parameter to string - Scrutinizer seems to |
680
|
|
|
* think that it can alternatively be an array<integer,?> which looks like a |
681
|
|
|
* false positive. If there really is an issue, let PHP complain about it at |
682
|
|
|
* runtime. |
683
|
|
|
* |
684
|
|
|
* @param string $realm the realm (potentially with the local@ part that should be used for anonymous identities) |
685
|
|
|
* @return void |
686
|
|
|
*/ |
687
|
|
|
public function setRealm(string $realm) |
688
|
|
|
{ |
689
|
|
|
$this->databaseHandle->exec("UPDATE profile SET realm = ? WHERE profile_id = ?", "si", $realm, $this->identifier); |
690
|
|
|
$this->realm = $realm; |
691
|
|
|
} |
692
|
|
|
|
693
|
|
|
/** |
694
|
|
|
* register new supported EAP method for this profile |
695
|
|
|
* |
696
|
|
|
* @param \core\common\EAP $type The EAP Type, as defined in class EAP |
697
|
|
|
* @param int $preference preference of this EAP Type. If a preference value is re-used, the order of EAP types of the same preference level is undefined. |
698
|
|
|
* @return void |
699
|
|
|
*/ |
700
|
|
|
public function addSupportedEapMethod(\core\common\EAP $type, $preference) |
701
|
|
|
{ |
702
|
|
|
$eapInt = $type->getIntegerRep(); |
703
|
|
|
$this->databaseHandle->exec("INSERT INTO supported_eap (profile_id, eap_method_id, preference) VALUES (?, ?, ?)", "iii", $this->identifier, $eapInt, $preference); |
704
|
|
|
$this->updateFreshness(); |
705
|
|
|
} |
706
|
|
|
|
707
|
|
|
/** |
708
|
|
|
* Produces an array of EAP methods supported by this profile, ordered by preference |
709
|
|
|
* |
710
|
|
|
* @param int $completeOnly if set and non-zero limits the output to methods with complete information |
711
|
|
|
* @return array list of EAP methods, (in object representation) |
712
|
|
|
*/ |
713
|
|
|
public function getEapMethodsinOrderOfPreference($completeOnly = 0) |
714
|
|
|
{ |
715
|
|
|
$temparray = []; |
716
|
|
|
|
717
|
|
|
if ($completeOnly == 0) { |
718
|
|
|
return $this->privEaptypes; |
719
|
|
|
} |
720
|
|
|
foreach ($this->privEaptypes as $type) { |
721
|
|
|
if ($this->isEapTypeDefinitionComplete($type) === true) { |
722
|
|
|
$temparray[] = $type; |
723
|
|
|
} |
724
|
|
|
} |
725
|
|
|
return($temparray); |
726
|
|
|
} |
727
|
|
|
|
728
|
|
|
/** |
729
|
|
|
* Performs a sanity check for a given EAP type - did the admin submit enough information to create installers for him? |
730
|
|
|
* |
731
|
|
|
* @param common\EAP $eaptype the EAP type |
732
|
|
|
* @return mixed TRUE if the EAP type is complete; an array of missing attributes if it's incomplete; FALSE if it's incomplete for other reasons |
733
|
|
|
*/ |
734
|
|
|
public function isEapTypeDefinitionComplete($eaptype) |
735
|
|
|
{ |
736
|
|
|
if ($eaptype->needsServerCACert() && $eaptype->needsServerName()) { |
737
|
|
|
$missing = []; |
738
|
|
|
// silverbullet needs a support email address configured |
739
|
|
|
if ($eaptype->getIntegerRep() == common\EAP::INTEGER_SILVERBULLET && count($this->getAttributes("support:email")) == 0) { |
740
|
|
|
return ["support:email"]; |
741
|
|
|
} |
742
|
|
|
$cnOption = $this->getAttributes("eap:server_name"); // cannot be set per device or eap type |
743
|
|
|
$caOption = $this->getAttributes("eap:ca_file"); // cannot be set per device or eap type |
744
|
|
|
|
745
|
|
|
if (count($caOption) > 0 && count($cnOption) > 0) {// see if we have at least one root CA cert |
746
|
|
|
foreach ($caOption as $oneCa) { |
747
|
|
|
$x509 = new \core\common\X509(); |
748
|
|
|
$caParsed = $x509->processCertificate($oneCa['value']); |
749
|
|
|
if ($caParsed['root'] == 1) { |
750
|
|
|
return TRUE; |
751
|
|
|
} |
752
|
|
|
} |
753
|
|
|
$missing[] = "eap:ca_file"; |
754
|
|
|
} |
755
|
|
|
if (count($caOption) == 0) { |
756
|
|
|
$missing[] = "eap:ca_file"; |
757
|
|
|
} |
758
|
|
|
if (count($cnOption) == 0) { |
759
|
|
|
$missing[] = "eap:server_name"; |
760
|
|
|
} |
761
|
|
|
if (count($missing) == 0) { |
762
|
|
|
return TRUE; |
763
|
|
|
} |
764
|
|
|
return $missing; |
765
|
|
|
} |
766
|
|
|
return TRUE; |
767
|
|
|
} |
768
|
|
|
|
769
|
|
|
/** |
770
|
|
|
* list all devices marking their availabiblity and possible redirects |
771
|
|
|
* |
772
|
|
|
* @return array of device ids display names and their status |
773
|
|
|
*/ |
774
|
|
|
public function listDevices() |
775
|
|
|
{ |
776
|
|
|
$returnarray = []; |
777
|
|
|
$redirect = $this->getAttributes("device-specific:redirect"); // this might return per-device ones or the general one |
778
|
|
|
// if it was a general one, we are done. Find out if there is one such |
779
|
|
|
// which has device = NULL |
780
|
|
|
$generalRedirect = NULL; |
781
|
|
|
foreach ($redirect as $index => $oneRedirect) { |
782
|
|
|
if ($oneRedirect["level"] == Options::LEVEL_PROFILE) { |
783
|
|
|
$generalRedirect = $index; |
784
|
|
|
} |
785
|
|
|
} |
786
|
|
|
if ($generalRedirect !== NULL) { // could be index 0 |
787
|
|
|
return [['id' => '0', 'redirect' => $redirect[$generalRedirect]['value']]]; |
788
|
|
|
} |
789
|
|
|
$preferredEap = $this->getEapMethodsinOrderOfPreference(1); |
790
|
|
|
$eAPOptions = []; |
791
|
|
|
if (count($this->getAttributes("media:openroaming")) == 1 && $this->getAttributes("media:openroaming")[0]['value'] == 'always-preagreed') { |
792
|
|
|
$orAlways = 1; |
793
|
|
|
} else { |
794
|
|
|
$orAlways = 0; |
795
|
|
|
} |
796
|
|
|
|
797
|
|
|
foreach (\devices\Devices::listDevices($this->identifier, $orAlways) as $deviceIndex => $deviceProperties) { |
798
|
|
|
$factory = new DeviceFactory($deviceIndex); |
799
|
|
|
$dev = $factory->device; |
800
|
|
|
// find the attribute pertaining to the specific device |
801
|
|
|
$group = ''; |
802
|
|
|
$redirectUrl = 0; |
803
|
|
|
$redirects = []; |
804
|
|
|
foreach ($redirect as $index => $oneRedirect) { |
805
|
|
|
if ($oneRedirect["device"] == $deviceIndex) { |
806
|
|
|
$redirects[] = $oneRedirect; |
807
|
|
|
} |
808
|
|
|
} |
809
|
|
|
if (count($redirects) > 0) { |
810
|
|
|
$redirectUrl = $this->languageInstance->getLocalisedValue($redirects); |
811
|
|
|
} |
812
|
|
|
$devStatus = self::AVAILABLE; |
813
|
|
|
$message = 0; |
814
|
|
|
if (isset($deviceProperties['options']) && isset($deviceProperties['options']['message']) && $deviceProperties['options']['message']) { |
815
|
|
|
$message = $deviceProperties['options']['message']; |
816
|
|
|
} |
817
|
|
|
if (isset($deviceProperties['group'])) { |
818
|
|
|
$group = $deviceProperties['group']; |
819
|
|
|
} |
820
|
|
|
$eapCustomtext = 0; |
821
|
|
|
$deviceCustomtext = 0; |
822
|
|
|
$geteduroam = 0; |
823
|
|
|
if ($redirectUrl === 0) { |
824
|
|
|
if (isset($deviceProperties['options']) && isset($deviceProperties['options']['redirect']) && $deviceProperties['options']['redirect']) { |
825
|
|
|
$devStatus = self::HIDDEN; |
826
|
|
|
} else { |
827
|
|
|
$dev->calculatePreferredEapType($preferredEap); |
828
|
|
|
$eap = $dev->selectedEap; |
829
|
|
|
if (count($eap) > 0) { |
830
|
|
|
if (isset($eAPOptions["eap-specific:customtext"][serialize($eap)])) { |
831
|
|
|
$eapCustomtext = $eAPOptions["eap-specific:customtext"][serialize($eap)]; |
832
|
|
|
} else { |
833
|
|
|
// fetch customtexts from method-level attributes |
834
|
|
|
$eapCustomtext = 0; |
835
|
|
|
$customTextAttributes = []; |
836
|
|
|
$attributeList = $this->getAttributes("eap-specific:customtext"); // eap-specific attributes always have the array index 'eapmethod' set |
837
|
|
|
foreach ($attributeList as $oneAttribute) { |
838
|
|
|
if ($oneAttribute["eapmethod"] == $eap) { |
839
|
|
|
$customTextAttributes[] = $oneAttribute; |
840
|
|
|
} |
841
|
|
|
} |
842
|
|
|
if (count($customTextAttributes) > 0) { |
843
|
|
|
$eapCustomtext = $this->languageInstance->getLocalisedValue($customTextAttributes); |
844
|
|
|
} |
845
|
|
|
$eAPOptions["eap-specific:customtext"][serialize($eap)] = $eapCustomtext; |
846
|
|
|
} |
847
|
|
|
// fetch customtexts for device |
848
|
|
|
$customTextAttributes = []; |
849
|
|
|
$attributeList = $this->getAttributes("device-specific:customtext"); |
850
|
|
|
foreach ($attributeList as $oneAttribute) { |
851
|
|
|
if ($oneAttribute["device"] == $deviceIndex) { // device-specific attributes always have the array index "device" set |
852
|
|
|
$customTextAttributes[] = $oneAttribute; |
853
|
|
|
} |
854
|
|
|
} |
855
|
|
|
$deviceCustomtext = $this->languageInstance->getLocalisedValue($customTextAttributes); |
856
|
|
|
} else { |
857
|
|
|
$devStatus = self::UNAVAILABLE; |
858
|
|
|
} |
859
|
|
|
$geteduroamOpts = $this->getAttributes("device-specific:geteduroam"); |
860
|
|
|
foreach ($geteduroamOpts as $dev) { |
861
|
|
|
if ($dev['device'] == $deviceIndex) { |
862
|
|
|
$geteduroam = $dev['value'] == 'on' ? 1 : 0; |
863
|
|
|
} |
864
|
|
|
} |
865
|
|
|
} |
866
|
|
|
} |
867
|
|
|
$returnarray[] = ['id' => $deviceIndex, 'display' => $deviceProperties['display'], 'status' => $devStatus, 'redirect' => $redirectUrl, 'eap_customtext' => $eapCustomtext, 'device_customtext' => $deviceCustomtext, 'message' => $message, 'options' => $deviceProperties['options'], 'group' => $group, 'geteduroam' => $geteduroam]; |
868
|
|
|
} |
869
|
|
|
return $returnarray; |
870
|
|
|
} |
871
|
|
|
|
872
|
|
|
/** |
873
|
|
|
* prepare profile attributes for device modules |
874
|
|
|
* Gets profile attributes taking into account the most specific level on which they may be defined |
875
|
|
|
* as well as the chosen language. |
876
|
|
|
* can be called with an optional $eap argument |
877
|
|
|
* |
878
|
|
|
* @param array $eap if specified, retrieves all attributes except those not pertaining to the given EAP type |
879
|
|
|
* @return array list of attributes in collapsed style (index is the attrib name, value is an array of different values) |
880
|
|
|
*/ |
881
|
|
|
public function getCollapsedAttributes($eap = []) |
882
|
|
|
{ |
883
|
|
|
$collapsedList = []; |
884
|
|
|
foreach ($this->getAttributes() as $attribute) { |
885
|
|
|
// filter out eap-level attributes not pertaining to EAP type $eap |
886
|
|
|
if (count($eap) > 0 && isset($attribute['eapmethod']) && $attribute['eapmethod'] != 0 && $attribute['eapmethod'] != $eap) { |
887
|
|
|
continue; |
888
|
|
|
} |
889
|
|
|
// create new array indexed by attribute name |
890
|
|
|
|
891
|
|
|
if (isset($attribute['device'])) { |
892
|
|
|
$collapsedList[$attribute['name']][$attribute['device']][] = $attribute['value']; |
893
|
|
|
} else { |
894
|
|
|
$collapsedList[$attribute['name']][] = $attribute['value']; |
895
|
|
|
} |
896
|
|
|
// and keep all language-variant names in a separate sub-array |
897
|
|
|
if ($attribute['flag'] == "ML") { |
898
|
|
|
$collapsedList[$attribute['name']]['langs'][$attribute['lang']] = $attribute['value']; |
899
|
|
|
} |
900
|
|
|
} |
901
|
|
|
// once we have the final list, populate the respective "best-match" |
902
|
|
|
// language to choose for the ML attributes |
903
|
|
|
|
904
|
|
|
foreach ($collapsedList as $attribName => $valueArray) { |
905
|
|
|
if (isset($valueArray['langs'])) { // we have at least one language-dependent name in this attribute |
906
|
|
|
// for printed names on screen: |
907
|
|
|
// assign to exact match language, fallback to "default" language, fallback to English, fallback to whatever comes first in the list |
908
|
|
|
$collapsedList[$attribName][0] = $valueArray['langs'][$this->languageInstance->getLang()] ?? $valueArray['langs']['C'] ?? $valueArray['langs']['en'] ?? array_shift($valueArray['langs']); |
909
|
|
|
// for names usable in filesystems (closer to good old ASCII...) |
910
|
|
|
// prefer English, otherwise the "default" language, otherwise the same that we got above |
911
|
|
|
$collapsedList[$attribName][1] = $valueArray['langs']['en'] ?? $valueArray['langs']['C'] ?? $collapsedList[$attribName][0]; |
912
|
|
|
} |
913
|
|
|
} |
914
|
|
|
return $collapsedList; |
915
|
|
|
} |
916
|
|
|
|
917
|
|
|
/** |
918
|
|
|
* Is the profile global redirection set? |
919
|
|
|
* |
920
|
|
|
* @return bool |
921
|
|
|
*/ |
922
|
|
|
public function isRedirected() { |
923
|
|
|
$result = $this->databaseHandle->exec("SELECT profile_id FROM profile_option WHERE profile_id = ? AND option_name='device-specific:redirect' AND device_id IS NULL", "i", $this->identifier); |
924
|
|
|
if ($result->num_rows == 0) { |
925
|
|
|
return false; |
926
|
|
|
} |
927
|
|
|
return true; |
928
|
|
|
} |
929
|
|
|
|
930
|
|
|
/** |
931
|
|
|
* Does the profile contain enough information to generate installers with |
932
|
|
|
* it? Silverbullet will always return TRUE; RADIUS profiles need to do some |
933
|
|
|
* heavy lifting here. |
934
|
|
|
* |
935
|
|
|
* @return int one of the constants above which tell if enough info is set to enable installers |
936
|
|
|
*/ |
937
|
|
|
public function readinessLevel() |
938
|
|
|
{ |
939
|
|
|
$result = $this->databaseHandle->exec("SELECT sufficient_config, showtime FROM profile WHERE profile_id = ?", "i", $this->identifier); |
940
|
|
|
// SELECT queries always return a resource, not a boolean |
941
|
|
|
$configQuery = mysqli_fetch_row(/** @scrutinizer ignore-type */ $result); |
942
|
|
|
if ($configQuery[0] == "0") { |
943
|
|
|
return self::READINESS_LEVEL_NOTREADY; |
944
|
|
|
} |
945
|
|
|
// at least fully configured, if not showtime! |
946
|
|
|
if ($configQuery[1] == "0") { |
947
|
|
|
return self::READINESS_LEVEL_SUFFICIENTCONFIG; |
948
|
|
|
} |
949
|
|
|
return self::READINESS_LEVEL_SHOWTIME; |
950
|
|
|
} |
951
|
|
|
|
952
|
|
|
/** |
953
|
|
|
* Checks all profile certificates validity periods comparing to the pre-defined |
954
|
|
|
* thresholds and returns the most critical status. |
955
|
|
|
* |
956
|
|
|
* @return int - one of constants defined in this profile |
957
|
|
|
*/ |
958
|
|
|
public function certificateStatus() |
959
|
|
|
{ |
960
|
|
|
$query = "SELECT option_value AS cert FROM profile_option WHERE option_name='eap:ca_file' AND profile_id = ?"; |
961
|
|
|
$result = $this->databaseHandle->exec($query, "i", $this->identifier); |
962
|
|
|
$rows = $result->fetch_all(); |
963
|
|
|
$x509 = new \core\common\X509(); |
964
|
|
|
$profileStatus = self::CERT_STATUS_NONE; |
965
|
|
|
foreach ($rows as $row) { |
966
|
|
|
$encodedCert = $row[0]; |
967
|
|
|
$tm = $x509->processCertificate(base64_decode($encodedCert))['full_details']['validTo_time_t']- time(); |
968
|
|
|
if ($tm < \config\ConfAssistant::CERT_WARNINGS['expiry_critical']) { |
969
|
|
|
$certStatus = self::CERT_STATUS_ERROR; |
970
|
|
|
} elseif ($tm < \config\ConfAssistant::CERT_WARNINGS['expiry_warning']) { |
971
|
|
|
$certStatus = self::CERT_STATUS_WARN; |
972
|
|
|
} else { |
973
|
|
|
$certStatus = self::CERT_STATUS_OK; |
974
|
|
|
} |
975
|
|
|
$profileStatus = max($profileStatus, $certStatus); |
976
|
|
|
} |
977
|
|
|
return $profileStatus; |
978
|
|
|
} |
979
|
|
|
|
980
|
|
|
/** |
981
|
|
|
* Checks if the profile has enough information to have something to show to end users. This does not necessarily mean |
982
|
|
|
* that there's a fully configured EAP type - it is sufficient if a redirect has been set for at least one device. |
983
|
|
|
* |
984
|
|
|
* @return boolean TRUE if enough information for showtime is set; FALSE if not |
985
|
|
|
*/ |
986
|
|
|
private function readyForShowtime() |
987
|
|
|
{ |
988
|
|
|
$properConfig = FALSE; |
989
|
|
|
$attribs = $this->getCollapsedAttributes(); |
990
|
|
|
// do we have enough to go live? Check if any of the configured EAP methods is completely configured ... |
991
|
|
|
if (sizeof($this->getEapMethodsinOrderOfPreference(1)) > 0) { |
992
|
|
|
$properConfig = TRUE; |
993
|
|
|
} |
994
|
|
|
// if not, it could still be that general redirect has been set |
995
|
|
|
if (!$properConfig) { |
996
|
|
|
if (isset($attribs['device-specific:redirect'])) { |
997
|
|
|
$properConfig = TRUE; |
998
|
|
|
} |
999
|
|
|
// just a per-device redirect? would be good enough... but this is not actually possible: |
1000
|
|
|
// per-device redirects can only be set on the "fine-tuning" page, which is only accessible |
1001
|
|
|
// if at least one EAP type is fully configured - which is caught above and makes readyForShowtime TRUE already |
1002
|
|
|
} |
1003
|
|
|
// do we know at least one SSID to configure, or work with wired? If not, it's not ready... |
1004
|
|
|
if (!isset($attribs['media:SSID']) && |
1005
|
|
|
(!isset(\config\ConfAssistant::CONSORTIUM['ssid']) || count(\config\ConfAssistant::CONSORTIUM['ssid']) == 0) && |
1006
|
|
|
!isset($attribs['media:wired'])) { |
1007
|
|
|
$properConfig = FALSE; |
1008
|
|
|
} |
1009
|
|
|
// institutions without a name are not really a corner case we should support |
1010
|
|
|
if (!isset($attribs['general:instname'])) { |
1011
|
|
|
$properConfig = FALSE; |
1012
|
|
|
} |
1013
|
|
|
return $properConfig; |
1014
|
|
|
} |
1015
|
|
|
|
1016
|
|
|
/** |
1017
|
|
|
* set the showtime property if prepShowTime says that there is enough info *and* the admin flagged the profile for showing |
1018
|
|
|
* |
1019
|
|
|
* @return void |
1020
|
|
|
*/ |
1021
|
|
|
public function prepShowtime() |
1022
|
|
|
{ |
1023
|
|
|
$properConfig = $this->readyForShowtime(); |
1024
|
|
|
$this->databaseHandle->exec("UPDATE profile SET sufficient_config = ".($properConfig ? "TRUE" : "FALSE")." WHERE profile_id = ".$this->identifier); |
1025
|
|
|
|
1026
|
|
|
$attribs = $this->getCollapsedAttributes(); |
1027
|
|
|
// if not enough info to go live, set FALSE |
1028
|
|
|
// even if enough info is there, admin has the ultimate say: |
1029
|
|
|
// if he doesn't want to go live, no further checks are needed, set FALSE as well |
1030
|
|
|
if (!$properConfig || !isset($attribs['profile:production']) || (isset($attribs['profile:production']) && $attribs['profile:production'][0] != "on")) { |
1031
|
|
|
$this->databaseHandle->exec("UPDATE profile SET showtime = FALSE WHERE profile_id = ?", "i", $this->identifier); |
1032
|
|
|
return; |
1033
|
|
|
} |
1034
|
|
|
$this->databaseHandle->exec("UPDATE profile SET showtime = TRUE WHERE profile_id = ?", "i", $this->identifier); |
1035
|
|
|
} |
1036
|
|
|
|
1037
|
|
|
/** |
1038
|
|
|
* internal helper - some attributes are added by the constructor "ex officio" |
1039
|
|
|
* without actual input from the admin. We can streamline their addition in |
1040
|
|
|
* this function to avoid duplication. |
1041
|
|
|
* |
1042
|
|
|
* @param array $internalAttributes - only names and value |
1043
|
|
|
* @return array full attributes with all properties set |
1044
|
|
|
*/ |
1045
|
|
|
protected function addInternalAttributes($internalAttributes) |
1046
|
|
|
{ |
1047
|
|
|
// internal attributes share many attribute properties, so condense the generation |
1048
|
|
|
$retArray = []; |
1049
|
|
|
foreach ($internalAttributes as $attName => $attValue) { |
1050
|
|
|
$retArray[] = ["name" => $attName, |
1051
|
|
|
"lang" => NULL, |
1052
|
|
|
"value" => $attValue, |
1053
|
|
|
"level" => Options::LEVEL_PROFILE, |
1054
|
|
|
"row_id" => 0, |
1055
|
|
|
"flag" => NULL, |
1056
|
|
|
]; |
1057
|
|
|
} |
1058
|
|
|
return $retArray; |
1059
|
|
|
} |
1060
|
|
|
|
1061
|
|
|
/** |
1062
|
|
|
* Retrieves profile attributes stored in the database |
1063
|
|
|
* |
1064
|
|
|
* @return array The attributes in one array |
1065
|
|
|
*/ |
1066
|
|
|
protected function addDatabaseAttributes() |
1067
|
|
|
{ |
1068
|
|
|
$databaseAttributes = $this->retrieveOptionsFromDatabase("SELECT DISTINCT option_name, option_lang, option_value, row_id |
1069
|
|
|
FROM $this->entityOptionTable |
1070
|
|
|
WHERE $this->entityIdColumn = ? |
1071
|
|
|
AND device_id IS NULL AND eap_method_id = 0 |
1072
|
|
|
ORDER BY option_name", "Profile"); |
1073
|
|
|
return $databaseAttributes; |
1074
|
|
|
} |
1075
|
|
|
} |
1076
|
|
|
|
The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g.
excluded_paths: ["lib/*"]
, you can move it to the dependency path list as follows:For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths