Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merge Numbas Integration feature into 8.0.x #443

Merged
merged 130 commits into from
Aug 9, 2024
Merged
Show file tree
Hide file tree
Changes from 53 commits
Commits
Show all changes
130 commits
Select commit Hold shift + click to select a range
3ae95f8
new:numbas feature backend intital commit
maddernd Dec 5, 2023
fa2de36
new:numbas feature backend intital commit
maddernd Dec 5, 2023
f035c44
fix test_attempts api paths
ublefo Mar 11, 2024
4165b10
Enforce authentication for the test_attempts api
ublefo Mar 13, 2024
55fcbc5
Modify the api to fetch test attempts with task id
ublefo Mar 13, 2024
061f49f
Numbas data management features
ublefo Mar 20, 2024
d53610a
feat: add Numbas config options to task def backend
satikaj Mar 22, 2024
dade800
Remove foreign key reference and add missing columns to csv file
ublefo Mar 22, 2024
aa8926e
refactor: simplify Numbas config storage
satikaj Mar 22, 2024
3f19ffa
fix: reset Numbas configs if no zip file has been uploaded
satikaj Mar 23, 2024
54c27ce
feat: change Numbas time delay config to enable incremental delays
satikaj Mar 24, 2024
20d5265
fix: expose enable Numbas test config to all users
satikaj Mar 27, 2024
5d80830
fix: use correct Numbas data path in Numbas api
satikaj Mar 28, 2024
0cc4915
fix: use custom endpoint for Numbas
satikaj Apr 9, 2024
27253bd
fix: consolidate numbas api endpoints
ublefo Apr 9, 2024
ff5ff62
fix: refactor numbas config reset logic
ublefo Apr 15, 2024
89a6615
fix: add missing numbas config fields to fix unit tests
ublefo Apr 15, 2024
ee992f4
fix: use correct endpoint url and include exam result for numbas test…
satikaj Apr 30, 2024
a7c4006
fix: use test attempt entity in file instead
satikaj Apr 30, 2024
b4d3f9d
fix: temporarily disable auth and fix test attempt lookup
ublefo May 4, 2024
e650ef1
refactor: modify test attempt files to match PoC
satikaj May 5, 2024
3f5aa2b
feat: add numbas task comment on test completion
satikaj May 6, 2024
26d75f5
feat: restrict test attempts by limit and comments to when test is co…
satikaj May 6, 2024
d9f640f
refactor: rewrite test_attempt api and generalize scorm variables
ublefo May 11, 2024
8047f4c
chore: address rubocop errors
ublefo May 14, 2024
3539d95
fix: add allow review property to task def related files
satikaj May 14, 2024
0812e20
fix: post scorm comment after test attempt termination
ublefo May 14, 2024
b498924
fix: rename entity file and add update fields in task spreadsheet
ublefo May 14, 2024
d71ea14
fix: enforce attempt limit
ublefo May 14, 2024
910eecd
fix: expose scorm configs to student
satikaj May 15, 2024
2a04a06
fix: use project and task def to fix issue where task is undefined on…
satikaj Jun 1, 2024
8c9a68b
feat: enable reviewing, passing, and deleting test attempts
satikaj Jun 1, 2024
c5055b8
feat: enable students to request extra scorm attempt
satikaj Jun 5, 2024
1e00d41
test: add sample numbas zip
satikaj Jun 5, 2024
28f3279
fix: calculate attempt number and limit instead of using stored int
satikaj Jun 5, 2024
7d31f7c
feat: add test attempt auth
satikaj Jun 6, 2024
273b62e
refactor: remove attempt number in task comment text
satikaj Jun 11, 2024
3d44ef2
fix: grant same number of extra attempts as scorm limit
satikaj Jun 11, 2024
1240b3f
fix: prevent new attempt if last is incomplete or passed
satikaj Jun 11, 2024
1ae0347
fix: prevent scorm extensions if no attempt limit
satikaj Jun 15, 2024
4255347
fix: check for attempts before accessing properties
satikaj Jun 15, 2024
fce7e75
feat: add attribute to allow file upload before scorm is passed
satikaj Jun 15, 2024
69053ee
fix: change scorm comment text
satikaj Jun 18, 2024
4139690
fix: add scorm bypass to excel file
satikaj Jun 18, 2024
fc8134a
feat: create unique token for scorm asset retrieval
satikaj Jun 26, 2024
c854fc1
test: add scorm unit tests
satikaj Jul 2, 2024
6108b52
fix: check if no old scorm tokens exist
satikaj Jul 2, 2024
3d86d04
refactor: remove stale todo comment
satikaj Jul 2, 2024
c5240d8
fix: validate attempt id
satikaj Jul 2, 2024
5db5f35
fix: reorder columns for csv export
satikaj Jul 2, 2024
08a0090
fix: use unique perms for scorm test retrieval
satikaj Jul 2, 2024
835d2b0
Merge branch 'new/numbas-integration' into 8.0.x
satikaj Jul 3, 2024
52450be
fix: ensure main convenor validation on change only
macite Jul 3, 2024
e740d82
fix: streamline archiving units in maintenance task
macite Jul 3, 2024
eacbac1
fix: remove rollover teaching period
macite Jul 3, 2024
7f3b752
feat: allow new unit code to be provided to rollover
macite Jul 3, 2024
9bdbe0e
chore(release): 8.0.9
macite Jul 3, 2024
6373eee
fix: ensure task definitions render when upload requirements are nil
macite Jul 4, 2024
4359cf2
refactor: correct syntax for rubocop
satikaj Jul 5, 2024
a49fc8c
fix: turn it in enabled property
macite Jul 10, 2024
bb29f84
fix: ensure sidekiq moves to Rails root before task pdf creation
macite Jul 10, 2024
5b8f5d3
fix: ensure turn it in actions only occur when tii enabled
macite Jul 10, 2024
9288fae
Merge branch '8.0.x' of https://github.com/macite/doubtfire-api into …
macite Jul 10, 2024
e3fab0d
fix: ensure logger only logs to stdout in development
macite Jul 10, 2024
58d8281
fix: guard access of pwd incase pwd is invalid
macite Jul 10, 2024
32b1d9f
fix: ensure failure to send email is handled
macite Jul 10, 2024
460681a
chore(release): 8.0.10
macite Jul 10, 2024
210c3f4
test: add tests for multi unit sync and connection errors in sync
macite Jul 12, 2024
d4808b0
feat: ensure deakin sync retries failed connections
macite Jul 12, 2024
6aea5c1
Merge branch '8.0.x' of https://github.com/macite/doubtfire-api into …
macite Jul 12, 2024
439669a
chore(release): 8.0.11
macite Jul 12, 2024
2fcb477
quality: ensure tii enabled always through class
macite Jul 15, 2024
e01ed19
feat: allow register webhooks to be controlled via config
macite Jul 15, 2024
61f2f49
chore(release): 8.0.12
macite Jul 15, 2024
ebbacb9
fix: check need to register webhooks in tii action
macite Jul 15, 2024
4dae884
fix: ensure tii module looks for appropriate user
macite Jul 15, 2024
0bf29eb
feat: add env var to configure log to stdout
macite Jul 15, 2024
6b08013
feat: check that old tii submissions upload when eula accepted
macite Jul 16, 2024
57acea2
Merge branch '8.0.x' of https://github.com/macite/doubtfire-api into …
macite Jul 16, 2024
03e9214
fix: limit tii action log to 25 entries
macite Jul 16, 2024
be21763
fix: ensure webhook test will register hooks
macite Jul 16, 2024
789fbad
fix: tii action retry resets retries
macite Jul 16, 2024
1eadf31
Merge branch '8.0.x' of https://github.com/macite/doubtfire-api into …
macite Jul 16, 2024
4f3979b
fix: merge student records for deakin students
macite Jul 17, 2024
6c74d7d
chore(release): 8.0.13
macite Jul 17, 2024
de3ec39
fix: add logging info to debug hmac issues
macite Jul 18, 2024
7d47eda
feat: allow logging to stdout using env var
macite Jul 18, 2024
cc2f674
chore: fix test to use assert_not and ensure actions are destroyed
macite Jul 18, 2024
72b4c69
chore: switch bundle exec to rails in shell scripts
macite Jul 18, 2024
15ca645
chore: update portfolio autogen shell script
macite Jul 18, 2024
68a7248
chore(release): 8.0.14
macite Jul 18, 2024
a249662
fix: correct turn it in hmac calculation
macite Jul 20, 2024
1b2a43c
fix: ensure pax header is not included in tex on 2nd pass
macite Jul 22, 2024
38a6922
chore(release): 8.0.15
macite Jul 22, 2024
232dcaa
fix: ensure comment added on task pdf convert fail
macite Jul 22, 2024
b70aeac
chore(release): 8.0.16
macite Jul 22, 2024
b868799
quality: address minor cop issue with automated error messages
macite Jul 22, 2024
4e38e6c
test: fix issues with file locn and tii features
macite Jul 22, 2024
908c803
chore(release): 8.0.17
macite Jul 22, 2024
752f26e
test: groups can be locked
macite Jul 24, 2024
7e9adaa
feat: add ability to manually remove webhooks from rails console
macite Jul 25, 2024
efa6692
fix: ensure tii signing secret is sent as a base64 string
macite Jul 25, 2024
547dbb9
chore(release): 8.0.18
macite Jul 25, 2024
d84856b
fix: remove newlines from signing key base64 encoding
macite Jul 26, 2024
b047e4c
chore: update log level for tii hook debug
macite Jul 26, 2024
166a0ad
quality: add tests for filename in task definitions
macite Jul 26, 2024
cea12e5
fix: ensure accept submission checks number of files
macite Jul 26, 2024
edcd36a
chore(release): 8.0.19
macite Jul 26, 2024
527b810
test: fix tests to conform to new validations
macite Jul 26, 2024
70f095c
fix: webhook registration key check
macite Jul 26, 2024
875b831
chore(release): 8.0.20
macite Jul 29, 2024
6753d80
fix: delay pdf generation to ensure sufficient time for async task to…
macite Jul 30, 2024
3a4eb51
chore(release): 8.0.21
macite Jul 30, 2024
7dfd547
test: task pdf download
macite Aug 1, 2024
33ee3ce
fix: ensure scoop files checks files are a hash
macite Aug 1, 2024
3e0a1ba
fix: add awaiting processing pdf
macite Aug 1, 2024
e3d36c2
fix: only try overseer assessment when overseer enabled
macite Aug 1, 2024
218afb9
fix: change zip of new upload to avoid loss
macite Aug 1, 2024
1e3acd4
feat: add email on accept submission error
macite Aug 1, 2024
8139f41
feat: report high usage on database timeout
macite Aug 1, 2024
758a51d
fix: avoid attempts to read negative size in file stream helper
macite Aug 1, 2024
eb8859b
chore(release): 8.0.22
macite Aug 1, 2024
842f233
fix: remove global error and report failures to admin user for tii
macite Aug 5, 2024
cbec03d
fix: ensure folders are removed when we move files with file helper
macite Aug 5, 2024
7023650
chore(release): 8.0.23
macite Aug 5, 2024
edbacb3
Merge branch '8.0.x' of https://github.com/satikaj/doubtfire-api into…
macite Aug 5, 2024
7de1a8f
quality: remove unused scorm entity
macite Aug 5, 2024
e7a6eed
fix: update auth token to work with scorm and general
macite Aug 8, 2024
02af2cf
test: fix scorm and auth tests with new tokens
macite Aug 9, 2024
8059213
fix: preload unit in test attempt and ensure limit flexibility in val…
macite Aug 9, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions app/api/api_root.rb
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,7 @@ class ApiRoot < Grape::API
mount BreaksApi
mount DiscussionCommentApi
mount ExtensionCommentsApi
mount ScormExtensionCommentsApi
mount GroupSetsApi
mount LearningOutcomesApi
mount LearningAlignmentApi
Expand All @@ -78,6 +79,8 @@ class ApiRoot < Grape::API
mount Tii::TiiGroupAttachmentApi
mount Tii::TiiActionApi

