@@ -2,8 +2,8 @@ module Api
2
2
module V1
3
3
class SiteController < ApiBaseController
4
4
before_action :set_current_api_user!
5
- before_action :authenticate_api_user! , only : [ :download_data , :stream_data , :get_study_analysis_config ,
6
- :submit_study_analysis , :get_study_submissions ,
5
+ before_action :authenticate_api_user! , only : [ :download_data , :stream_data , :submit_differential_expression ,
6
+ :get_study_analysis_config , : submit_study_analysis, :get_study_submissions ,
7
7
:get_study_submission , :sync_submission_outputs ]
8
8
before_action :set_study , except : [ :studies , :check_terra_tos_acceptance , :analyses , :get_analysis , :renew_user_access_token ]
9
9
before_action :set_analysis_configuration , only : [ :get_analysis , :get_study_analysis_config ]
@@ -403,6 +403,151 @@ def stream_data
403
403
end
404
404
end
405
405
406
+ swagger_path '/site/studies/{accession}/differential_expression' do
407
+ operation :post do
408
+ key :tags , [
409
+ 'Site'
410
+ ]
411
+ key :summary , 'Submit a differential expression calculation request'
412
+ key :description , 'Request differential expression calculations for a given cluster/annotation in a study'
413
+ key :operationId , 'site_study_submit_differential_expression_path'
414
+ parameter do
415
+ key :name , :accession
416
+ key :in , :path
417
+ key :description , 'Accession of Study to use'
418
+ key :required , true
419
+ key :type , :string
420
+ end
421
+ parameter do
422
+ key :name , :de_job
423
+ key :type , :object
424
+ key :in , :body
425
+ schema do
426
+ property :cluster_name do
427
+ key :description , 'Name of cluster group. Use "_default" to use the default cluster'
428
+ key :required , true
429
+ key :type , :string
430
+ end
431
+ property :annotation_name do
432
+ key :description , 'Name of annotation'
433
+ key :required , true
434
+ key :type , :string
435
+ end
436
+ property :annotation_scope do
437
+ key :description , 'Scope of annotation. One of "study" or "cluster".'
438
+ key :type , :string
439
+ key :required , true
440
+ key :enum , Api ::V1 ::Visualization ::AnnotationsController ::VALID_SCOPE_VALUES
441
+ end
442
+ property :de_type do
443
+ key :description , 'Type of differential expression analysis. Either "rest" (one-vs-rest) or "pairwise"'
444
+ key :type , :string
445
+ key :required , true
446
+ key :enum , %w[ rest pairwise ]
447
+ end
448
+ property :group1 do
449
+ key :description , 'First group for pairwise analysis (optional)'
450
+ key :type , :string
451
+ end
452
+ property :group2 do
453
+ key :description , 'Second group for pairwise analysis (optional)'
454
+ key :type , :string
455
+ end
456
+ end
457
+ end
458
+ response 204 do
459
+ key :description , 'Job successfully submitted'
460
+ end
461
+ response 401 do
462
+ key :description , ApiBaseController . unauthorized
463
+ end
464
+ response 403 do
465
+ key :description , ApiBaseController . forbidden ( 'view study, study has author DE' )
466
+ end
467
+ response 404 do
468
+ key :description , ApiBaseController . not_found ( Study , 'Cluster' , 'Annotation' )
469
+ end
470
+ response 406 do
471
+ key :description , ApiBaseController . not_acceptable
472
+ end
473
+ response 409 do
474
+ key :description , "Results are processing or already exist"
475
+ end
476
+ response 410 do
477
+ key :description , ApiBaseController . resource_gone
478
+ end
479
+ response 422 do
480
+ key :description , "Job parameters failed validation"
481
+ end
482
+ response 429 do
483
+ key :description , 'Weekly user quota exceeded'
484
+ end
485
+ end
486
+ end
487
+
488
+ def submit_differential_expression
489
+ # disallow DE calculation requests for studies with author DE
490
+ if @study . differential_expression_results . where ( is_author_de : true ) . any?
491
+ render json : {
492
+ error : 'User requests are disabled for this study as it contains author-supplied differential expression results'
493
+ } , status : 403 and return
494
+ end
495
+
496
+ # check user quota before proceeding
497
+ if DifferentialExpressionService . job_exceeds_quota? ( current_api_user )
498
+ # minimal log props to help gauge overall user interest, as well as annotation/de types
499
+ log_props = {
500
+ studyAccession : @study . accession , annotationName : params [ :annotation_name ] , de_type : params [ :de_type ]
501
+ }
502
+ MetricsService . log ( 'quota-exceeded-de' , log_props , current_api_user , request :)
503
+ current_quota = DifferentialExpressionService . get_weekly_user_quota
504
+ render json : { error : "You have exceeded your weekly quota of #{ current_quota } requests" } ,
505
+ status : 429 and return
506
+ end
507
+
508
+ cluster_name = params [ :cluster_name ]
509
+ cluster = cluster_name == '_default' ? @study . default_cluster : @study . cluster_groups . by_name ( cluster_name )
510
+ render json : { error : "Requested cluster #{ cluster_name } not found" } , status : 404 and return if cluster . nil?
511
+
512
+ annotation_name = params [ :annotation_name ]
513
+ annotation_scope = params [ :annotation_scope ]
514
+ de_type = params [ :de_type ]
515
+ pairwise = de_type == 'pairwise'
516
+ group1 = params [ :group1 ]
517
+ group2 = params [ :group2 ]
518
+ annotation = AnnotationVizService . get_selected_annotation (
519
+ @study , cluster :, annot_name : annotation_name , annot_type : 'group' , annot_scope : annotation_scope
520
+ )
521
+ render json : { error : 'No matching annotation found' } , status : 404 and return if annotation . nil?
522
+
523
+ de_params = { annotation_name :, annotation_scope :, de_type :, group1 :, group2 : }
524
+
525
+ # check if these results already exist
526
+ # for pairwise, also check if requested comparisons exist
527
+ result = DifferentialExpressionResult . find_by (
528
+ study : @study , cluster_group : cluster , annotation_name :, annotation_scope :, is_author_de : false
529
+ )
530
+ if result && ( !pairwise || ( pairwise && result . has_pairwise_comparison? ( group1 , group2 ) ) )
531
+ render json : { error : "Requested results already exist" } , status : 409 and return
532
+ end
533
+
534
+ begin
535
+ submitted = DifferentialExpressionService . run_differential_expression_job (
536
+ cluster , @study , current_api_user , **de_params
537
+ )
538
+ if submitted
539
+ DifferentialExpressionService . increment_user_quota ( current_api_user )
540
+ head 204
541
+ else
542
+ # submitted: false here means that there is a matching running DE job
543
+ render json : { error : "Requested results are processing - please check back later" } , status : 409
544
+ end
545
+ rescue ArgumentError => e
546
+ # job parameters failed to validate
547
+ render json : { error : e . message } , status : 422 and return
548
+ end
549
+ end
550
+
406
551
swagger_path '/site/analyses' do
407
552
operation :get do
408
553
key :tags , [
@@ -1136,6 +1281,10 @@ def get_download_quota
1136
1281
end
1137
1282
end
1138
1283
1284
+ def de_job_params
1285
+ params . require ( :de_job ) . permit ( :cluster_name , :annotation_name , :annotation_scope , :de_type , :group1 , :group2 )
1286
+ end
1287
+
1139
1288
# update AnalysisSubmissions when loading study analysis tab
1140
1289
# will not backfill existing workflows to keep our submission history clean
1141
1290
def update_analysis_submission ( submission )
0 commit comments