Merge branch 'MDL-73883-master' of https://github.com/andrewnicols/moodle
[moodle.git] / lib / licenselib.php
blob017ffc0337c13c49e3f3ca7ea866898b319aa68a
1 <?php
3 // This file is part of Moodle - http://moodle.org/
4 //
5 // Moodle is free software: you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation, either version 3 of the License, or
8 // (at your option) any later version.
9 //
10 // Moodle is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 // GNU General Public License for more details.
15 // You should have received a copy of the GNU General Public License
16 // along with Moodle. If not, see <http://www.gnu.org/licenses/>.
19 /**
20 * A namespace contains license specific functions
22 * @since Moodle 2.0
23 * @package core
24 * @subpackage lib
25 * @copyright 2010 Dongsheng Cai <dongsheng@moodle.com>
26 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
29 defined('MOODLE_INTERNAL') || die();
31 class license_manager {
33 /**
34 * License is a core license and can not be updated or deleted.
36 const CORE_LICENSE = 0;
38 /**
39 * License is a custom license and can be updated and/or deleted.
41 const CUSTOM_LICENSE = 1;
43 /**
44 * Integer representation of boolean for a license that is enabled.
46 const LICENSE_ENABLED = 1;
48 /**
49 * Integer representation of boolean for a license that is disabled.
51 const LICENSE_DISABLED = 0;
53 /**
54 * Integer for moving a license up order.
56 const LICENSE_MOVE_UP = -1;
58 /**
59 * Integer for moving a license down order.
61 const LICENSE_MOVE_DOWN = 1;
63 /**
64 * Save a license record.
66 * @param object $license {
67 * shortname => string a shortname of license, will be refered by files table[required]
68 * fullname => string the fullname of the license [required]
69 * source => string the homepage of the license type[required]
70 * enabled => int is it enabled?
71 * version => int a version number used by moodle [required]
72 * }
74 static public function save($license) {
76 $existinglicense = self::get_license_by_shortname($license->shortname);
78 if (!empty($existinglicense)) {
79 $id = $existinglicense->id;
80 if ($existinglicense->custom == self::CORE_LICENSE) {
81 // Can only update the enabled status and sortorder for core licenses.
82 $existinglicense->enabled = $license->enabled;
83 $existinglicense->sortorder = $license->sortorder;
84 $license = $existinglicense;
86 $license->id = $id;
87 self::update($license);
88 } else {
89 self::create($license);
92 return true;
95 /**
96 * Adding a new license type
98 * @deprecated Since Moodle 3.9, MDL-45184.
99 * @todo MDL-67344 This will be deleted in Moodle 4.1.
100 * @see license_manager::save()
102 * @param object $license the license record to add.
104 * @return bool true on success.
106 public function add($license) : bool {
107 debugging('add() is deprecated. Please use license_manager::save() instead.', DEBUG_DEVELOPER);
109 return self::save($license);
113 * Create a license record.
115 * @param object $license the license to create record for.
117 static protected function create($license) {
118 global $DB;
120 $licensecount = count(self::get_licenses());
121 $license->sortorder = $licensecount + 1;
122 // Enable all created license by default.
123 $license->enabled = self::LICENSE_ENABLED;
124 // API can only create custom licenses, core licenses
125 // are directly created at install or upgrade.
126 $license->custom = self::CUSTOM_LICENSE;
128 $DB->insert_record('license', $license);
129 self::reset_license_cache();
130 // Update the config setting of active licenses.
131 self::set_active_licenses();
135 * Read licens record(s) from database.
137 * @param array $params license parameters to return licenses for.
139 * @return array $filteredlicenses object[] of licenses.
141 static public function read(array $params = []) {
142 $licenses = self::get_licenses();
144 $filteredlicenses = [];
146 foreach ($licenses as $shortname => $license) {
147 $filtermatch = true;
148 foreach ($params as $key => $value) {
149 if ($license->$key != $value) {
150 $filtermatch = false;
153 if ($filtermatch) {
154 $filteredlicenses[$shortname] = $license;
157 return $filteredlicenses;
162 * Update a license record.
164 * @param object $license the license to update record for.
166 * @throws \moodle_exception if attempting to update a core license.
168 static protected function update($license) {
169 global $DB;
171 $DB->update_record('license', $license);
172 self::reset_license_cache();
176 * Delete a custom license.
178 * @param string $licenseshortname the shortname of license.
180 * @throws \moodle_exception when attempting to delete a license you are not allowed to.
182 static public function delete($licenseshortname) {
183 global $DB;
185 $licensetodelete = self::get_license_by_shortname($licenseshortname);
187 if (!empty($licensetodelete)) {
188 if ($licensetodelete->custom == self::CUSTOM_LICENSE) {
189 // Check that the license is not in use by any files, if so it cannot be deleted.
190 $countfilesusinglicense = $DB->count_records('files', ['license' => $licenseshortname]);
191 if ($countfilesusinglicense > 0) {
192 throw new moodle_exception('cannotdeletelicenseinuse', 'license');
194 $deletedsortorder = $licensetodelete->sortorder;
195 $DB->delete_records('license', ['id' => $licensetodelete->id]);
197 // We've deleted a license, so update our list of licenses so we don't save the deleted license again.
198 self::reset_license_cache();
199 $licenses = self::get_licenses();
201 foreach ($licenses as $license) {
202 if ($license->sortorder > $deletedsortorder) {
203 $license->sortorder = $license->sortorder - 1;
204 self::save($license);
208 // Update the config setting of active licenses as well.
209 self::set_active_licenses();
211 } else {
212 throw new moodle_exception('cannotdeletecore', 'license');
214 } else {
215 throw new moodle_exception('licensenotfoundshortname', 'license', $licenseshortname);
220 * Get license records.
222 * @return array|false object[] of license records of false if none.
224 static public function get_licenses() {
225 global $DB;
227 $cache = \cache::make('core', 'license');
228 $licenses = $cache->get('licenses');
230 if ($licenses === false) {
231 $licenses = [];
232 $records = $DB->get_records_select('license', null, null, 'sortorder ASC');
233 foreach ($records as $license) {
234 $licenses[$license->shortname] = $license;
236 $cache->set('licenses', $licenses);
239 foreach ($licenses as $license) {
240 // Localise the license names.
241 if ($license->custom == self::CORE_LICENSE) {
242 $license->fullname = get_string($license->shortname, 'core_license');
243 } else {
244 $license->fullname = format_string($license->fullname);
248 return $licenses;
252 * Change the sort order of a license (and it's sibling license as a result).
254 * @param int $direction value to change sortorder of license by.
255 * @param string $licenseshortname the shortname of license to changes sortorder for.
257 * @throws \moodle_exception if attempting to use invalid direction value.
259 static public function change_license_sortorder(int $direction, string $licenseshortname) : void {
261 if ($direction != self::LICENSE_MOVE_UP && $direction != self::LICENSE_MOVE_DOWN) {
262 throw new coding_exception(
263 'Must use a valid licence API move direction constant (LICENSE_MOVE_UP or LICENSE_MOVE_DOWN)');
266 $licenses = self::get_licenses();
267 $licensetoupdate = $licenses[$licenseshortname];
269 $currentsortorder = $licensetoupdate->sortorder;
270 $targetsortorder = $currentsortorder + $direction;
272 if ($targetsortorder > 0 && $targetsortorder <= count($licenses) ) {
273 foreach ($licenses as $license) {
274 if ($license->sortorder == $targetsortorder) {
275 $license->sortorder = $license->sortorder - $direction;
276 self::update($license);
279 $licensetoupdate->sortorder = $targetsortorder;
280 self::update($licensetoupdate);
285 * Get license record by shortname
287 * @param string $name the shortname of license
288 * @return object|null the license or null if no license found.
290 static public function get_license_by_shortname(string $name) {
291 $licenses = self::read(['shortname' => $name]);
293 if (!empty($licenses)) {
294 $license = reset($licenses);
295 } else {
296 $license = null;
299 return $license;
303 * Enable a license
304 * @param string $license the shortname of license
305 * @return boolean
307 static public function enable($license) {
308 if ($license = self::get_license_by_shortname($license)) {
309 $license->enabled = self::LICENSE_ENABLED;
310 self::update($license);
312 self::set_active_licenses();
314 return true;
318 * Disable a license
319 * @param string $license the shortname of license
320 * @return boolean
322 static public function disable($license) {
323 global $CFG;
324 // Site default license cannot be disabled!
325 if ($license == $CFG->sitedefaultlicense) {
326 print_error('error');
328 if ($license = self::get_license_by_shortname($license)) {
329 $license->enabled = self::LICENSE_DISABLED;
330 self::update($license);
332 self::set_active_licenses();
334 return true;
338 * Store active licenses in global config.
340 static protected function set_active_licenses() {
341 $licenses = self::read(['enabled' => self::LICENSE_ENABLED]);
342 $result = array();
343 foreach ($licenses as $l) {
344 $result[] = $l->shortname;
346 set_config('licenses', implode(',', $result));
350 * Get the globally configured active licenses.
352 * @return array of license objects.
353 * @throws \coding_exception
355 static public function get_active_licenses() {
356 global $CFG;
358 $result = [];
360 if (!empty($CFG->licenses)) {
361 $activelicenses = explode(',', $CFG->licenses);
362 $licenses = self::get_licenses();
363 foreach ($licenses as $license) {
364 if (in_array($license->shortname, $activelicenses)) {
365 $result[$license->shortname] = $license;
370 return $result;
374 * Get the globally configured active licenses as an array.
376 * @return array $licenses an associative array of licenses shaped as ['shortname' => 'fullname']
378 static public function get_active_licenses_as_array() {
379 $activelicenses = self::get_active_licenses();
381 $licenses = [];
382 foreach ($activelicenses as $license) {
383 $licenses[$license->shortname] = $license->fullname;
386 return $licenses;
390 * Install moodle built-in licenses.
392 static public function install_licenses() {
393 global $CFG;
395 require_once($CFG->libdir . '/db/upgradelib.php');
397 upgrade_core_licenses();
401 * Reset the license cache so it rebuilds next time licenses are fetched.
403 static public function reset_license_cache() {
404 $cache = \cache::make('core', 'license');
405 $cache->delete('licenses');