mount ScormApi
mount TestAttemptsApi
mount CampusesPublicApi
mount CampusesAuthenticatedApi
mount TutorialsApi
Expand All @@ -98,6 +101,7 @@ class ApiRoot < Grape::API
AuthenticationHelpers.add_auth_to BreaksApi
AuthenticationHelpers.add_auth_to DiscussionCommentApi
AuthenticationHelpers.add_auth_to ExtensionCommentsApi
AuthenticationHelpers.add_auth_to ScormExtensionCommentsApi
AuthenticationHelpers.add_auth_to GroupSetsApi
AuthenticationHelpers.add_auth_to LearningOutcomesApi
AuthenticationHelpers.add_auth_to LearningAlignmentApi
Expand All @@ -124,6 +128,8 @@ class ApiRoot < Grape::API
AuthenticationHelpers.add_auth_to UnitRolesApi
AuthenticationHelpers.add_auth_to UnitsApi
AuthenticationHelpers.add_auth_to WebcalApi
AuthenticationHelpers.add_auth_to ScormApi
AuthenticationHelpers.add_auth_to TestAttemptsApi

add_swagger_documentation \
base_path: nil,
Expand Down
18 changes: 18 additions & 0 deletions app/api/authentication_api.rb
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@
class AuthenticationApi < Grape::API
helpers LogHelper
helpers AuthenticationHelpers
helpers AuthorisationHelpers

