From 29c8dd9aa465266d80fba4c003dadeac852f458b Mon Sep 17 00:00:00 2001 From: PMKuipers Date: Fri, 2 Feb 2024 23:36:56 +0100 Subject: [PATCH] Fixed some issues with listing categories and other rights issues --- amd/build/page-edit-plan.min.js.map | 2 +- amd/src/page-edit-plan.js | 2 +- classes/courseservice.php | 10 ++++------ 3 files changed, 6 insertions(+), 8 deletions(-) diff --git a/amd/build/page-edit-plan.min.js.map b/amd/build/page-edit-plan.min.js.map index e1d9dfc..a1b7dcc 100644 --- a/amd/build/page-edit-plan.min.js.map +++ b/amd/build/page-edit-plan.min.js.map @@ -1 +1 @@ -{"version":3,"file":"page-edit-plan.min.js","sources":["../src/page-edit-plan.js"],"sourcesContent":["/*eslint no-var: \"error\" */\n/*eslint no-unused-vars: \"off\" */\n/*eslint linebreak-style: \"off\" */\n/*eslint no-trailing-spaces: \"off\" */\n/*eslint-env es6*/\n// Put this file in path/to/plugin/amd/src\n// You can call it anything you like\n\nimport {get_string,get_strings} from 'core/str';\nimport {call} from 'core/ajax';\nimport notification from 'core/notification';\n\nimport Vue from './vue/vue';\n\n\nimport EditorComponents from './studyplan-editor-components';\nVue.use(EditorComponents);\n\nimport TSComponents from './treestudyplan-components';\nVue.use(TSComponents);\n\nimport ModalComponents from './modedit-modal';\nVue.use(ModalComponents);\nimport Debugger from './util/debugger';\n\nimport {load_strings} from './util/string-helper';\nimport {ProcessStudyplan} from './studyplan-processor';\nimport {download,upload} from './downloader';\nimport {studyplanTiming} from './util/date-helper';\n\nimport PortalVue from './portal-vue/portal-vue.esm';\nVue.use(PortalVue);\nimport BootstrapVue from './bootstrap-vue/bootstrap-vue';\nVue.use(BootstrapVue);\n\nimport {Drag, Drop, DropList} from './vue-easy-dnd/vue-easy-dnd.esm';\nVue.component('drag',Drag);\nVue.component('drop',Drop);\nVue.component('drop-list',DropList);\n\nconst debug = new Debugger(\"treestudyplan\");\n\nlet strings = load_strings({\n studyplan: {\n studyplan_select_placeholder: 'studyplan_select_placeholder',\n },\n});\n\n/**\n * Initialize the Page\n * @param {int} contextid The context we should attempt to work in (1:1 related to the category)\n * @param {int} categoryid The category we shoud attempt to work in (1:1 related to the context)\n * @param {object} options Options to be passed to this script\n */\nexport function init(contextid,categoryid,options) {\n // Make sure the id's are numeric and integer\n if(undefined === contextid || !Number.isInteger(Number(contextid)) || contextid < 1 ){ contextid = 1;}\n else { contextid = Number(contextid);} // ensure a numeric value instead of string\n if(undefined === categoryid || !Number.isInteger(Number(categoryid))){ categoryid = 0;}\n else { categoryid = Number(categoryid);} // ensure a numeric value instead of string\n\n debug.info(\"options\",options);\n if ( options !== null && typeof options === 'object' && !Array.isArray(options) ) {\n if ( !options.defaultAggregation ) {\n options.defaultAggregation = \"core\";\n }\n if ( !options.editMode ) {\n options.editMode = false;\n }\n } else {\n options = { defaultAggregation: \"core\", editMode: false};\n }\n\n // Setup the initial Vue app for this page\n let app = new Vue({\n el: '#root',\n data: {\n create: {\n studyplan: {\n name: '',\n shortname: '',\n description: '',\n idnumber: '',\n slots : 4,\n startdate: '2020-08-01',\n enddate: '',\n context: contextid,\n aggregation: options.defaultAggregation,\n aggregation_config: '',\n }\n },\n toolbox: {\n shown: false,\n right: true,\n },\n filters: {\n systembadges: \"\",\n relatedbadges: \"\",\n },\n activestudyplan: null,\n activepage: null,\n loadingstudyplan: false,\n studyplans: [],\n frameworks: [],\n relatedbadges: [],\n systembadges: [],\n courses: [],\n text: strings.studyplan,\n usedcontexts: [],\n initialEditMode: !!options.editMode,\n },\n created() {\n this.$root.$on('studyplanRemoved',(studyplan)=>{\n\n if(app.activestudyplan == studyplan){\n app.activestudyplan = null;\n }\n\n // remove studyplan from index list\n let index = null;\n for(let idx in app.studyplans){\n if(app.studyplans[idx].id == studyplan.id){\n index = idx;\n break;\n }\n }\n if(index){\n app.studyplans.splice(index, 1);\n }\n\n });\n },\n mounted() {\n this.initialize();\n },\n computed: {\n dropdown_title(){\n if(this.activestudyplan && this.activestudyplan.name){\n return this.activestudyplan.name;\n }\n else{\n return this.text.studyplan_select_placeholder;\n }\n },\n contextid(){\n return contextid;\n },\n filterComponentType(){\n return {\n item: false,\n component: true,\n span: 1,\n type: 'filter',\n };\n },\n },\n methods: {\n initialize() {\n call([{\n methodname: 'local_treestudyplan_list_studyplans',\n args: { context_id: contextid}\n }])[0].then(function(response){\n const timingval = { future: 0, present: 1, past: 2, };\n response.sort((a,b) => {\n const timinga = studyplanTiming(a);\n const timingb = studyplanTiming(b);\n \n let t = timingval[timinga] - timingval[timingb];\n if(t == 0){\n // sort by start date if timing is equal\n t = new Date(b.startdate).getTime() - new Date(a.startdate).getTime();\n \n if (t == 0) {\n // sort by name if timing is equal\n t = a.name.localeCompare(b.name);\n }\n }\n return t;\n });\n app.studyplans = response;\n \n // load studyplan from hash if applicable\n const hash = location.hash.replace('#','');\n if(hash){\n const id = hash;\n app.selectStudyplan(id);\n }\n }).catch(notification.exception);\n call([{\n methodname: 'local_treestudyplan_map_categories',\n args: {}\n }])[0].then(function(response){\n app.courses = response;\n }).catch(notification.exception);\n call([{\n methodname: 'local_treestudyplan_list_used_categories',\n args: { operation: 'edit', refcontext_id: contextid}\n }])[0].then(function(response){\n app.usedcontexts = response;\n }).catch(notification.exception);\n this.filter_systembadges();\n },\n closeStudyplan() {\n app.activestudyplan = null;\n window.location.hash = '';\n },\n movedStudyplan(plan,from,to) {\n // reload the page in the new context (needed, since a number of links are not reactive in the page)\n const params = new URLSearchParams(location.search);\n params.delete('categoryid');\n params.set(\"contextid\", to);\n setTimeout(() => {\n // Reload page in a timeout to give other form javasccript the change to remove the beforeunload handler.\n window.location.search = params.toString();\n },50);\n },\n onStudyPlanCreated(newstudyplan){\n if (newstudyplan.context_id != contextid) {\n // Study plan has changed context id - reload page into new context id and show the plan\n const params = new URLSearchParams(location.search);\n params.delete('categoryid');\n params.set(\"contextid\", newstudyplan.context_id);\n setTimeout(() => {\n // Reload page in a timeout to give other form javasccript the change to remove the beforeunload handler.\n window.location = window.location.pathname + \"?\" + params.toString() + \"#\" + newstudyplan.id;\n },50);\n } else {\n app.studyplans.push(newstudyplan);\n app.selectStudyplan(newstudyplan);\n }\n },\n switchContext(ctxid){\n const params = new URLSearchParams(location.search);\n params.delete('categoryid');\n params.set('contextid', ctxid);\n setTimeout(() => {\n // Reload page in a timeout to give other form javasccript the change to remove the beforeunload handler.\n window.location.href = window.location.pathname + \"?\" + params.toString();\n },50);\n },\n selectStudyplan(studyplanid){\n // fetch studyplan\n app.loadingstudyplan = true;\n app.activestudyplan = null;\n call([{\n methodname: 'local_treestudyplan_get_studyplan_map',\n args: { id: studyplanid}\n }])[0].then(function(response){\n app.activestudyplan = ProcessStudyplan(response);\n debug.info('studyplan processed');\n app.loadingstudyplan = false;\n window.location.hash = app.activestudyplan.id;\n }).catch(function(error){\n notification.exception(error);\n app.loadingstudyplan = false;\n });\n },\n onPageChange(page) {\n this.activepage = page;\n this.filter_relatedbadges();\n },\n import_studyplan(){\n const self = this;\n upload((filename,content)=>{\n call([{\n methodname: 'local_treestudyplan_import_plan',\n args: {\n content: content,\n format: \"application/json\",\n context_id: contextid,\n },\n }])[0].then(function(response){\n if(response.success){\n self.initialize();\n } else {\n debug.error(\"Import failed: \",response.msg);\n }\n\n }).catch(notification.exception);\n }, \"application/json\");\n },\n export_plan(plan,format){\n let self = this;\n if(format == undefined || ![\"json\",\"csv\"].includes(format)){\n format = \"json\";\n }\n call([{\n methodname: 'local_treestudyplan_export_plan',\n args: {\n studyplan_id: plan.id,\n format: format\n },\n }])[0].then(function(response){\n\n download(plan.shortname+\".json\",response.content,response.format);\n }).catch(notification.exception);\n },\n toggletoolbox(event) {\n debug.info(event);\n this.toolbox.shown = event;\n },\n filter_systembadges() {\n const self = this;\n call([{\n methodname: 'local_treestudyplan_search_badges',\n args: { \n search: this.filters.systembadges || \"\"\n }\n }])[0].then(function(response){\n self.systembadges = response;\n }).catch(notification.exception);\n },\n filter_relatedbadges() {\n const self = this;\n if (this.activepage) {\n call([{\n methodname: 'local_treestudyplan_search_related_badges',\n args: { \n page_id: this.activepage.id,\n search: this.filters.relatedbadges || \"\"\n }\n }])[0].then(function(response){\n self.relatedbadges = response;\n }).catch(notification.exception);\n }\n },\n reset_systembadges() {\n this.filters.systembadges = \"\";\n this.filter_systembadges();\n },\n reset_relatedbadges() {\n this.filters.relatedbadges = \"\";\n this.filter_relatedbadges();\n },\n },\n });\n}\n\n"],"names":["contextid","categoryid","options","undefined","Number","isInteger","debug","info","Array","isArray","defaultAggregation","editMode","app","Vue","el","data","create","studyplan","name","shortname","description","idnumber","slots","startdate","enddate","context","aggregation","aggregation_config","toolbox","shown","right","filters","systembadges","relatedbadges","activestudyplan","activepage","loadingstudyplan","studyplans","frameworks","courses","text","strings","usedcontexts","initialEditMode","created","$root","$on","index","idx","id","splice","mounted","initialize","computed","dropdown_title","this","studyplan_select_placeholder","filterComponentType","item","component","span","type","methods","methodname","args","context_id","then","response","timingval","future","present","past","sort","a","b","timinga","timingb","t","Date","getTime","localeCompare","hash","location","replace","selectStudyplan","catch","notification","exception","operation","refcontext_id","filter_systembadges","closeStudyplan","window","movedStudyplan","plan","from","to","params","URLSearchParams","search","delete","set","setTimeout","toString","onStudyPlanCreated","newstudyplan","pathname","push","switchContext","ctxid","href","studyplanid","error","onPageChange","page","filter_relatedbadges","import_studyplan","self","filename","content","format","success","msg","export_plan","includes","studyplan_id","toggletoolbox","event","page_id","reset_systembadges","reset_relatedbadges","use","EditorComponents","TSComponents","ModalComponents","PortalVue","BootstrapVue","Drag","Drop","DropList","Debugger"],"mappings":"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"} \ No newline at end of file +{"version":3,"file":"page-edit-plan.min.js","sources":["../src/page-edit-plan.js"],"sourcesContent":["/*eslint no-var: \"error\" */\n/*eslint no-unused-vars: \"off\" */\n/*eslint linebreak-style: \"off\" */\n/*eslint no-trailing-spaces: \"off\" */\n/*eslint-env es6*/\n// Put this file in path/to/plugin/amd/src\n// You can call it anything you like\n\nimport {get_string,get_strings} from 'core/str';\nimport {call} from 'core/ajax';\nimport notification from 'core/notification';\n\nimport Vue from './vue/vue';\n\n\nimport EditorComponents from './studyplan-editor-components';\nVue.use(EditorComponents);\n\nimport TSComponents from './treestudyplan-components';\nVue.use(TSComponents);\n\nimport ModalComponents from './modedit-modal';\nVue.use(ModalComponents);\nimport Debugger from './util/debugger';\n\nimport {load_strings} from './util/string-helper';\nimport {ProcessStudyplan} from './studyplan-processor';\nimport {download,upload} from './downloader';\nimport {studyplanTiming} from './util/date-helper';\n\nimport PortalVue from './portal-vue/portal-vue.esm';\nVue.use(PortalVue);\nimport BootstrapVue from './bootstrap-vue/bootstrap-vue';\nVue.use(BootstrapVue);\n\nimport {Drag, Drop, DropList} from './vue-easy-dnd/vue-easy-dnd.esm';\nVue.component('drag',Drag);\nVue.component('drop',Drop);\nVue.component('drop-list',DropList);\n\nconst debug = new Debugger(\"treestudyplan\");\n\nlet strings = load_strings({\n studyplan: {\n studyplan_select_placeholder: 'studyplan_select_placeholder',\n },\n});\n\n/**\n * Initialize the Page\n * @param {int} contextid The context we should attempt to work in (1:1 related to the category)\n * @param {int} categoryid The category we shoud attempt to work in (1:1 related to the context)\n * @param {object} options Options to be passed to this script\n */\nexport function init(contextid,categoryid,options) {\n // Make sure the id's are numeric and integer\n if(undefined === contextid || !Number.isInteger(Number(contextid)) || contextid < 1 ){ contextid = 1;}\n else { contextid = Number(contextid);} // ensure a numeric value instead of string\n if(undefined === categoryid || !Number.isInteger(Number(categoryid))){ categoryid = 0;}\n else { categoryid = Number(categoryid);} // ensure a numeric value instead of string\n\n debug.info(\"options\",options);\n if ( options !== null && typeof options === 'object' && !Array.isArray(options) ) {\n if ( !options.defaultAggregation ) {\n options.defaultAggregation = \"core\";\n }\n if ( !options.editMode ) {\n options.editMode = false;\n }\n } else {\n options = { defaultAggregation: \"core\", editMode: false};\n }\n\n // Setup the initial Vue app for this page\n let app = new Vue({\n el: '#root',\n data: {\n create: {\n studyplan: {\n name: '',\n shortname: '',\n description: '',\n idnumber: '',\n slots : 4,\n startdate: '2020-08-01',\n enddate: '',\n context: contextid,\n aggregation: options.defaultAggregation,\n aggregation_config: '',\n }\n },\n toolbox: {\n shown: false,\n right: true,\n },\n filters: {\n systembadges: \"\",\n relatedbadges: \"\",\n },\n activestudyplan: null,\n activepage: null,\n loadingstudyplan: false,\n studyplans: [],\n frameworks: [],\n relatedbadges: [],\n systembadges: [],\n courses: [],\n text: strings.studyplan,\n usedcontexts: [],\n initialEditMode: !!options.editMode,\n },\n created() {\n this.$root.$on('studyplanRemoved',(studyplan)=>{\n\n if(app.activestudyplan == studyplan){\n app.activestudyplan = null;\n }\n\n // remove studyplan from index list\n let index = null;\n for(let idx in app.studyplans){\n if(app.studyplans[idx].id == studyplan.id){\n index = idx;\n break;\n }\n }\n if(index){\n app.studyplans.splice(index, 1);\n }\n\n });\n },\n mounted() {\n this.initialize();\n },\n computed: {\n dropdown_title(){\n if(this.activestudyplan && this.activestudyplan.name){\n return this.activestudyplan.name;\n }\n else{\n return this.text.studyplan_select_placeholder;\n }\n },\n contextid(){\n return contextid;\n },\n filterComponentType(){\n return {\n item: false,\n component: true,\n span: 1,\n type: 'filter',\n };\n },\n },\n methods: {\n initialize() {\n call([{\n methodname: 'local_treestudyplan_list_studyplans',\n args: { context_id: contextid}\n }])[0].then(function(response){\n const timingval = { future: 0, present: 1, past: 2, };\n response.sort((a,b) => {\n const timinga = studyplanTiming(a);\n const timingb = studyplanTiming(b);\n \n let t = timingval[timinga] - timingval[timingb];\n if(t == 0){\n // sort by start date if timing is equal\n t = new Date(b.startdate).getTime() - new Date(a.startdate).getTime();\n \n if (t == 0) {\n // sort by name if timing is equal\n t = a.name.localeCompare(b.name);\n }\n }\n return t;\n });\n app.studyplans = response;\n \n // load studyplan from hash if applicable\n const hash = location.hash.replace('#','');\n if(hash){\n const id = hash;\n app.selectStudyplan(id);\n }\n }).catch(notification.exception);\n call([{\n methodname: 'local_treestudyplan_map_categories',\n args: { }\n }])[0].then(function(response){\n app.courses = response;\n }).catch(notification.exception);\n call([{\n methodname: 'local_treestudyplan_list_used_categories',\n args: { operation: 'edit', refcontext_id: contextid}\n }])[0].then(function(response){\n app.usedcontexts = response;\n }).catch(notification.exception);\n this.filter_systembadges();\n },\n closeStudyplan() {\n app.activestudyplan = null;\n window.location.hash = '';\n },\n movedStudyplan(plan,from,to) {\n // reload the page in the new context (needed, since a number of links are not reactive in the page)\n const params = new URLSearchParams(location.search);\n params.delete('categoryid');\n params.set(\"contextid\", to);\n setTimeout(() => {\n // Reload page in a timeout to give other form javasccript the change to remove the beforeunload handler.\n window.location.search = params.toString();\n },50);\n },\n onStudyPlanCreated(newstudyplan){\n if (newstudyplan.context_id != contextid) {\n // Study plan has changed context id - reload page into new context id and show the plan\n const params = new URLSearchParams(location.search);\n params.delete('categoryid');\n params.set(\"contextid\", newstudyplan.context_id);\n setTimeout(() => {\n // Reload page in a timeout to give other form javasccript the change to remove the beforeunload handler.\n window.location = window.location.pathname + \"?\" + params.toString() + \"#\" + newstudyplan.id;\n },50);\n } else {\n app.studyplans.push(newstudyplan);\n app.selectStudyplan(newstudyplan);\n }\n },\n switchContext(ctxid){\n const params = new URLSearchParams(location.search);\n params.delete('categoryid');\n params.set('contextid', ctxid);\n setTimeout(() => {\n // Reload page in a timeout to give other form javasccript the change to remove the beforeunload handler.\n window.location.href = window.location.pathname + \"?\" + params.toString();\n },50);\n },\n selectStudyplan(studyplanid){\n // fetch studyplan\n app.loadingstudyplan = true;\n app.activestudyplan = null;\n call([{\n methodname: 'local_treestudyplan_get_studyplan_map',\n args: { id: studyplanid}\n }])[0].then(function(response){\n app.activestudyplan = ProcessStudyplan(response);\n debug.info('studyplan processed');\n app.loadingstudyplan = false;\n window.location.hash = app.activestudyplan.id;\n }).catch(function(error){\n notification.exception(error);\n app.loadingstudyplan = false;\n });\n },\n onPageChange(page) {\n this.activepage = page;\n this.filter_relatedbadges();\n },\n import_studyplan(){\n const self = this;\n upload((filename,content)=>{\n call([{\n methodname: 'local_treestudyplan_import_plan',\n args: {\n content: content,\n format: \"application/json\",\n context_id: contextid,\n },\n }])[0].then(function(response){\n if(response.success){\n self.initialize();\n } else {\n debug.error(\"Import failed: \",response.msg);\n }\n\n }).catch(notification.exception);\n }, \"application/json\");\n },\n export_plan(plan,format){\n let self = this;\n if(format == undefined || ![\"json\",\"csv\"].includes(format)){\n format = \"json\";\n }\n call([{\n methodname: 'local_treestudyplan_export_plan',\n args: {\n studyplan_id: plan.id,\n format: format\n },\n }])[0].then(function(response){\n\n download(plan.shortname+\".json\",response.content,response.format);\n }).catch(notification.exception);\n },\n toggletoolbox(event) {\n debug.info(event);\n this.toolbox.shown = event;\n },\n filter_systembadges() {\n const self = this;\n call([{\n methodname: 'local_treestudyplan_search_badges',\n args: { \n search: this.filters.systembadges || \"\"\n }\n }])[0].then(function(response){\n self.systembadges = response;\n }).catch(notification.exception);\n },\n filter_relatedbadges() {\n const self = this;\n if (this.activepage) {\n call([{\n methodname: 'local_treestudyplan_search_related_badges',\n args: { \n page_id: this.activepage.id,\n search: this.filters.relatedbadges || \"\"\n }\n }])[0].then(function(response){\n self.relatedbadges = response;\n }).catch(notification.exception);\n }\n },\n reset_systembadges() {\n this.filters.systembadges = \"\";\n this.filter_systembadges();\n },\n reset_relatedbadges() {\n this.filters.relatedbadges = \"\";\n this.filter_relatedbadges();\n },\n },\n });\n}\n\n"],"names":["contextid","categoryid","options","undefined","Number","isInteger","debug","info","Array","isArray","defaultAggregation","editMode","app","Vue","el","data","create","studyplan","name","shortname","description","idnumber","slots","startdate","enddate","context","aggregation","aggregation_config","toolbox","shown","right","filters","systembadges","relatedbadges","activestudyplan","activepage","loadingstudyplan","studyplans","frameworks","courses","text","strings","usedcontexts","initialEditMode","created","$root","$on","index","idx","id","splice","mounted","initialize","computed","dropdown_title","this","studyplan_select_placeholder","filterComponentType","item","component","span","type","methods","methodname","args","context_id","then","response","timingval","future","present","past","sort","a","b","timinga","timingb","t","Date","getTime","localeCompare","hash","location","replace","selectStudyplan","catch","notification","exception","operation","refcontext_id","filter_systembadges","closeStudyplan","window","movedStudyplan","plan","from","to","params","URLSearchParams","search","delete","set","setTimeout","toString","onStudyPlanCreated","newstudyplan","pathname","push","switchContext","ctxid","href","studyplanid","error","onPageChange","page","filter_relatedbadges","import_studyplan","self","filename","content","format","success","msg","export_plan","includes","studyplan_id","toggletoolbox","event","page_id","reset_systembadges","reset_relatedbadges","use","EditorComponents","TSComponents","ModalComponents","PortalVue","BootstrapVue","Drag","Drop","DropList","Debugger"],"mappings":"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"} \ No newline at end of file diff --git a/amd/src/page-edit-plan.js b/amd/src/page-edit-plan.js index b79f3d3..c532ca1 100644 --- a/amd/src/page-edit-plan.js +++ b/amd/src/page-edit-plan.js @@ -188,7 +188,7 @@ export function init(contextid,categoryid,options) { }).catch(notification.exception); call([{ methodname: 'local_treestudyplan_map_categories', - args: {} + args: { } }])[0].then(function(response){ app.courses = response; }).catch(notification.exception); diff --git a/classes/courseservice.php b/classes/courseservice.php index bc9e740..8de10fe 100644 --- a/classes/courseservice.php +++ b/classes/courseservice.php @@ -95,10 +95,7 @@ class courseservice extends \external_api { public static function map_categories($rootid = 0) { global $CFG, $DB; - $root = \core_course_category::get($rootid); - $context = $root->get_context(); - // Make sure the user has access to the context for editing purposes. - webservicehelper::require_capabilities(self::CAP_EDIT, $context); + $root = \core_course_category::get($rootid,\MUST_EXIST,true); // Determine top categories from provided context. @@ -115,6 +112,7 @@ class courseservice extends \external_api { $children = [$root]; } + $list = []; foreach ($children as $cat) { $list[] = static::map_category($cat, false); } @@ -357,7 +355,7 @@ class courseservice extends \external_api { if ($ctx->contextlevel == CONTEXT_SYSTEM) { $cat = \core_course_category::top(); } else if ($ctx->contextlevel == CONTEXT_COURSECAT) { - $cat = \core_course_category::get($ctx->instanceid); + $cat = \core_course_category::get($ctx->instanceid,\MUST_EXIST,true); } $cats[] = $cat; if ($operation == "edit" && $ctxid == $refctxid) { @@ -458,7 +456,7 @@ class courseservice extends \external_api { if ($ctx->contextlevel == CONTEXT_SYSTEM) { $cat = \core_course_category::top(); } else if ($ctx->contextlevel == CONTEXT_COURSECAT) { - $cat = \core_course_category::get($ctx->instanceid); + $cat = \core_course_category::get($ctx->instanceid,\MUST_EXIST,true); } $cats[] = $cat; if ($operation == "view" && $ctxid == $refctxid) {