mirror of
https://github.com/abapGit/abapGit.git
synced 2025-04-30 11:46:38 +08:00
308 lines
7.8 KiB
ABAP
308 lines
7.8 KiB
ABAP
CLASS zcl_abapgit_serialize DEFINITION
|
|
PUBLIC
|
|
CREATE PUBLIC .
|
|
|
|
PUBLIC SECTION.
|
|
|
|
METHODS constructor.
|
|
METHODS on_end_of_task
|
|
IMPORTING
|
|
!p_task TYPE clike .
|
|
METHODS serialize
|
|
IMPORTING
|
|
!it_tadir TYPE zif_abapgit_definitions=>ty_tadir_tt
|
|
!iv_language TYPE langu DEFAULT sy-langu
|
|
!ii_log TYPE REF TO zif_abapgit_log OPTIONAL
|
|
!iv_force_sequential TYPE abap_bool DEFAULT abap_false
|
|
RETURNING
|
|
VALUE(rt_files) TYPE zif_abapgit_definitions=>ty_files_item_tt
|
|
RAISING
|
|
zcx_abapgit_exception .
|
|
|
|
PROTECTED SECTION.
|
|
|
|
CLASS-DATA gv_max_threads TYPE i .
|
|
DATA mt_files TYPE zif_abapgit_definitions=>ty_files_item_tt .
|
|
DATA mv_free TYPE i .
|
|
DATA mi_log TYPE REF TO zif_abapgit_log .
|
|
DATA mv_group TYPE rzlli_apcl .
|
|
|
|
METHODS add_to_return
|
|
IMPORTING
|
|
!iv_path TYPE string
|
|
!is_fils_item TYPE zcl_abapgit_objects=>ty_serialization .
|
|
METHODS run_parallel
|
|
IMPORTING
|
|
!is_tadir TYPE zif_abapgit_definitions=>ty_tadir
|
|
!iv_language TYPE langu
|
|
!iv_task TYPE sychar32
|
|
RAISING
|
|
zcx_abapgit_exception .
|
|
METHODS run_sequential
|
|
IMPORTING
|
|
!is_tadir TYPE zif_abapgit_definitions=>ty_tadir
|
|
!iv_language TYPE langu
|
|
RAISING
|
|
zcx_abapgit_exception .
|
|
METHODS determine_max_threads
|
|
IMPORTING
|
|
!iv_force_sequential TYPE abap_bool DEFAULT abap_false
|
|
RETURNING
|
|
VALUE(rv_threads) TYPE i
|
|
RAISING
|
|
zcx_abapgit_exception .
|
|
PRIVATE SECTION.
|
|
|
|
METHODS is_merged
|
|
RETURNING
|
|
VALUE(rv_result) TYPE abap_bool .
|
|
ENDCLASS.
|
|
|
|
|
|
|
|
CLASS ZCL_ABAPGIT_SERIALIZE IMPLEMENTATION.
|
|
|
|
|
|
METHOD add_to_return.
|
|
|
|
FIELD-SYMBOLS: <ls_file> LIKE LINE OF is_fils_item-files,
|
|
<ls_return> LIKE LINE OF mt_files.
|
|
|
|
|
|
LOOP AT is_fils_item-files ASSIGNING <ls_file>.
|
|
APPEND INITIAL LINE TO mt_files ASSIGNING <ls_return>.
|
|
<ls_return>-file = <ls_file>.
|
|
<ls_return>-file-path = iv_path.
|
|
<ls_return>-item = is_fils_item-item.
|
|
ENDLOOP.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD constructor.
|
|
|
|
DATA lo_settings TYPE REF TO zcl_abapgit_settings.
|
|
|
|
lo_settings = zcl_abapgit_persist_settings=>get_instance( )->read( ).
|
|
|
|
IF is_merged( ) = abap_true
|
|
OR lo_settings->get_parallel_proc_disabled( ) = abap_true.
|
|
gv_max_threads = 1.
|
|
ENDIF.
|
|
|
|
mv_group = 'parallel_generators' ##NO_TEXT.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD determine_max_threads.
|
|
|
|
IF iv_force_sequential = abap_true.
|
|
rv_threads = 1.
|
|
RETURN.
|
|
ENDIF.
|
|
|
|
IF gv_max_threads >= 1.
|
|
* SPBT_INITIALIZE gives error PBT_ENV_ALREADY_INITIALIZED if called
|
|
* multiple times in same session
|
|
rv_threads = gv_max_threads.
|
|
RETURN.
|
|
ENDIF.
|
|
|
|
CALL FUNCTION 'FUNCTION_EXISTS'
|
|
EXPORTING
|
|
funcname = 'Z_ABAPGIT_SERIALIZE_PARALLEL'
|
|
EXCEPTIONS
|
|
function_not_exist = 1
|
|
OTHERS = 2.
|
|
IF sy-subrc <> 0.
|
|
gv_max_threads = 1.
|
|
ELSE.
|
|
* todo, add possibility to set group name in user exit
|
|
CALL FUNCTION 'SPBT_INITIALIZE'
|
|
EXPORTING
|
|
group_name = mv_group
|
|
IMPORTING
|
|
free_pbt_wps = gv_max_threads
|
|
EXCEPTIONS
|
|
invalid_group_name = 1
|
|
internal_error = 2
|
|
pbt_env_already_initialized = 3
|
|
currently_no_resources_avail = 4
|
|
no_pbt_resources_found = 5
|
|
cant_init_different_pbt_groups = 6
|
|
OTHERS = 7.
|
|
IF sy-subrc <> 0.
|
|
* fallback to running sequentially. If SPBT_INITIALIZE fails, check transactions
|
|
* RZ12, SM50, SM21, SARFC
|
|
gv_max_threads = 1.
|
|
ENDIF.
|
|
ENDIF.
|
|
|
|
IF gv_max_threads > 1.
|
|
gv_max_threads = gv_max_threads - 1.
|
|
ENDIF.
|
|
|
|
ASSERT gv_max_threads >= 1.
|
|
|
|
IF gv_max_threads > 32.
|
|
* https://en.wikipedia.org/wiki/Amdahl%27s_law
|
|
gv_max_threads = 32.
|
|
ENDIF.
|
|
|
|
rv_threads = gv_max_threads.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD is_merged.
|
|
|
|
DATA lo_marker TYPE REF TO data ##NEEDED.
|
|
|
|
TRY.
|
|
CREATE DATA lo_marker TYPE REF TO ('LIF_ABAPMERGE_MARKER') ##no_text.
|
|
"No exception --> marker found
|
|
rv_result = abap_true.
|
|
|
|
CATCH cx_sy_create_data_error ##no_handler.
|
|
ENDTRY.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD on_end_of_task.
|
|
|
|
DATA: lv_result TYPE xstring,
|
|
lv_path TYPE string,
|
|
ls_fils_item TYPE zcl_abapgit_objects=>ty_serialization.
|
|
|
|
|
|
RECEIVE RESULTS FROM FUNCTION 'Z_ABAPGIT_SERIALIZE_PARALLEL'
|
|
IMPORTING
|
|
ev_result = lv_result
|
|
ev_path = lv_path
|
|
EXCEPTIONS
|
|
error = 1
|
|
OTHERS = 2.
|
|
IF sy-subrc <> 0.
|
|
IF NOT mi_log IS INITIAL.
|
|
mi_log->add_error( |{ sy-msgv1 }{ sy-msgv2 }{ sy-msgv3 }{ sy-msgv3 }| ).
|
|
ENDIF.
|
|
ELSE.
|
|
IMPORT data = ls_fils_item FROM DATA BUFFER lv_result. "#EC CI_SUBRC
|
|
ASSERT sy-subrc = 0.
|
|
add_to_return( is_fils_item = ls_fils_item
|
|
iv_path = lv_path ).
|
|
ENDIF.
|
|
|
|
mv_free = mv_free + 1.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD run_parallel.
|
|
|
|
DATA: lv_msg TYPE c LENGTH 100,
|
|
lv_free LIKE mv_free.
|
|
|
|
|
|
ASSERT mv_free > 0.
|
|
|
|
DO.
|
|
CALL FUNCTION 'Z_ABAPGIT_SERIALIZE_PARALLEL'
|
|
STARTING NEW TASK iv_task
|
|
DESTINATION IN GROUP mv_group
|
|
CALLING on_end_of_task ON END OF TASK
|
|
EXPORTING
|
|
iv_obj_type = is_tadir-object
|
|
iv_obj_name = is_tadir-obj_name
|
|
iv_devclass = is_tadir-devclass
|
|
iv_language = iv_language
|
|
iv_path = is_tadir-path
|
|
EXCEPTIONS
|
|
system_failure = 1 MESSAGE lv_msg
|
|
communication_failure = 2 MESSAGE lv_msg
|
|
resource_failure = 3
|
|
OTHERS = 4.
|
|
IF sy-subrc = 3.
|
|
lv_free = mv_free.
|
|
WAIT UNTIL mv_free <> lv_free UP TO 1 SECONDS.
|
|
CONTINUE.
|
|
ELSEIF sy-subrc <> 0.
|
|
ASSERT lv_msg = '' AND 0 = 1.
|
|
ENDIF.
|
|
EXIT.
|
|
ENDDO.
|
|
|
|
mv_free = mv_free - 1.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD run_sequential.
|
|
|
|
DATA: lx_error TYPE REF TO zcx_abapgit_exception,
|
|
ls_fils_item TYPE zcl_abapgit_objects=>ty_serialization.
|
|
|
|
|
|
ls_fils_item-item-obj_type = is_tadir-object.
|
|
ls_fils_item-item-obj_name = is_tadir-obj_name.
|
|
ls_fils_item-item-devclass = is_tadir-devclass.
|
|
|
|
TRY.
|
|
ls_fils_item = zcl_abapgit_objects=>serialize(
|
|
is_item = ls_fils_item-item
|
|
iv_language = iv_language ).
|
|
|
|
add_to_return( is_fils_item = ls_fils_item
|
|
iv_path = is_tadir-path ).
|
|
CATCH zcx_abapgit_exception INTO lx_error.
|
|
IF NOT mi_log IS INITIAL.
|
|
mi_log->add_error( lx_error->get_text( ) ).
|
|
ENDIF.
|
|
ENDTRY.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD serialize.
|
|
|
|
DATA: lv_max TYPE i,
|
|
li_progress TYPE REF TO zif_abapgit_progress.
|
|
|
|
FIELD-SYMBOLS: <ls_tadir> LIKE LINE OF it_tadir.
|
|
|
|
|
|
CLEAR mt_files.
|
|
|
|
lv_max = determine_max_threads( iv_force_sequential ).
|
|
mv_free = lv_max.
|
|
mi_log = ii_log.
|
|
|
|
li_progress = zcl_abapgit_progress=>get_instance( lines( it_tadir ) ).
|
|
|
|
LOOP AT it_tadir ASSIGNING <ls_tadir>.
|
|
|
|
li_progress->show(
|
|
iv_current = sy-tabix
|
|
iv_text = |Serialize { <ls_tadir>-obj_name }, { lv_max } threads| ) ##NO_TEXT.
|
|
|
|
IF lv_max = 1.
|
|
run_sequential(
|
|
is_tadir = <ls_tadir>
|
|
iv_language = iv_language ).
|
|
ELSE.
|
|
run_parallel(
|
|
is_tadir = <ls_tadir>
|
|
iv_task = |{ sy-tabix }|
|
|
iv_language = iv_language ).
|
|
WAIT UNTIL mv_free > 0 UP TO 10 SECONDS.
|
|
ENDIF.
|
|
ENDLOOP.
|
|
|
|
WAIT UNTIL mv_free = lv_max UP TO 10 SECONDS.
|
|
rt_files = mt_files.
|
|
|
|
ENDMETHOD.
|
|
ENDCLASS.
|