#
# Sign in - only mounted if AAF auth is NOT used
Expand Down Expand Up @@ -368,4 +369,21 @@ class AuthenticationApi < Grape::API

present nil
end

desc 'Get SCORM authentication token'
get '/auth/scorm' do
if authenticated?
unless authorise? current_user, User, :get_scorm_token
error!({ error: 'You cannot get SCORM tokens' }, 403)
end

token = current_user.auth_tokens.find_by(token_type: 'scorm')
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What can a scorm token access? It looks like there is not scope applied to these at the moment... where are they used in the auth process?

if token.nil? || token.auth_token_expiry <= Time.zone.now
token&.destroy
token = current_user.generate_scorm_authentication_token!
end

present :scorm_auth_token, token.authentication_token
end
end
end
6 changes: 6 additions & 0 deletions app/api/entities/scorm_entity.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
module Entities
class ScormEntity < Grape::Entity
macite marked this conversation as resolved.
Show resolved Hide resolved
expose :file_content, documentation: { type: 'string', desc: 'File content' }
expose :content_type, documentation: { type: 'string', desc: 'Content type' }
end
end
6 changes: 6 additions & 0 deletions app/api/entities/task_definition_entity.rb
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,12 @@ def staff?(my_role)
expose :has_task_sheet?, as: :has_task_sheet
expose :has_task_resources?, as: :has_task_resources
expose :has_task_assessment_resources?, as: :has_task_assessment_resources, if: ->(unit, options) { staff?(options[:my_role]) }
expose :has_scorm_data?, as: :has_scorm_data
expose :scorm_enabled
expose :scorm_allow_review
expose :scorm_bypass_test
expose :scorm_time_delay_enabled
expose :scorm_attempt_limit
expose :is_graded
expose :max_quality_pts
expose :overseer_image_id, if: ->(unit, options) { staff?(options[:my_role]) }, expose_nil: false
Expand Down
1 change: 1 addition & 0 deletions app/api/entities/task_entity.rb
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ class TaskEntity < Grape::Entity
end

