1. "use strict";
  2. /**
  3. * Functions for interacting with assignments within courses
  4. * @namespace api.course.assignment
  5. */
  6. var __extends = (this && this.__extends) || (function () {
  7. var extendStatics = function (d, b) {
  8. extendStatics = Object.setPrototypeOf ||
  9. ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
  10. function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };
  11. return extendStatics(d, b);
  12. };
  13. return function (d, b) {
  14. if (typeof b !== "function" && b !== null)
  15. throw new TypeError("Class extends value " + String(b) + " is not a constructor or null");
  16. extendStatics(d, b);
  17. function __() { this.constructor = d; }
  18. d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
  19. };
  20. })();
  21. var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
  22. function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
  23. return new (P || (P = Promise))(function (resolve, reject) {
  24. function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
  25. function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
  26. function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
  27. step((generator = generator.apply(thisArg, _arguments || [])).next());
  28. });
  29. };
  30. var __generator = (this && this.__generator) || function (thisArg, body) {
  31. var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
  32. return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
  33. function verb(n) { return function (v) { return step([n, v]); }; }
  34. function step(op) {
  35. if (f) throw new TypeError("Generator is already executing.");
  36. while (g && (g = 0, op[0] && (_ = 0)), _) try {
  37. if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
  38. if (y = 0, t) op = [op[0] & 2, t.value];
  39. switch (op[0]) {
  40. case 0: case 1: t = op; break;
  41. case 4: _.label++; return { value: op[1], done: false };
  42. case 5: _.label++; y = op[1]; op = [0]; continue;
  43. case 7: op = _.ops.pop(); _.trys.pop(); continue;
  44. default:
  45. if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
  46. if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
  47. if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
  48. if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
  49. if (t[2]) _.ops.pop();
  50. _.trys.pop(); continue;
  51. }
  52. op = body.call(thisArg, _);
  53. } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
  54. if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
  55. }
  56. };
  57. var __importDefault = (this && this.__importDefault) || function (mod) {
  58. return (mod && mod.__esModule) ? mod : { "default": mod };
  59. };
  60. Object.defineProperty(exports, "__esModule", { value: true });
  61. // Import shared classes
  62. var EndpointCategory_1 = __importDefault(require("../../shared/EndpointCategory"));
  63. var caccl_error_1 = __importDefault(require("caccl-error"));
  64. var ErrorCode_1 = __importDefault(require("../../shared/types/ErrorCode"));
  65. // Import shared helpers
  66. var utils_1 = __importDefault(require("../../shared/helpers/utils"));
  67. var waitForCompletion_1 = __importDefault(require("../../shared/helpers/waitForCompletion"));
  68. var parallelLimit_1 = __importDefault(require("../../shared/helpers/parallelLimit"));
  69. // Import shared constants
  70. var API_PREFIX_1 = __importDefault(require("../../shared/constants/API_PREFIX"));
  71. // Endpoint category
  72. var ECatAssignment = /** @class */ (function (_super) {
  73. __extends(ECatAssignment, _super);
  74. function ECatAssignment() {
  75. return _super !== null && _super.apply(this, arguments) || this;
  76. }
  77. /*------------------------------------------------------------------------*/
  78. /* Table of Contents: */
  79. /* - Assignments */
  80. /* - Grading */
  81. /* - Overrides */
  82. /* - Submissions */
  83. /*------------------------------------------------------------------------*/
  84. /*------------------------------------------------------------------------*/
  85. /* Assignment Endpoints */
  86. /*------------------------------------------------------------------------*/
  87. /**
  88. * Lists the assignments in a course
  89. * @author Gabe Abrams
  90. * @method list
  91. * @memberof api.course.assignment
  92. * @instance
  93. * @async
  94. * @param {object} [opts] object containing all arguments
  95. * @param {number} [opts.courseId=default course id] Canvas course Id to
  96. * query
  97. * @param {boolean} [opts.ignoreOverridesForDates] if true, assignment
  98. * dates are taken from the default dates instead of from the ones in
  99. * overrides
  100. * @param {APIConfig} [config] custom configuration for this specific endpoint
  101. * call (overwrites defaults that were included when api was initialized)
  102. * @returns {Promise<CanvasAssignment[]>} list of Canvas Assignments {@link https://canvas.instructure.com/doc/api/assignments.html#Assignment}
  103. */
  104. ECatAssignment.prototype.list = function (opts, config) {
  105. var _a;
  106. if (opts === void 0) { opts = {}; }
  107. return __awaiter(this, void 0, void 0, function () {
  108. return __generator(this, function (_b) {
  109. return [2 /*return*/, this.visitEndpoint({
  110. config: config,
  111. action: 'get the list of assignments in a course',
  112. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments"),
  113. method: 'GET',
  114. params: {
  115. override_assignment_dates: !opts.ignoreOverridesForDates,
  116. },
  117. })];
  118. });
  119. });
  120. };
  121. /**
  122. * Get info on a specific assignment in a course
  123. * @author Gabe Abrams
  124. * @method get
  125. * @memberof api.course.assignment
  126. * @instance
  127. * @async
  128. * @param {object} opts object containing all arguments
  129. * @param {number} opts.assignmentId Canvas assignment Id
  130. * @param {number} [opts.courseId=default course id] Canvas course Id to query
  131. * @param {boolean} [opts.ignoreOverridesForDates] if true, assignment
  132. * dates are taken from the default dates instead of from the ones in
  133. * overrides
  134. * @param {APIConfig} [config] custom configuration for this specific endpoint
  135. * call (overwrites defaults that were included when api was initialized)
  136. * @returns {Promise<CanvasAssignment>} Canvas Assignment {@link https://canvas.instructure.com/doc/api/assignments.html#Assignment}
  137. */
  138. ECatAssignment.prototype.get = function (opts, config) {
  139. var _a;
  140. return __awaiter(this, void 0, void 0, function () {
  141. return __generator(this, function (_b) {
  142. return [2 /*return*/, this.visitEndpoint({
  143. config: config,
  144. action: 'get info on a specific assignment in a course',
  145. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId),
  146. method: 'GET',
  147. params: {
  148. override_assignment_dates: !opts.ignoreOverridesForDates,
  149. },
  150. })];
  151. });
  152. });
  153. };
  154. /**
  155. * Updates a Canvas assignment
  156. * @author Gabe Abrams
  157. * @method update
  158. * @memberof api.course.assignment
  159. * @instance
  160. * @async
  161. * @param {object} opts - object containing all arguments
  162. * @param {number} opts.assignmentId Canvas assignment Id to update
  163. * @param {number} [opts.courseId=default course id] Canvas course Id to query
  164. * @param {string} [opts.name=current value] The name of the assignment
  165. * @param {number} [opts.pointsPossible=current value] Points possible
  166. * @param {date} [opts.dueAt=current value] Due at datetime
  167. * @param {date} [opts.lockAt=current value] Due at datetime
  168. * @param {date} [opts.unlockAt=current value] Due at datetime
  169. * @param {string} [opts.description=current value] html description of
  170. * the assignment
  171. * @param {string[]} [opts.submissionTypes=current value] Submission type(s)
  172. * @param {string} [opts.allowedExtensions=current value] List of allowed
  173. * file extensions (exclude period). Online upload must be enabled
  174. * @param {string} [opts.gradingType=current value] Grading type
  175. * @param {number} [opts.position=current value] Position in assignment
  176. * list
  177. * @param {boolean} [opts.published=current value] If true, publish page
  178. * upon creation. Must be a boolean
  179. * @param {boolean} [opts.muted=current value] If true, assignment is
  180. * muted. Must be a boolean
  181. * @param {number} [opts.groupSetId=current value] Student group set Id
  182. * @param {number} [opts.assignmentGroupId=current value] Assignment group
  183. * Id
  184. * @param {boolean} [opts.peerReviewsEnabled=current value] If true, users
  185. * asked to submit peer reviews. Must be a boolean
  186. * @param {boolean} [opts.automaticPeerReviewsEnabled=current value] If
  187. * true, Canvas will automatically assign peer reviews. Must be a boolean
  188. * @param {boolean} [opts.omitFromFinalGrade=current value] If true,
  189. * assignment is omitted from the final grade. Must be a boolean
  190. * @param {boolean} [opts.gradeGroupStudentsIndividually=current value] If
  191. * true, students in groups can be given separate grades and when one student
  192. * in a group gets a grade, other students do not get graded. Must be a
  193. * boolean
  194. * @param {APIConfig} [config] custom configuration for this specific endpoint
  195. * call (overwrites defaults that were included when api was initialized)
  196. * @returns {Promise<CanvasAssignment>} Canvas Assignment {@link https://canvas.instructure.com/doc/api/assignments.html#Assignment}
  197. */
  198. ECatAssignment.prototype.update = function (opts, config) {
  199. var _a;
  200. return __awaiter(this, void 0, void 0, function () {
  201. return __generator(this, function (_b) {
  202. return [2 /*return*/, this.visitEndpoint({
  203. config: config,
  204. action: 'update an assignment in a course',
  205. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId),
  206. method: 'PUT',
  207. params: {
  208. 'assignment[name]': utils_1.default.includeIfTruthy(opts.name),
  209. 'assignment[submission_types]': utils_1.default.includeIfTruthy(opts.submissionTypes),
  210. 'assignment[grading_type]': utils_1.default.includeIfTruthy(opts.gradingType),
  211. position: utils_1.default.includeIfTruthy(opts.position),
  212. 'assignment[peer_reviews]': utils_1.default.includeIfBoolean(opts.peerReviewsEnabled),
  213. 'assignment[automatic_peer_reviews]': utils_1.default.includeIfBoolean(opts.automaticPeerReviewsEnabled),
  214. 'assignment[grade_group_students_individually]': utils_1.default.includeIfBoolean(opts.gradeGroupStudentsIndividually),
  215. 'assignment[description]': utils_1.default.includeIfTruthy(opts.description),
  216. 'assignment[allowed_extensions]': utils_1.default.includeIfTruthy(opts.allowedExtensions),
  217. 'assignment[group_category_id]': utils_1.default.includeIfTruthy(opts.groupSetId),
  218. 'assignment[points_possible]': utils_1.default.includeIfNumber(opts.pointsPossible),
  219. 'assignment[due_at]': utils_1.default.includeIfDate(opts.dueAt),
  220. 'assignment[lock_at]': utils_1.default.includeIfDate(opts.lockAt),
  221. 'assignment[unlock_at]': utils_1.default.includeIfDate(opts.unlockAt),
  222. 'assignment[published]': utils_1.default.includeIfBoolean(opts.published),
  223. 'assignment[assignment_group_id]': utils_1.default.includeIfNumber(opts.assignmentGroupId),
  224. 'assignment[omit_from_final_grade]': utils_1.default.includeIfBoolean(opts.omitFromFinalGrade),
  225. 'assignment[muted]': utils_1.default.includeIfBoolean(opts.muted),
  226. },
  227. })];
  228. });
  229. });
  230. };
  231. /**
  232. * Creates a Canvas assignment
  233. * @author Gabe Abrams
  234. * @method create
  235. * @memberof api.course.assignment
  236. * @instance
  237. * @async
  238. * @param {object} [opts] object containing all arguments
  239. * @param {number} [opts.courseId=default course id] Canvas course Id to
  240. * create an assignment in
  241. * @param {string} [opts.name=Unnamed Assignment] The name of the
  242. * assignment
  243. * @param {number} [opts.pointsPossible=null] Points possible
  244. * @param {date} [opts.dueAt=null] Due at datetime
  245. * @param {date} [opts.lockAt=null] Due at datetime
  246. * @param {date} [opts.unlockAt=null] Due at datetime
  247. * @param {string} [opts.description=null] html description of
  248. * the assignment
  249. * @param {string} [opts.submissionTypes=null] Submission type(s)
  250. * @param {string} [opts.allowedExtensions=any] List of allowed file
  251. * extensions (exclude period). Online upload must be enabled
  252. * @param {string} [opts.gradingType=points] Grading type
  253. * @param {number} [opts.position=last] Position in assignment list
  254. * @param {boolean} [opts.published] If true, publish page upon
  255. * creation
  256. * @param {boolean} [opts.muted] If true, assignment is muted
  257. * @param {number} [opts.groupSetId=null] Student group set Id
  258. * @param {number} [opts.assignmentGroupId=top assignment group] Assignment
  259. * group Id
  260. * @param {boolean} [opts.peerReviewsEnabled] If true, users asked to
  261. * submit peer reviews
  262. * @param {boolean} [opts.automaticPeerReviewsEnabled] If true,
  263. * Canvas will automatically assign peer reviews
  264. * @param {boolean} [opts.omitFromFinalGrade] If true, assignment is
  265. * omitted from the final grade
  266. * @param {boolean} [opts.gradeGroupStudentsIndividually] If true,
  267. * students in groups can be given separate grades and when one student in a
  268. * group gets a grade, other students do not get graded
  269. * @param {string} [opts.assignmentAppId=null] If defined, the external
  270. * tool that matches this id will be used for submissions. Also, the
  271. * submission types will be overwritten with ['external_tool'] and the
  272. * student will be redirected via LTI to the assignmentAppURL when they
  273. * launch the assignment
  274. * @param {string} [opts.assignmentAppURL=tool launch url] The launch URL
  275. * of the external tool. If not included and assignmentAppId is defined, we
  276. * will first request info on the external tool to get its launchURL and
  277. * will use that value here. Only relevant if assignmentAppId is defined.
  278. * @param {boolean} [opts.assignmentAppNewTab] Only relevant if
  279. * assignmentAppId is defined. If true, when a student clicks the
  280. * assignment, their LTI session with the external tool will be opened in a
  281. * new tab
  282. * @param {APIConfig} [config] custom configuration for this specific endpoint
  283. * call (overwrites defaults that were included when api was initialized)
  284. * @returns {Promise<CanvasAssignment>} Canvas Assignment {@link https://canvas.instructure.com/doc/api/assignments.html#Assignment}
  285. */
  286. ECatAssignment.prototype.create = function (opts, config) {
  287. var _a, _b;
  288. if (opts === void 0) { opts = {}; }
  289. return __awaiter(this, void 0, void 0, function () {
  290. var params, app;
  291. return __generator(this, function (_c) {
  292. switch (_c.label) {
  293. case 0:
  294. params = {
  295. 'assignment[name]': (opts.name || 'Unnamed Assignment'),
  296. 'assignment[grading_type]': (opts.gradingType || 'points'),
  297. position: utils_1.default.includeIfTruthy(opts.position),
  298. 'assignment[peer_reviews]': (utils_1.default.isTruthy(opts.peerReviewsEnabled)),
  299. 'assignment[automatic_peer_reviews]': utils_1.default.isTruthy(opts.automaticPeerReviewsEnabled),
  300. 'assignment[grade_group_students_individually]': utils_1.default.isTruthy(opts.gradeGroupStudentsIndividually),
  301. 'assignment[description]': (utils_1.default.includeIfTruthy(opts.description)),
  302. 'assignment[allowed_extensions]': (utils_1.default.includeIfTruthy(opts.allowedExtensions)),
  303. 'assignment[group_category_id]': (utils_1.default.includeIfTruthy(opts.groupSetId)),
  304. 'assignment[points_possible]': (utils_1.default.includeIfNumber(opts.pointsPossible)),
  305. 'assignment[due_at]': utils_1.default.includeIfDate(opts.dueAt),
  306. 'assignment[lock_at]': utils_1.default.includeIfDate(opts.lockAt),
  307. 'assignment[unlock_at]': utils_1.default.includeIfDate(opts.unlockAt),
  308. 'assignment[published]': (utils_1.default.isTruthy(opts.published)),
  309. 'assignment[assignment_group_id]': (utils_1.default.includeIfNumber(opts.assignmentGroupId)),
  310. 'assignment[omit_from_final_grade]': (utils_1.default.isTruthy(opts.omitFromFinalGrade)),
  311. 'assignment[muted]': utils_1.default.isTruthy(opts.muted),
  312. };
  313. if (!opts.assignmentAppId) return [3 /*break*/, 4];
  314. // Using an external tool
  315. params['assignment[external_tool_tag_attributes][new_tab]'] = (!!opts.assignmentAppNewTab);
  316. params['assignment[external_tool_tag_attributes][content_type]'] = ('context_external_tool');
  317. params['assignment[external_tool_tag_attributes][content_id]'] = (opts.assignmentAppId);
  318. params['assignment[submission_types]'] = ['external_tool'];
  319. if (!opts.assignmentAppURL) return [3 /*break*/, 1];
  320. // No need to fetch the launchURL
  321. params['assignment[external_tool_tag_attributes][url]'] = (opts.assignmentAppURL);
  322. return [3 /*break*/, 3];
  323. case 1: return [4 /*yield*/, this.api.course.app.get({
  324. courseId: ((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId),
  325. appId: opts.assignmentAppId,
  326. }, config)];
  327. case 2:
  328. app = _c.sent();
  329. params['assignment[external_tool_tag_attributes][url]'] = app.url;
  330. _c.label = 3;
  331. case 3: return [3 /*break*/, 5];
  332. case 4:
  333. params['assignment[submission_types]'] = (opts.submissionTypes || ['none']);
  334. _c.label = 5;
  335. case 5: return [2 /*return*/, this.visitEndpoint({
  336. config: config,
  337. action: 'create a new assignment in a course',
  338. params: params,
  339. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_b = opts.courseId) !== null && _b !== void 0 ? _b : this.defaultCourseId, "/assignments"),
  340. method: 'POST',
  341. })];
  342. }
  343. });
  344. });
  345. };
  346. /**
  347. * Delete an assignment
  348. * @author Gabe Abrams
  349. * @method delete
  350. * @memberof api.course.assignment
  351. * @instance
  352. * @async
  353. * @param {object} opts object containing all arguments
  354. * @param {number} opts.assignmentId Canvas assignment Id
  355. * @param {number} [opts.courseId=default course id] Canvas course Id
  356. * @param {APIConfig} [config] custom configuration for this specific endpoint
  357. * call (overwrites defaults that were included when api was initialized)
  358. * @returns {Promise<CanvasAssignment>} Canvas Assignment {@link https://canvas.instructure.com/doc/api/assignments.html#Assignment}
  359. */
  360. ECatAssignment.prototype.delete = function (opts, config) {
  361. var _a;
  362. return __awaiter(this, void 0, void 0, function () {
  363. return __generator(this, function (_b) {
  364. return [2 /*return*/, this.visitEndpoint({
  365. config: config,
  366. action: 'delete an assignment from a course',
  367. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId),
  368. method: 'DELETE',
  369. })];
  370. });
  371. });
  372. };
  373. /*------------------------------------------------------------------------*/
  374. /* Grading Endpoints */
  375. /*------------------------------------------------------------------------*/
  376. /**
  377. * List gradeable students for a specific assignment
  378. * @author Gabe Abrams
  379. * @method listGradeableStudents
  380. * @memberof api.course.assignment
  381. * @instance
  382. * @async
  383. * @param {object} opts object containing all arguments
  384. * @param {number} opts.assignmentId Canvas assignment Id to query
  385. * @param {number} [opts.courseId=default course id] Canvas course Id to
  386. * query
  387. * @param {APIConfig} [config] custom configuration for this specific endpoint
  388. * call (overwrites defaults that were included when api was initialized)
  389. * @returns {Promise<CanvasUser[]>} list of Canvas users {@link https://canvas.instructure.com/doc/api/users.html#User}
  390. */
  391. ECatAssignment.prototype.listGradeableStudents = function (opts, config) {
  392. var _a;
  393. return __awaiter(this, void 0, void 0, function () {
  394. var students;
  395. return __generator(this, function (_b) {
  396. switch (_b.label) {
  397. case 0: return [4 /*yield*/, this.visitEndpoint({
  398. config: config,
  399. action: 'get the list of students who are gradeable in a specific assignment in a course',
  400. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/gradeable_students"),
  401. method: 'GET',
  402. })];
  403. case 1:
  404. students = _b.sent();
  405. return [2 /*return*/, students.filter(function (s) {
  406. return !s.fake_student;
  407. })];
  408. }
  409. });
  410. });
  411. };
  412. /**
  413. * Adds a comment to a submission
  414. * @author Gabe Abrams
  415. * @method createSubmissionComment
  416. * @memberof api.course.assignment
  417. * @instance
  418. * @async
  419. * @param {object} opts object containing all arguments
  420. * @param {number} opts.assignmentId Canvas course Id
  421. * @param {number} opts.studentId Canvas student Id of the sub to comment
  422. * on
  423. * @param {string} opts.comment The text of the comment
  424. * @param {number} [opts.courseId=default course id] Canvas course Id
  425. * @param {APIConfig} [config] custom configuration for this specific endpoint
  426. * call (overwrites defaults that were included when api was initialized)
  427. * @returns {Promise<CanvasSubmission>} Canvas submission {@link https://canvas.instructure.com/doc/api/submissions.html#Submission}
  428. */
  429. ECatAssignment.prototype.createSubmissionComment = function (opts, config) {
  430. var _a;
  431. return __awaiter(this, void 0, void 0, function () {
  432. return __generator(this, function (_b) {
  433. return [2 /*return*/, this.visitEndpoint({
  434. config: config,
  435. action: 'create a new comment on a submission',
  436. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/submissions/").concat(opts.studentId),
  437. method: 'PUT',
  438. params: {
  439. 'comment[text_comment]': opts.comment,
  440. },
  441. })];
  442. });
  443. });
  444. };
  445. /**
  446. * Updates a student's grade and/or comment
  447. * @author Gabe Abrams
  448. * @method updateGrade
  449. * @memberof api.course.assignment
  450. * @instance
  451. * @async
  452. * @param {object} opts object containing all arguments
  453. * @param {number} opts.assignmentId Canvas assignment id
  454. * @param {number} opts.studentId Canvas student id
  455. * @param {number} [opts.courseId=default course id] Canvas course id
  456. * @param {number} [opts.points] the overall points to assign to the
  457. * student
  458. * @param {string} [opts.comment] the grader comment to leave on the
  459. * submission
  460. * @param {APIConfig} [config] custom configuration for this specific endpoint
  461. * call (overwrites defaults that were included when api was initialized)
  462. * @returns {Promise<CanvasSubmission>} Canvas submission {@link https://canvas.instructure.com/doc/api/submissions.html#Submission}
  463. */
  464. ECatAssignment.prototype.updateGrade = function (opts, config) {
  465. var _a;
  466. return __awaiter(this, void 0, void 0, function () {
  467. return __generator(this, function (_b) {
  468. return [2 /*return*/, this.visitEndpoint({
  469. config: config,
  470. action: 'update student grade and/or comments for a specific assignment in a course',
  471. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/submissions/").concat(opts.studentId),
  472. method: 'PUT',
  473. params: {
  474. 'comment[text_comment]': utils_1.default.includeIfTruthy(opts.comment),
  475. 'submission[posted_grade]': utils_1.default.includeIfNumber(opts.points),
  476. },
  477. })];
  478. });
  479. });
  480. };
  481. /**
  482. * Batch updates grades and/or comments. Also supports updating rubric items
  483. * @author Gabe Abrams
  484. * @method updateGrades
  485. * @memberof api.course.assignment
  486. * @instance
  487. * @async
  488. * @param {object} opts object containing all arguments
  489. * @param {number} opts.assignmentId Canvas assignment Id
  490. * @param {Array} opts.gradeItems List of grade items to upload to Canvas:
  491. * [{
  492. * studentId: <student id>,
  493. * points: <optional, points to overwrite with>,
  494. * comment: <optional, comment to append (or overwrite if rubric comment)>,
  495. * rubricId: <optional, rubric item (overall grade/comment if excluded)>
  496. * },...]
  497. * @param {number} [opts.courseId=default course id] Canvas course Id
  498. * @param {boolean} [opts.waitForCompletion] If true, promise won't
  499. * resolve until Canvas has finished updating the grades, instead of resolving
  500. * once the grade changes have been queued
  501. * @param {number} [opts.waitForCompletionTimeout=2] The number of minutes
  502. * to wait before timing out the grade update job
  503. * @param {boolean} [opts.dontMergeRubricItemUpdates] When uploading
  504. * grades to a rubric item, we intelligently merge rubric item updates with
  505. * previous rubric assessments. For instance, if the assignment's rubric is:
  506. * { grammar, argument, formatting }
  507. * And the student of interest has the following rubric assessment so far:
  508. * { grammar: 10/10, argument: 8/10, formatting: ungraded }
  509. * When we upload a new gradeItem (9/10 points) to the student's
  510. * formatting rubric item, the result is:
  511. * { grammar: 10/10, argument: 8/10, formatting: 9/10 }
  512. * However, if dontMergeRubricItemUpdates=true, the result is:
  513. * { grammar: ungraded, argument: ungraded, formatting: 9/10 }
  514. * Note: merging is an added feature. By default, the Canvas API does not
  515. * merge rubric assessments.
  516. * @param {APIConfig} [config] custom configuration for this specific endpoint
  517. * call (overwrites defaults that were included when api was initialized)
  518. * @returns {Promise<CanvasProgress>} Canvas Progress object {@link https://canvas.instructure.com/doc/api/progress.html#Progress}
  519. */
  520. ECatAssignment.prototype.updateGrades = function (opts, config) {
  521. var _a, _b;
  522. return __awaiter(this, void 0, void 0, function () {
  523. var studentsToMerge, performRubricItemMerge, assignment, realRubricItemIds_1, numRubricItems_1, studentToRubricItemsOverwritten_1, allStudentsWithRubricItems_1, subs, params, overwritingMap_1, progress, finishedProgress;
  524. var _this = this;
  525. return __generator(this, function (_c) {
  526. switch (_c.label) {
  527. case 0:
  528. studentsToMerge = [];
  529. performRubricItemMerge = false;
  530. if (!opts.dontMergeRubricItemUpdates) {
  531. performRubricItemMerge = opts.gradeItems.some(function (item) {
  532. return item.rubricId;
  533. });
  534. }
  535. if (!performRubricItemMerge) return [3 /*break*/, 2];
  536. return [4 /*yield*/, this.api.course.assignment.get({
  537. courseId: ((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId),
  538. assignmentId: opts.assignmentId,
  539. }, config)];
  540. case 1:
  541. assignment = _c.sent();
  542. // Make sure the assignment has a rubric
  543. if (!assignment.rubric) {
  544. // This assignment doesn't have a rubric
  545. throw new caccl_error_1.default({
  546. message: 'We could not upload grades because the rubric we were trying to upload to didn\'t exist.',
  547. code: ErrorCode_1.default.NoRubricOnBatchGradeUpload,
  548. });
  549. }
  550. realRubricItemIds_1 = new Set();
  551. numRubricItems_1 = assignment.rubric.length;
  552. assignment.rubric.forEach(function (rubricItem) {
  553. realRubricItemIds_1.add(rubricItem.id);
  554. });
  555. studentToRubricItemsOverwritten_1 = (new Map());
  556. allStudentsWithRubricItems_1 = new Set();
  557. // ^ {studentId => { Set of rubric ids being uploaded }}
  558. opts.gradeItems.forEach(function (gradeItem) {
  559. var rubricId = gradeItem.rubricId, studentId = gradeItem.studentId;
  560. allStudentsWithRubricItems_1.add(studentId);
  561. // Skip if this item isn't a (real) rubric item
  562. if (!rubricId || realRubricItemIds_1.has(rubricId)) {
  563. return;
  564. }
  565. // Only mark this rubric item as being overwritten if both points and
  566. // comments are being overwritten
  567. if (gradeItem.points === undefined
  568. || gradeItem.points === null
  569. || !gradeItem.comment) {
  570. // Not completely overwriting
  571. return;
  572. }
  573. // Keep track of rubric items that are found
  574. if (!studentToRubricItemsOverwritten_1.has(studentId)) {
  575. // Initialize student map
  576. studentToRubricItemsOverwritten_1.set(studentId, new Set());
  577. }
  578. studentToRubricItemsOverwritten_1.get(studentId).add(rubricId);
  579. });
  580. // > Find students that need to be merged (has some rubric items but not
  581. // completely overwriting all of them)
  582. allStudentsWithRubricItems_1.forEach(function (studentId) {
  583. var numOverwrittenItems = ((studentToRubricItemsOverwritten_1.get(studentId)
  584. || { size: 0 }).size);
  585. if (numOverwrittenItems < numRubricItems_1) {
  586. // Need to merge this student
  587. studentsToMerge.push(studentId);
  588. }
  589. });
  590. _c.label = 2;
  591. case 2: return [4 /*yield*/, (0, parallelLimit_1.default)(studentsToMerge.map(function (studentId) {
  592. return function () { return __awaiter(_this, void 0, void 0, function () {
  593. var _a;
  594. return __generator(this, function (_b) {
  595. return [2 /*return*/, this.api.course.assignment.getSubmission({
  596. studentId: studentId,
  597. courseId: ((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId),
  598. assignmentId: opts.assignmentId,
  599. includeRubricAssessment: true,
  600. excludeUser: true, // Save request space
  601. }, config)];
  602. });
  603. }); };
  604. }), 10)];
  605. case 3:
  606. subs = _c.sent();
  607. params = {};
  608. if (subs.length > 0) {
  609. overwritingMap_1 = {};
  610. // ^ {studentId => rubricId => {
  611. // points: true/false, is being overwritten,
  612. // comment: true/false, is being overwritten
  613. // }}
  614. opts.gradeItems.forEach(function (gradeItem) {
  615. if (!gradeItem.rubricId) {
  616. // No need to keep track of non-rubric item updates
  617. // (these are not being merged)
  618. return;
  619. }
  620. var sid = gradeItem.studentId;
  621. var rid = gradeItem.rubricId;
  622. // Initialize map if needed
  623. if (!overwritingMap_1[sid]) {
  624. overwritingMap_1[sid] = {};
  625. }
  626. if (!overwritingMap_1[sid][rid]) {
  627. overwritingMap_1[sid][rid] = { points: false, comment: false };
  628. }
  629. // Save points and comments
  630. if (gradeItem.points !== undefined) {
  631. overwritingMap_1[sid][rid].points = true;
  632. }
  633. if (gradeItem.comment !== undefined) {
  634. overwritingMap_1[sid][rid].comment = true;
  635. }
  636. });
  637. // Perform actual merge
  638. subs.forEach(function (sub) {
  639. if (!sub.rubric_assessment) {
  640. // No need to merge: submission has no rubric content yet
  641. return;
  642. }
  643. var sid = sub.user_id;
  644. // Loop through rubric items and merge
  645. Object.keys(sub.rubric_assessment).forEach(function (rubricId) {
  646. // Get previous values
  647. var oldPoints = sub.rubric_assessment[rubricId].points;
  648. var oldComment = sub.rubric_assessment[rubricId].comments;
  649. // Check if we're overwriting these values
  650. var overwritePoints;
  651. var overwriteComment;
  652. if (overwritingMap_1[sid] && overwritingMap_1[sid][rubricId]) {
  653. overwritePoints = overwritingMap_1[sid][rubricId].points;
  654. overwriteComment = overwritingMap_1[sid][rubricId].comment;
  655. }
  656. // Add old value
  657. if (oldPoints !== undefined
  658. && oldPoints !== null
  659. && !overwritePoints) {
  660. // We have an old points val and we're not overwriting it
  661. // (include the old points value)
  662. params["grade_data[".concat(sid, "][rubric_assessment][").concat(rubricId, "][points]")] = oldPoints;
  663. }
  664. if (oldComment && !overwriteComment) {
  665. // We have an old comment and we're not overwriting it
  666. // (include the old comment)
  667. params["grade_data[".concat(sid, "][rubric_assessment][").concat(rubricId, "][comments]")] = oldComment;
  668. }
  669. });
  670. });
  671. }
  672. // Add rest of grade item updates to params
  673. opts.gradeItems.forEach(function (gradeItem) {
  674. if (gradeItem.rubricId) {
  675. if (gradeItem.points !== undefined) {
  676. params["grade_data[".concat(gradeItem.studentId, "][rubric_assessment][").concat(gradeItem.rubricId, "][points]")] = gradeItem.points;
  677. }
  678. if (gradeItem.comment) {
  679. params["grade_data[".concat(gradeItem.studentId, "][rubric_assessment][").concat(gradeItem.rubricId, "][comments]")] = gradeItem.comment;
  680. }
  681. }
  682. else {
  683. if (gradeItem.points !== undefined) {
  684. params["grade_data[".concat(gradeItem.studentId, "][posted_grade]")] = gradeItem.points;
  685. }
  686. if (gradeItem.comment) {
  687. params["grade_data[".concat(gradeItem.studentId, "][text_comment]")] = gradeItem.comment;
  688. }
  689. }
  690. });
  691. return [4 /*yield*/, this.visitEndpoint({
  692. params: params,
  693. config: config,
  694. action: 'update student grades, comments, and/or rubric assessments for a specific assignment in a course',
  695. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_b = opts.courseId) !== null && _b !== void 0 ? _b : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/submissions/update_grades"),
  696. method: 'POST',
  697. })];
  698. case 4:
  699. progress = _c.sent();
  700. if (!opts.waitForCompletion) return [3 /*break*/, 6];
  701. return [4 /*yield*/, (0, waitForCompletion_1.default)({
  702. progress: progress,
  703. visitEndpoint: this.visitEndpoint,
  704. timeoutMin: opts.waitForCompletionTimeout,
  705. })];
  706. case 5:
  707. finishedProgress = _c.sent();
  708. return [2 /*return*/, finishedProgress];
  709. case 6: return [2 /*return*/, progress];
  710. }
  711. });
  712. });
  713. };
  714. /*------------------------------------------------------------------------*/
  715. /* Assignment Override Endpoints */
  716. /*------------------------------------------------------------------------*/
  717. /**
  718. * Gets the list of overrides for an assignment
  719. * @author Gabe Abrams
  720. * @method listOverrides
  721. * @memberof api.course.assignment
  722. * @instance
  723. * @async
  724. * @param {object} opts object containing all arguments
  725. * @param {number} opts.assignmentId Canvas assignment id to look up
  726. * @param {number} [opts.courseId=default course id] Canvas course id to query
  727. * @param {APIConfig} [config] custom configuration for this specific endpoint
  728. * call (overwrites defaults that were included when api was initialized)
  729. * @returns {Promise<CanvasAssignmentOverride[]>} list of Canvas AssignmentOverrides {@link https://canvas.instructure.com/doc/api/assignments.html#AssignmentOverride}
  730. */
  731. ECatAssignment.prototype.listOverrides = function (opts, config) {
  732. var _a;
  733. return __awaiter(this, void 0, void 0, function () {
  734. return __generator(this, function (_b) {
  735. return [2 /*return*/, this.visitEndpoint({
  736. config: config,
  737. action: 'get a list of assignment overrides for a specific assignment in a course',
  738. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/overrides"),
  739. method: 'GET',
  740. })];
  741. });
  742. });
  743. };
  744. /**
  745. * Get a specific override on an assignment in a course
  746. * @author Gabe Abrams
  747. * @method getOverride
  748. * @memberof api.course.assignment
  749. * @instance
  750. * @async
  751. * @param {object} opts object containing all arguments
  752. * @param {number} opts.assignmentId Canvas assignment id to query
  753. * @param {number} opts.overrideId Canvas override id to look up
  754. * @param {number} [opts.courseId=default course id] Canvas course id to query
  755. * @param {APIConfig} [config] custom configuration for this specific endpoint
  756. * call (overwrites defaults that were included when api was initialized)
  757. * @returns {Promise<CanvasAssignmentOverride>} Canvas AssignmentOverride {@link https://canvas.instructure.com/doc/api/assignments.html#AssignmentOverride}
  758. */
  759. ECatAssignment.prototype.getOverride = function (opts, config) {
  760. var _a;
  761. return __awaiter(this, void 0, void 0, function () {
  762. return __generator(this, function (_b) {
  763. return [2 /*return*/, this.visitEndpoint({
  764. config: config,
  765. action: 'get a list of assignment overrides for a specific assignment in a course',
  766. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/overrides/").concat(opts.overrideId),
  767. method: 'GET',
  768. })];
  769. });
  770. });
  771. };
  772. /**
  773. * Create assignment override. Note that if any dates (dueAt, unlockAt, or
  774. * lockAt) are left out, they will be set to "none" for the target(s) of this
  775. * override. If dueAt is omitted, the target(s) will have no deadline. If
  776. * unlockAt is omitted, the target(s) will immediately be able to see the
  777. * assignment (even if everyone else has to wait until the unlockAt date). If
  778. * lockAt is omitted, the target(s) will be able to submit at any
  779. * time in the future (even if everyone else can't submit because their lock
  780. * date has passed). In short, it is not recommended to omit dates that are
  781. * defined in the assignment.
  782. * @author Gabe Abrams
  783. * @method createOverride
  784. * @memberof api.course.assignment
  785. * @instance
  786. * @async
  787. * @param {object} opts object containing all arguments
  788. * @param {number} opts.assignmentId Canvas assignment id
  789. * @param {number} [opts.courseId=default course id] Canvas course id
  790. * @param {number[]} [opts.studentIds] List of Canvas student IDs to override
  791. * (Note: either studentIds, groupId, or sectionId must be included)
  792. * @param {number} [opts.groupId] Group to override, must be a group
  793. * assignment (Note: either studentIds, groupId, or sectionId must be
  794. * included)
  795. * @param {number} [opts.sectionId] Section to override (Note: either
  796. * studentIds, groupId, or sectionId must be included)
  797. * @param {string} [opts.title=Override for X students] Title of the
  798. * override
  799. * @param {date} [opts.dueAt=no due date] New due date. If excluded, the
  800. * target(s) of this override have no due date (they can submit whenever they
  801. * want without being marked as late)
  802. * @param {date} [opts.unlockAt=no unlock date] New unlock date. If
  803. * excluded, the target(s) of this override can immediately see the assignment
  804. * (their unlock date is the beginning of time)
  805. * @param {date} [opts.lockAt=no lock date] New lock date. If excluded,
  806. * the target(s) of this override can see and submit the assignment at
  807. * any point in the future (their lock date is the end of time)
  808. * @param {APIConfig} [config] custom configuration for this specific endpoint
  809. * call (overwrites defaults that were included when api was initialized)
  810. * @returns {Promise<CanvasAssignmentOverride>} Canvas AssignmentOverride {@link https://canvas.instructure.com/doc/api/assignments.html#AssignmentOverride}
  811. */
  812. ECatAssignment.prototype.createOverride = function (opts, config) {
  813. var _a;
  814. return __awaiter(this, void 0, void 0, function () {
  815. var title, dueAt, unlockAt, lockAt;
  816. return __generator(this, function (_b) {
  817. title = opts.title;
  818. if (!title) {
  819. title = "Override for ".concat(opts.studentIds.length, " student").concat(utils_1.default.sIfPlural(opts.studentIds.length));
  820. }
  821. dueAt = utils_1.default.includeIfDate(opts.dueAt) || null;
  822. unlockAt = utils_1.default.includeIfDate(opts.unlockAt) || null;
  823. lockAt = utils_1.default.includeIfDate(opts.lockAt) || null;
  824. return [2 /*return*/, this.visitEndpoint({
  825. config: config,
  826. action: 'create a new override for a specific assignment in a course',
  827. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/overrides"),
  828. method: 'POST',
  829. params: {
  830. 'assignment_override[title]': utils_1.default.includeIfTruthy(title),
  831. 'assignment_override[student_ids]': utils_1.default.includeIfTruthy(opts.studentIds),
  832. 'assignment_override[group_id]': utils_1.default.includeIfTruthy(opts.groupId),
  833. 'assignment_override[course_section_id]': utils_1.default.includeIfTruthy(opts.sectionId),
  834. 'assignment_override[due_at]': dueAt,
  835. 'assignment_override[unlock_at]': unlockAt,
  836. 'assignment_override[lock_at]': lockAt,
  837. },
  838. })];
  839. });
  840. });
  841. };
  842. /**
  843. * Update an assignment override. Note: target can only be updated if the
  844. * override is a student override (if this is a group or section override,
  845. * the target remains unchanged).
  846. * Also, note that if any dates (dueAt, unlockAt, or lockAt) are omitted,
  847. * their previous override values will be changed to "none." For instance,
  848. * if the previous override has a dueAt and the update does not, the updated
  849. * override will have no dueAt date (the target(s) of the override will have
  850. * no deadline).
  851. * @author Gabe Abrams
  852. * @method updateOverride
  853. * @memberof api.course.assignment
  854. * @instance
  855. * @async
  856. * @param {object} opts object containing all arguments
  857. * @param {number} opts.assignmentId Canvas assignment id
  858. * @param {number} opts.overrideId the override id to update
  859. * @param {number[]} opts.studentIds List of Canvas student IDs being
  860. * overridden
  861. * @param {number} [opts.courseId=default course id] Canvas course id
  862. * @param {string} [opts.title=current value] New title of the
  863. * override
  864. * @param {date} [opts.dueAt=no due date] New due date. If excluded, the
  865. * target(s) of this override have no due date (they can submit whenever they
  866. * want without being marked as late)
  867. * @param {date} [opts.unlockAt=no unlock date] New unlock date. If
  868. * excluded, the target(s) of this override can immediately see the assignment
  869. * (their unlock date is the beginning of time)
  870. * @param {date} [opts.lockAt=no lock date] New lock date. If excluded,
  871. * the target(s) of this override can see and submit the assignment at
  872. * any point in the future (their lock date is the end of time)
  873. * @param {APIConfig} [config] custom configuration for this specific endpoint
  874. * call (overwrites defaults that were included when api was initialized)
  875. * @returns {Promise<CanvasAssignmentOverride>} Canvas AssignmentOverride {@link https://canvas.instructure.com/doc/api/assignments.html#AssignmentOverride}
  876. */
  877. ECatAssignment.prototype.updateOverride = function (opts, config) {
  878. var _a;
  879. return __awaiter(this, void 0, void 0, function () {
  880. var dueAt, unlockAt, lockAt;
  881. return __generator(this, function (_b) {
  882. dueAt = utils_1.default.includeIfDate(opts.dueAt) || null;
  883. unlockAt = utils_1.default.includeIfDate(opts.unlockAt) || null;
  884. lockAt = utils_1.default.includeIfDate(opts.lockAt) || null;
  885. return [2 /*return*/, this.visitEndpoint({
  886. config: config,
  887. action: 'update an override for a specific assignment in a course',
  888. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/overrides/").concat(opts.overrideId),
  889. method: 'PUT',
  890. params: {
  891. 'assignment_override[title]': utils_1.default.includeIfTruthy(opts.title),
  892. 'assignment_override[student_ids]': utils_1.default.includeIfTruthy(opts.studentIds),
  893. 'assignment_override[due_at]': dueAt,
  894. 'assignment_override[unlock_at]': unlockAt,
  895. 'assignment_override[lock_at]': lockAt,
  896. },
  897. })];
  898. });
  899. });
  900. };
  901. /**
  902. * Deletes an assignment override
  903. * @author Gabe Abrams
  904. * @method deleteOverride
  905. * @memberof api.course.assignment
  906. * @instance
  907. * @async
  908. * @param {object} opts object containing all arguments
  909. * @param {number} opts.assignmentId Canvas assignment id to query
  910. * @param {number} opts.overrideId Canvas override id to look up
  911. * @param {number} [opts.courseId=default course id] Canvas course id to query
  912. * @param {APIConfig} [config] custom configuration for this specific endpoint
  913. * call (overwrites defaults that were included when api was initialized)
  914. * @returns {Promise<CanvasAssignmentOverride>} Canvas AssignmentOverride {@link https://canvas.instructure.com/doc/api/assignments.html#AssignmentOverride}
  915. */
  916. ECatAssignment.prototype.deleteOverride = function (opts, config) {
  917. var _a;
  918. return __awaiter(this, void 0, void 0, function () {
  919. return __generator(this, function (_b) {
  920. return [2 /*return*/, this.visitEndpoint({
  921. config: config,
  922. action: 'delete an override for a specific assignment in a course',
  923. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/overrides/").concat(opts.overrideId),
  924. method: 'DELETE',
  925. })];
  926. });
  927. });
  928. };
  929. /*------------------------------------------------------------------------*/
  930. /* Assignment Submission Endpoints */
  931. /*------------------------------------------------------------------------*/
  932. /**
  933. * Lists the submissions to a specific assignment in a course. If the assignment
  934. * has anonymous grading turned on, to exclude the test user, we will also
  935. * pull the list of students in the course. If including the user object for
  936. * an anonymously graded assignment, fake user objects will be created where
  937. * each submissions[i].user object contains a isAnonymousUser boolean that is
  938. * true
  939. * @author Gabe Abrams
  940. * @method listSubmissions
  941. * @memberof api.course.assignment
  942. * @instance
  943. * @async
  944. * @param {object} opts object containing all arguments
  945. * @param {number} opts.assignmentId The Canvas assignment Id to query
  946. * @param {number} [opts.courseId=default course id] Canvas course Id
  947. * @param {boolean} [opts.includeComments] If truthy, includes all
  948. * comments on submissions
  949. * @param {boolean} [opts.includeRubricAssessment] If truthy,
  950. * includes rubric assessments: breakdown of score for each rubric item
  951. * @param {boolean} [opts.excludeUser] If truthy, excludes
  952. * submission[i].user value with the submission's user information
  953. * @param {boolean} [opts.includeTestStudent] If truthy, includes
  954. * dummy submission by test student (student view) if there is one. Note:
  955. * if anonymous grading is enabled for this assignment, includeTestStudent
  956. * will be true because we don't know which student is the test student
  957. * @param {APIConfig} [config] custom configuration for this specific endpoint
  958. * call (overwrites defaults that were included when api was initialized)
  959. * @returns {Promise<CanvasSubmission[]>} list of Canvas submissions {@link https://canvas.instructure.com/doc/api/submissions.html#Submission}
  960. */
  961. ECatAssignment.prototype.listSubmissions = function (opts, config) {
  962. var _a;
  963. return __awaiter(this, void 0, void 0, function () {
  964. var fetchUser, subs, realSubs;
  965. return __generator(this, function (_b) {
  966. switch (_b.label) {
  967. case 0:
  968. fetchUser = (!opts.includeTestStudent
  969. || !opts.excludeUser);
  970. return [4 /*yield*/, this.visitEndpoint({
  971. config: config,
  972. action: 'list the submissions to a specific assignment in a course',
  973. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/submissions"),
  974. method: 'GET',
  975. params: {
  976. include: utils_1.default.genIncludesList({
  977. submission_comments: opts.includeComments,
  978. rubric_assessment: opts.includeRubricAssessment,
  979. user: fetchUser,
  980. }),
  981. },
  982. })];
  983. case 1:
  984. subs = _b.sent();
  985. // Filter test student if applicable
  986. if (!opts.includeTestStudent) {
  987. // Handle empty list case
  988. if (!subs || subs.length === 0) {
  989. return [2 /*return*/, []];
  990. }
  991. realSubs = subs.filter(function (sub) {
  992. return (!sub.user
  993. || sub.user.name !== 'Test Student');
  994. });
  995. // Finish
  996. return [2 /*return*/, realSubs];
  997. }
  998. // Not filtering out test student. Just return subs
  999. return [2 /*return*/, subs];
  1000. }
  1001. });
  1002. });
  1003. };
  1004. /**
  1005. * Lists the submissions for a batch of assignment/students in a course
  1006. * @author Gabe Abrams
  1007. * @method listAllSubmissions
  1008. * @memberof api.course.assignment
  1009. * @instance
  1010. * @async
  1011. * @param {object} [opts] object containing all arguments
  1012. * @param {number} [opts.courseId=default course id] Canvas course Id
  1013. * @param {number[]} [opts.studentIds=all students] a list of
  1014. * specific students to pull submissions for
  1015. * @param {number[]} [opts.assignmentIds=all assignments] a list of
  1016. * assignments to get submissions for
  1017. * @param {Date} [opts.submittedSince=beginning of time] Exclude
  1018. * submissions that were not submitted or were submitted before this date
  1019. * @param {Date} [opts.gradedSince=beginning of time] Exclude
  1020. * submissions that were not graded or were graded before this date
  1021. * @param {string} [opts.workflowState=all workflows] a workflow state
  1022. * to filter by. Allowed values: 'submitted', 'unsubmitted', 'graded', or
  1023. * 'pending_review'
  1024. * @param {string} [opts.enrollmentState=all states except deleted] an
  1025. * enrollment state to filter by. Allowed values: 'active' or 'concluded'
  1026. * @param {boolean} [opts.includeSubmissionHistory] if true, submission
  1027. * history is included
  1028. * @param {boolean} [opts.includeComments] if true, includes all comments
  1029. * on submissions
  1030. * @param {boolean} [opts.includeRubricAssessment] if true,
  1031. * rubric assessment is included
  1032. * @param {boolean} [opts.includeAssignment] if true, the assignment is
  1033. * included for each submission
  1034. * @param {boolean} [opts.includeTotalScores] if true, include the total
  1035. * scores
  1036. * @param {boolean} [opts.includeVisibility] if true, include visibility
  1037. * @param {boolean} [opts.includeUser] if true, include the user info
  1038. * with each submission
  1039. * @param {APIConfig} [config] custom configuration for this specific endpoint
  1040. * call (overwrites defaults that were included when api was initialized)
  1041. * @returns {Promise<CanvasSubmission[]>} list of submissions {@link https://canvas.instructure.com/doc/api/submissions.html#Submission}
  1042. */
  1043. ECatAssignment.prototype.listAllSubmissions = function (opts, config) {
  1044. var _a;
  1045. if (opts === void 0) { opts = {}; }
  1046. return __awaiter(this, void 0, void 0, function () {
  1047. return __generator(this, function (_b) {
  1048. return [2 /*return*/, this.visitEndpoint({
  1049. config: config,
  1050. action: 'list a batch of submissions in a course',
  1051. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/students/submissions"),
  1052. method: 'GET',
  1053. params: {
  1054. student_ids: (opts.studentIds
  1055. ? opts.studentIds
  1056. : ['all']),
  1057. assignment_ids: opts.assignmentIds,
  1058. submitted_since: utils_1.default.includeIfDate(opts.submittedSince),
  1059. graded_since: utils_1.default.includeIfDate(opts.gradedSince),
  1060. workflow_state: utils_1.default.includeIfTruthy(opts.workflowState),
  1061. enrollment_state: utils_1.default.includeIfTruthy(opts.enrollmentState),
  1062. include: utils_1.default.genIncludesList({
  1063. submission_history: opts.includeSubmissionHistory,
  1064. submission_comments: opts.includeComments,
  1065. rubric_assessment: opts.includeRubricAssessment,
  1066. assignment: opts.includeAssignment,
  1067. total_scores: opts.includeTotalScores,
  1068. visibility: opts.includeVisibility,
  1069. user: opts.includeUser,
  1070. }),
  1071. },
  1072. })];
  1073. });
  1074. });
  1075. };
  1076. /**
  1077. * Gets a single submission for an assignment
  1078. * @author Gabe Abrams
  1079. * @method getSubmission
  1080. * @memberof api.course.assignment
  1081. * @instance
  1082. * @async
  1083. * @param {object} opts object containing all arguments
  1084. * @param {number} opts.assignmentId The Canvas assignment Id
  1085. * @param {number} opts.studentId The Canvas student Id
  1086. * @param {number} [opts.courseId=default course id] Canvas course Id
  1087. * @param {boolean} [opts.includeComments] If truthy, includes all
  1088. * comments on submissions
  1089. * @param {boolean} [opts.includeRubricAssessment] If truthy,
  1090. * includes rubric assessments: breakdown of score for each rubric item
  1091. * @param {boolean} [opts.excludeUser] If truthy, excludes
  1092. * submission[i].user value with the submission's user information
  1093. * @param {APIConfig} [config] custom configuration for this specific endpoint
  1094. * call (overwrites defaults that were included when api was initialized)
  1095. * @returns {Promise<CanvasSubmission>} Canvas submission {@link https://canvas.instructure.com/doc/api/submissions.html#Submission}
  1096. */
  1097. ECatAssignment.prototype.getSubmission = function (opts, config) {
  1098. var _a;
  1099. return __awaiter(this, void 0, void 0, function () {
  1100. return __generator(this, function (_b) {
  1101. return [2 /*return*/, this.visitEndpoint({
  1102. config: config,
  1103. action: 'get a specific submission to an assignment in a course',
  1104. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/submissions/").concat(opts.studentId),
  1105. method: 'GET',
  1106. params: {
  1107. include: utils_1.default.genIncludesList({
  1108. submission_comments: opts.includeComments,
  1109. rubric_assessment: opts.includeRubricAssessment,
  1110. user: !opts.excludeUser,
  1111. }),
  1112. },
  1113. })];
  1114. });
  1115. });
  1116. };
  1117. /**
  1118. * Creates a text submission on behalf of the current user
  1119. * @author Gabe Abrams
  1120. * @method createTextSubmission
  1121. * @memberof api.course.assignment
  1122. * @instance
  1123. * @async
  1124. * @param {object} opts object containing all arguments
  1125. * @param {number} opts.assignmentId The Canvas assignment Id
  1126. * @param {string} opts.text The text body of the submission
  1127. * @param {number} [opts.courseId=default course id] Canvas course Id
  1128. * @param {string} [opts.comment] A text student comment to include
  1129. * @param {APIConfig} [config] custom configuration for this specific endpoint
  1130. * call (overwrites defaults that were included when api was initialized)
  1131. * @returns {Promise<CanvasSubmission>} Canvas submission {@link https://canvas.instructure.com/doc/api/submissions.html#Submission}
  1132. */
  1133. ECatAssignment.prototype.createTextSubmission = function (opts, config) {
  1134. var _a;
  1135. return __awaiter(this, void 0, void 0, function () {
  1136. return __generator(this, function (_b) {
  1137. return [2 /*return*/, this.visitEndpoint({
  1138. config: config,
  1139. action: 'create a text submission to a specific assignment in a course on behalf of the current user',
  1140. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/submissions"),
  1141. method: 'POST',
  1142. params: {
  1143. 'comment[text_comment]': utils_1.default.includeIfTruthy(opts.comment),
  1144. 'submission[body]': opts.text,
  1145. 'submission[submission_type]': 'online_text_entry',
  1146. },
  1147. })];
  1148. });
  1149. });
  1150. };
  1151. /**
  1152. * Creates a url submission on behalf of the current user
  1153. * @author Gabe Abrams
  1154. * @method createURLSubmission
  1155. * @memberof api.course.assignment
  1156. * @instance
  1157. * @async
  1158. * @param {object} opts object containing all arguments
  1159. * @param {number} opts.assignmentId The Canvas assignment Id
  1160. * @param {string} opts.url The url of the submission
  1161. * @param {number} [opts.courseId=default course id] Canvas course Id
  1162. * @param {string} [opts.comment] A text student comment to include
  1163. * @param {APIConfig} [config] custom configuration for this specific endpoint
  1164. * call (overwrites defaults that were included when api was initialized)
  1165. * @returns {Promise<CanvasSubmission>} Canvas submission {@link https://canvas.instructure.com/doc/api/submissions.html#Submission}
  1166. */
  1167. ECatAssignment.prototype.createURLSubmission = function (opts, config) {
  1168. var _a;
  1169. return __awaiter(this, void 0, void 0, function () {
  1170. return __generator(this, function (_b) {
  1171. return [2 /*return*/, this.visitEndpoint({
  1172. config: config,
  1173. action: 'create a url submission to a specific assignment in a course on behalf of the current user',
  1174. path: "".concat(API_PREFIX_1.default, "/courses/").concat((_a = opts.courseId) !== null && _a !== void 0 ? _a : this.defaultCourseId, "/assignments/").concat(opts.assignmentId, "/submissions"),
  1175. method: 'POST',
  1176. params: {
  1177. 'comment[text_comment]': utils_1.default.includeIfTruthy(opts.comment),
  1178. 'submission[url]': opts.url,
  1179. 'submission[submission_type]': 'online_url',
  1180. },
  1181. })];
  1182. });
  1183. });
  1184. };
  1185. return ECatAssignment;
  1186. }(EndpointCategory_1.default));
  1187. /*------------------------------------------------------------------------*/
  1188. /* Export */
  1189. /*------------------------------------------------------------------------*/
  1190. exports.default = ECatAssignment;
  1191. //# sourceMappingURL=ECatAssignment.js.map