expose :extensions
expose :scorm_extensions

expose :times_assessed
expose :grade, expose_nil: false
Expand Down
12 changes: 12 additions & 0 deletions app/api/entities/test_attempt_entity.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
module Entities
class TestAttemptEntity < Grape::Entity
expose :id
expose :task_id
expose :attempted_time
expose :terminated
expose :success_status
expose :score_scaled
expose :completion_status
expose :cmi_datamodel
end
end
71 changes: 71 additions & 0 deletions app/api/scorm_api.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
require 'grape'
require 'zip'
require 'mime/types'
class ScormApi < Grape::API
# Include the AuthenticationHelpers for authentication functionality
helpers AuthenticationHelpers
helpers AuthorisationHelpers

before do
authenticated?
end

helpers do
# Method to stream a file from a zip archive at the specified path
# @param zip_path [String] the path to the zip archive
# @param file_path [String] the path of the file within the zip archive
def stream_file_from_zip(zip_path, file_path)
file_stream = nil

logger.debug "Streaming zip file at #{zip_path}"
# Get an input stream for the requested file within the ZIP archive
Zip::File.open(zip_path) do |zip_file|
zip_file.each do |entry|
next unless entry.name == file_path
logger.debug "Found file #{file_path} from SCORM container"
file_stream = entry.get_input_stream
break
end
end

# If the file was not found in the ZIP archive, return a 404 response
unless file_stream
error!({ error: 'File not found' }, 404)
end

# Set the content type based on the file extension
content_type = MIME::Types.type_for(file_path).first.content_type
logger.debug "Content type: #{content_type}"

# Set the content type header
header 'Content-Type', content_type

# Set cache control header to prevent caching
header 'Cache-Control', 'no-cache, no-store, must-revalidate'

# Set the body to the contents of the file_stream and return the response
body file_stream.read
end
end

desc 'Serve SCORM content'
params do
requires :task_def_id, type: Integer, desc: 'Task Definition ID to get SCORM test data for'
end
get '/scorm/:task_def_id/:username/:auth_token/*file_path' do
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should any of the api calls here be limited to scorm tokens?

task_def = TaskDefinition.find(params[:task_def_id])

unless authorise? current_user, task_def.unit, :get_unit
error!({ error: 'You cannot access SCORM tests of unit' }, 403)
end

env['api.format'] = :txt
if task_def.has_scorm_data?
zip_path = task_def.task_scorm_data
content_type 'application/octet-stream'
stream_file_from_zip(zip_path, params[:file_path])
else
error!({ error: 'SCORM data does not exist.' }, 404)
end
end
end
54 changes: 54 additions & 0 deletions app/api/scorm_extension_comments_api.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
require 'grape'

class ScormExtensionCommentsApi < Grape::API
helpers AuthenticationHelpers
helpers AuthorisationHelpers

desc 'Request a scorm extension for a task'
params do
requires :comment, type: String, desc: 'The details of the request'
end
post '/projects/:project_id/task_def_id/:task_definition_id/request_scorm_extension' do
project = Project.find(params[:project_id])
task_definition = project.unit.task_definitions.find(params[:task_definition_id])
task = project.task_for_task_definition(task_definition)

# check permissions using specific permission has with addition of request extension if allowed in unit
unless authorise? current_user, task, :request_scorm_extension
error!({ error: 'Not authorised to request a scorm extension for this task' }, 403)
end

if task_definition.scorm_attempt_limit == 0
error!({ message: 'This task allows unlimited attempts to complete the test' }, 400)
return
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

error! throws an exception - so this is unreachable... we should look and remove these unnecessary return statements

end

result = task.apply_for_scorm_extension(current_user, params[:comment])
present result.serialize(current_user), Grape::Presenters::Presenter
end

desc 'Assess a scorm extension for a task'
params do
requires :granted, type: Boolean, desc: 'Assess a scorm extension'
end
put '/projects/:project_id/task_def_id/:task_definition_id/assess_scorm_extension/:task_comment_id' do
project = Project.find(params[:project_id])
task_definition = project.unit.task_definitions.find(params[:task_definition_id])
task = project.task_for_task_definition(task_definition)

unless authorise? current_user, task, :assess_scorm_extension
error!({ error: 'Not authorised to assess a scorm extension for this task' }, 403)
end

task_comment = task.all_comments.find(params[:task_comment_id]).becomes(ScormExtensionComment)

unless task_comment.assess_scorm_extension(current_user, params[:granted])
if task_comment.errors.count >= 1
error!({ error: task_comment.errors.full_messages.first }, 403)
else
error!({ error: 'Error saving scorm extension' }, 403)
end
end
present task_comment.serialize(current_user), Grape::Presenters::Presenter
end
end
95 changes: 95 additions & 0 deletions app/api/task_definitions_api.rb
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,11 @@ class TaskDefinitionsApi < Grape::API
requires :restrict_status_updates, type: Boolean, desc: 'Restrict updating of the status to staff'
optional :upload_requirements, type: String, desc: 'Task file upload requirements'
requires :plagiarism_warn_pct, type: Integer, desc: 'The percent at which to record and warn about plagiarism'
requires :scorm_enabled, type: Boolean, desc: 'Whether SCORM assessment is enabled for this task'
requires :scorm_allow_review, type: Boolean, desc: 'Whether a student is allowed to review their completed test attempts'
requires :scorm_bypass_test, type: Boolean, desc: 'Whether a student is allowed to upload files before passing SCORM test'
requires :scorm_time_delay_enabled, type: Boolean, desc: 'Whether there is an incremental time delay between SCORM test attempts'
requires :scorm_attempt_limit, type: Integer, desc: 'The number of times a SCORM test can be attempted'
requires :is_graded, type: Boolean, desc: 'Whether or not this task definition is a graded task'
requires :max_quality_pts, type: Integer, desc: 'A range for quality points when quality is assessed'
optional :assessment_enabled, type: Boolean, desc: 'Enable or disable assessment'
Expand Down Expand Up @@ -55,6 +60,11 @@ class TaskDefinitionsApi < Grape::API
:abbreviation,
:restrict_status_updates,
:plagiarism_warn_pct,
:scorm_enabled,
:scorm_allow_review,
:scorm_bypass_test,
:scorm_time_delay_enabled,
:scorm_attempt_limit,
:is_graded,
:max_quality_pts,
:assessment_enabled,
Expand Down Expand Up @@ -106,6 +116,11 @@ class TaskDefinitionsApi < Grape::API
optional :restrict_status_updates, type: Boolean, desc: 'Restrict updating of the status to staff'
optional :upload_requirements, type: String, desc: 'Task file upload requirements'
optional :plagiarism_warn_pct, type: Integer, desc: 'The percent at which to record and warn about plagiarism'
optional :scorm_enabled, type: Boolean, desc: 'Whether or not SCORM test assessment is enabled for this task'
optional :scorm_allow_review, type: Boolean, desc: 'Whether a student is allowed to review their completed test attempts'
optional :scorm_bypass_test, type: Boolean, desc: 'Whether a student is allowed to upload files before passing SCORM test'
optional :scorm_time_delay_enabled, type: Boolean, desc: 'Whether or not there is an incremental time delay between SCORM test attempts'
optional :scorm_attempt_limit, type: Integer, desc: 'The number of times a SCORM test can be attempted'
optional :is_graded, type: Boolean, desc: 'Whether or not this task definition is a graded task'
optional :max_quality_pts, type: Integer, desc: 'A range for quality points when quality is assessed'
optional :assessment_enabled, type: Boolean, desc: 'Enable or disable assessment'
Expand Down Expand Up @@ -134,6 +149,11 @@ class TaskDefinitionsApi < Grape::API
:abbreviation,
:restrict_status_updates,
:plagiarism_warn_pct,
:scorm_enabled,
:scorm_allow_review,
:scorm_bypass_test,
:scorm_time_delay_enabled,
:scorm_attempt_limit,
:is_graded,
:max_quality_pts,
:assessment_enabled,
Expand Down Expand Up @@ -611,4 +631,79 @@ class TaskDefinitionsApi < Grape::API

stream_file path
end

desc 'Upload the SCORM container (zip file) for a task'
params do
requires :unit_id, type: Integer, desc: 'The related unit'
requires :task_def_id, type: Integer, desc: 'The related task definition'
requires :file, type: File, desc: 'The SCORM data container'
end
post '/units/:unit_id/task_definitions/:task_def_id/scorm_data' do
unit = Unit.find(params[:unit_id])

unless authorise? current_user, unit, :add_task_def
error!({ error: 'Not authorised to upload SCORM data for the unit' }, 403)
end

task_def = unit.task_definitions.find(params[:task_def_id])

if params[:file].blank?
error!({ error: "No file uploaded" }, 403)
end

file_path = params[:file][:tempfile].path

check_mime_against_list! file_path, 'zip', ['application/zip', 'multipart/x-gzip', 'multipart/x-zip', 'application/x-gzip', 'application/octet-stream']

# Actually import...
task_def.add_scorm_data(file_path)
true
end

desc 'Download the SCORM test data'
params do
requires :unit_id, type: Integer, desc: 'The unit to modify tasks for'
requires :task_def_id, type: Integer, desc: 'The task definition to get the SCORM test data of'
end
get '/units/:unit_id/task_definitions/:task_def_id/scorm_data' do
unit = Unit.find(params[:unit_id])
task_def = unit.task_definitions.find(params[:task_def_id])

unless authorise? current_user, unit, :get_unit
error!({ error: 'Not authorised to download task details of unit' }, 403)
end

if task_def.has_scorm_data?
path = task_def.task_scorm_data
content_type 'application/octet-stream'
header['Content-Disposition'] = "attachment; filename=#{task_def.abbreviation}-scorm.zip"
else
path = Rails.root.join('public/resources/FileNotFound.pdf')
content_type 'application/pdf'
header['Content-Disposition'] = 'attachment; filename=FileNotFound.pdf'
end
header['Access-Control-Expose-Headers'] = 'Content-Disposition'

env['api.format'] = :binary
File.read(path)
end

desc 'Remove the SCORM test data for a given task'
params do
requires :unit_id, type: Integer, desc: 'The related unit'
requires :task_def_id, type: Integer, desc: 'The related task definition'
end
delete '/units/:unit_id/task_definitions/:task_def_id/scorm_data' do
unit = Unit.find(params[:unit_id])

unless authorise? current_user, unit, :add_task_def
error!({ error: 'Not authorised to remove task SCORM data of unit' }, 403)
end

task_def = unit.task_definitions.find(params[:task_def_id])

# Actually remove...
task_def.remove_scorm_data
true
end
end
3 changes: 2 additions & 1 deletion app/api/tasks_api.rb
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,8 @@ class TasksApi < Grape::API
task_definition_id: task.task_definition_id,
status: TaskStatus.id_to_key(task.task_status_id),
due_date: task.due_date,
extensions: task.extensions
extensions: task.extensions,
scorm_extensions: task.scorm_extensions
}
end

Expand Down
Loading