diff --git a/src/pybind/mgr/dashboard/controllers/iscsi.py b/src/pybind/mgr/dashboard/controllers/iscsi.py index 1714f9ae66f..7bca1f15ef6 100644 --- a/src/pybind/mgr/dashboard/controllers/iscsi.py +++ b/src/pybind/mgr/dashboard/controllers/iscsi.py @@ -19,7 +19,7 @@ from ..services.iscsi_cli import IscsiGatewaysConfig from ..services.rbd import format_bitmask from ..services.tcmu_service import TcmuService from ..exceptions import DashboardException -from ..tools import TaskManager +from ..tools import str_to_bool, TaskManager @UiApiController('/iscsi', Scope.ISCSI) @@ -75,7 +75,29 @@ class IscsiUi(BaseController): @Endpoint() @ReadPermission def settings(self): - return IscsiClient.instance().get_settings() + settings = IscsiClient.instance().get_settings() + if 'target_controls_limits' in settings: + target_default_controls = settings['target_default_controls'] + for ctrl_k, ctrl_v in target_default_controls.items(): + limits = settings['target_controls_limits'].get(ctrl_k, {}) + if 'type' not in limits: + # default + limits['type'] = 'int' + # backward compatibility + if target_default_controls[ctrl_k] in ['Yes', 'No']: + limits['type'] = 'bool' + target_default_controls[ctrl_k] = str_to_bool(ctrl_v) + settings['target_controls_limits'][ctrl_k] = limits + if 'disk_controls_limits' in settings: + for backstore, disk_controls_limits in settings['disk_controls_limits'].items(): + disk_default_controls = settings['disk_default_controls'][backstore] + for ctrl_k, ctrl_v in disk_default_controls.items(): + limits = disk_controls_limits.get(ctrl_k, {}) + if 'type' not in limits: + # default + limits['type'] = 'int' + settings['disk_controls_limits'][backstore][ctrl_k] = limits + return settings @Endpoint() @ReadPermission @@ -747,9 +769,6 @@ class IscsiTarget(RESTController): groups.append(group) groups = IscsiTarget._sorted_groups(groups) target_controls = target_config['controls'] - for key, value in target_controls.items(): - if isinstance(value, bool): - target_controls[key] = 'Yes' if value else 'No' acl_enabled = target_config['acl_enabled'] target = { 'target_iqn': target_iqn, diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/block.module.ts b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/block.module.ts index 29b5692e5be..a2ba84cacfe 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/block.module.ts +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/block.module.ts @@ -15,6 +15,7 @@ import { TooltipModule } from 'ngx-bootstrap/tooltip'; import { ActionLabels, URLVerbs } from '../../shared/constants/app.constants'; import { FeatureTogglesGuardService } from '../../shared/services/feature-toggles-guard.service'; import { SharedModule } from '../../shared/shared.module'; +import { IscsiSettingComponent } from './iscsi-setting/iscsi-setting.component'; import { IscsiTabsComponent } from './iscsi-tabs/iscsi-tabs.component'; import { IscsiTargetDetailsComponent } from './iscsi-target-details/iscsi-target-details.component'; import { IscsiTargetDiscoveryModalComponent } from './iscsi-target-discovery-modal/iscsi-target-discovery-modal.component'; @@ -69,6 +70,7 @@ import { RbdTrashRestoreModalComponent } from './rbd-trash-restore-modal/rbd-tra declarations: [ RbdListComponent, IscsiComponent, + IscsiSettingComponent, IscsiTabsComponent, IscsiTargetListComponent, RbdDetailsComponent, diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.html b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.html new file mode 100644 index 00000000000..f3a1d5dc2f5 --- /dev/null +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.html @@ -0,0 +1,57 @@ +
+ + + + + + + + + +
+
+ + +
+
+ + +
+
+
+ + + Must be greater than or equal to {{ limits['min'] }}. + + + Must be less than or equal to {{ limits['max'] }}. + +
diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.scss b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.scss new file mode 100644 index 00000000000..e69de29bb2d diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.spec.ts b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.spec.ts new file mode 100644 index 00000000000..3f272d9bab5 --- /dev/null +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.spec.ts @@ -0,0 +1,37 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; +import { FormControl, NgForm, ReactiveFormsModule } from '@angular/forms'; + +import { configureTestBed } from '../../../../testing/unit-test-helper'; +import { CdFormGroup } from '../../../shared/forms/cd-form-group'; +import { SharedModule } from '../../../shared/shared.module'; +import { IscsiSettingComponent } from './iscsi-setting.component'; + +describe('IscsiSettingComponent', () => { + let component: IscsiSettingComponent; + let fixture: ComponentFixture; + + configureTestBed({ + imports: [SharedModule, ReactiveFormsModule], + declarations: [IscsiSettingComponent] + }); + + beforeEach(() => { + fixture = TestBed.createComponent(IscsiSettingComponent); + component = fixture.componentInstance; + component.settingsForm = new CdFormGroup({ + max_data_area_mb: new FormControl() + }); + component.formDir = new NgForm([], []); + component.setting = 'max_data_area_mb'; + component.limits = { + type: 'int', + min: 1, + max: 2048 + }; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.ts b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.ts new file mode 100644 index 00000000000..ade8ae4c5b2 --- /dev/null +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-setting/iscsi-setting.component.ts @@ -0,0 +1,31 @@ +import { Component, Input, OnInit } from '@angular/core'; +import { NgForm, Validators } from '@angular/forms'; + +import { CdFormGroup } from '../../../shared/forms/cd-form-group'; + +@Component({ + selector: 'cd-iscsi-setting', + templateUrl: './iscsi-setting.component.html', + styleUrls: ['./iscsi-setting.component.scss'] +}) +export class IscsiSettingComponent implements OnInit { + @Input() + settingsForm: CdFormGroup; + @Input() + formDir: NgForm; + @Input() + setting: string; + @Input() + limits: object; + + ngOnInit() { + const validators = []; + if ('min' in this.limits) { + validators.push(Validators.min(this.limits['min'])); + } + if ('max' in this.limits) { + validators.push(Validators.max(this.limits['max'])); + } + this.settingsForm.get(this.setting).setValidators(validators); + } +} diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-details/iscsi-target-details.component.ts b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-details/iscsi-target-details.component.ts index a3a9c2dd253..5b17c8b9ff8 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-details/iscsi-target-details.component.ts +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-details/iscsi-target-details.component.ts @@ -8,6 +8,7 @@ import { TableComponent } from '../../../shared/datatable/table/table.component' import { Icons } from '../../../shared/enum/icons.enum'; import { CdTableColumn } from '../../../shared/models/cd-table-column'; import { CdTableSelection } from '../../../shared/models/cd-table-selection'; +import { BooleanTextPipe } from '../../../shared/pipes/boolean-text.pipe'; import { IscsiBackstorePipe } from '../../../shared/pipes/iscsi-backstore.pipe'; @Component({ @@ -42,7 +43,11 @@ export class IscsiTargetDetailsComponent implements OnChanges, OnInit { title: string; tree: TreeModel; - constructor(private i18n: I18n, private iscsiBackstorePipe: IscsiBackstorePipe) {} + constructor( + private i18n: I18n, + private iscsiBackstorePipe: IscsiBackstorePipe, + private booleanTextPipe: BooleanTextPipe + ) {} ngOnInit() { this.columns = [ @@ -249,6 +254,13 @@ export class IscsiTargetDetailsComponent implements OnChanges, OnInit { }; } + private format(value) { + if (typeof value === 'boolean') { + return this.booleanTextPipe.transform(value); + } + return value; + } + onNodeSelected(e: NodeEvent) { if (e.node.id) { this.title = e.node.value; @@ -257,10 +269,11 @@ export class IscsiTargetDetailsComponent implements OnChanges, OnInit { if (e.node.id === 'root') { this.columns[2].isHidden = false; this.data = _.map(this.settings.target_default_controls, (value, key) => { + value = this.format(value); return { displayName: key, default: value, - current: tempData[key] || value + current: !_.isUndefined(tempData[key]) ? this.format(tempData[key]) : value }; }); // Target level authentication was introduced in ceph-iscsi config v11 @@ -276,10 +289,13 @@ export class IscsiTargetDetailsComponent implements OnChanges, OnInit { } else if (e.node.id.toString().startsWith('disk_')) { this.columns[2].isHidden = false; this.data = _.map(this.settings.disk_default_controls[tempData.backstore], (value, key) => { + value = this.format(value); return { displayName: key, default: value, - current: !_.isUndefined(tempData.controls[key]) ? tempData.controls[key] : value + current: !_.isUndefined(tempData.controls[key]) + ? this.format(tempData.controls[key]) + : value }; }); this.data.push({ @@ -293,7 +309,7 @@ export class IscsiTargetDetailsComponent implements OnChanges, OnInit { return { displayName: key, default: undefined, - current: value + current: this.format(value) }; }); } diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-form/iscsi-target-form.component.spec.ts b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-form/iscsi-target-form.component.spec.ts index 320afbd0af3..da510c86107 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-form/iscsi-target-form.component.spec.ts +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-form/iscsi-target-form.component.spec.ts @@ -31,7 +31,7 @@ describe('IscsiTargetFormComponent', () => { target_default_controls: { cmdsn_depth: 128, dataout_timeout: 20, - immediate_data: 'Yes' + immediate_data: true }, required_rbd_features: { 'backstore:1': 0, diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.html b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.html index ecfc41c317b..0540a022403 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.html +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.html @@ -35,20 +35,10 @@
- - - - Must be greater than or equal to {{ disk_controls_limits[bs][setting.key]['min'] }}. - - - Must be less than or equal to {{ disk_controls_limits[bs][setting.key]['max'] }}. - - {{ helpText[setting.key]?.help }} +
diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.spec.ts b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.spec.ts index c25b73c0057..4bb02ace02c 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.spec.ts +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.spec.ts @@ -7,6 +7,7 @@ import { BsModalRef } from 'ngx-bootstrap/modal'; import { configureTestBed, i18nProviders } from '../../../../testing/unit-test-helper'; import { SharedModule } from '../../../shared/shared.module'; +import { IscsiSettingComponent } from '../iscsi-setting/iscsi-setting.component'; import { IscsiTargetImageSettingsModalComponent } from './iscsi-target-image-settings-modal.component'; describe('IscsiTargetImageSettingsModalComponent', () => { @@ -14,7 +15,7 @@ describe('IscsiTargetImageSettingsModalComponent', () => { let fixture: ComponentFixture; configureTestBed({ - declarations: [IscsiTargetImageSettingsModalComponent], + declarations: [IscsiTargetImageSettingsModalComponent, IscsiSettingComponent], imports: [SharedModule, ReactiveFormsModule, HttpClientTestingModule, RouterTestingModule], providers: [BsModalRef, i18nProviders] }); @@ -34,6 +35,27 @@ describe('IscsiTargetImageSettingsModalComponent', () => { baz: 3 } }; + component.disk_controls_limits = { + 'backstore:1': { + foo: { + min: 1, + max: 512, + type: 'int' + }, + bar: { + min: 1, + max: 512, + type: 'int' + } + }, + 'backstore:2': { + baz: { + min: 1, + max: 512, + type: 'int' + } + } + }; component.backstores = ['backstore:1', 'backstore:2']; component.ngOnInit(); diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.ts b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.ts index e2687300dc9..3918aabfce1 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.ts +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-image-settings-modal/iscsi-target-image-settings-modal.component.ts @@ -1,5 +1,5 @@ import { Component, OnInit } from '@angular/core'; -import { FormControl, Validators } from '@angular/forms'; +import { FormControl } from '@angular/forms'; import * as _ from 'lodash'; import { BsModalRef } from 'ngx-bootstrap/modal'; @@ -20,37 +20,27 @@ export class IscsiTargetImageSettingsModalComponent implements OnInit { backstores: any; settingsForm: CdFormGroup; - helpText: any; constructor(public modalRef: BsModalRef, public iscsiService: IscsiService) {} ngOnInit() { - this.helpText = this.iscsiService.imageAdvancedSettings; - const fg = { backstore: new FormControl(this.imagesSettings[this.image]['backstore']) }; _.forEach(this.backstores, (backstore) => { const model = this.imagesSettings[this.image][backstore] || {}; _.forIn(this.disk_default_controls[backstore], (_value, key) => { - const validators = []; - if (this.disk_controls_limits && key in this.disk_controls_limits[backstore]) { - if ('min' in this.disk_controls_limits[backstore][key]) { - validators.push(Validators.min(this.disk_controls_limits[backstore][key]['min'])); - } - if ('max' in this.disk_controls_limits[backstore][key]) { - validators.push(Validators.max(this.disk_controls_limits[backstore][key]['max'])); - } - } - fg[key] = new FormControl(model[key], { - validators: validators - }); + fg[key] = new FormControl(model[key]); }); }); this.settingsForm = new CdFormGroup(fg); } + getDiskControlLimits(backstore, setting) { + return this.disk_controls_limits[backstore][setting]; + } + save() { const backstore = this.settingsForm.controls['backstore'].value; const settings = {}; diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.html b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.html index 8d00c80ce47..3382b616124 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.html +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.html @@ -14,44 +14,10 @@
- - - - Must be greater than or equal to {{ target_controls_limits[setting.key]['min'] }}. - - - Must be less than or equal to {{ target_controls_limits[setting.key]['max'] }}. - - - -
-
- - -
-
- - -
-
- - {{ helpText[setting.key]?.help }} +
diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.spec.ts b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.spec.ts index 6666731d359..521bcbf22d8 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.spec.ts +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.spec.ts @@ -7,6 +7,7 @@ import { BsModalRef } from 'ngx-bootstrap/modal'; import { configureTestBed, i18nProviders } from '../../../../testing/unit-test-helper'; import { SharedModule } from '../../../shared/shared.module'; +import { IscsiSettingComponent } from '../iscsi-setting/iscsi-setting.component'; import { IscsiTargetIqnSettingsModalComponent } from './iscsi-target-iqn-settings-modal.component'; describe('IscsiTargetIqnSettingsModalComponent', () => { @@ -14,7 +15,7 @@ describe('IscsiTargetIqnSettingsModalComponent', () => { let fixture: ComponentFixture; configureTestBed({ - declarations: [IscsiTargetIqnSettingsModalComponent], + declarations: [IscsiTargetIqnSettingsModalComponent, IscsiSettingComponent], imports: [SharedModule, ReactiveFormsModule, HttpClientTestingModule, RouterTestingModule], providers: [BsModalRef, i18nProviders] }); @@ -26,7 +27,24 @@ describe('IscsiTargetIqnSettingsModalComponent', () => { component.target_default_controls = { cmdsn_depth: 1, dataout_timeout: 2, - first_burst_length: 'Yes' + first_burst_length: true + }; + component.target_controls_limits = { + cmdsn_depth: { + min: 1, + max: 512, + type: 'int' + }, + dataout_timeout: { + min: 2, + max: 60, + type: 'int' + }, + first_burst_length: { + max: 16777215, + min: 512, + type: 'int' + } }; component.ngOnInit(); fixture.detectChanges(); diff --git a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.ts b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.ts index 622beb4619e..25ed109ac77 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.ts +++ b/src/pybind/mgr/dashboard/frontend/src/app/ceph/block/iscsi-target-iqn-settings-modal/iscsi-target-iqn-settings-modal.component.ts @@ -1,5 +1,5 @@ import { Component, OnInit } from '@angular/core'; -import { FormControl, Validators } from '@angular/forms'; +import { FormControl } from '@angular/forms'; import * as _ from 'lodash'; import { BsModalRef } from 'ngx-bootstrap/modal'; @@ -18,25 +18,13 @@ export class IscsiTargetIqnSettingsModalComponent implements OnInit { target_controls_limits: any; settingsForm: CdFormGroup; - helpText: any; constructor(public modalRef: BsModalRef, public iscsiService: IscsiService) {} ngOnInit() { const fg = {}; - this.helpText = this.iscsiService.targetAdvancedSettings; - _.forIn(this.target_default_controls, (_value, key) => { - const validators = []; - if (this.target_controls_limits && key in this.target_controls_limits) { - if ('min' in this.target_controls_limits[key]) { - validators.push(Validators.min(this.target_controls_limits[key]['min'])); - } - if ('max' in this.target_controls_limits[key]) { - validators.push(Validators.max(this.target_controls_limits[key]['max'])); - } - } - fg[key] = new FormControl(this.target_controls.value[key], { validators: validators }); + fg[key] = new FormControl(this.target_controls.value[key]); }); this.settingsForm = new CdFormGroup(fg); @@ -54,7 +42,7 @@ export class IscsiTargetIqnSettingsModalComponent implements OnInit { this.modalRef.hide(); } - isRadio(control) { - return ['Yes', 'No'].indexOf(this.target_default_controls[control]) !== -1; + getTargetControlLimits(setting) { + return this.target_controls_limits[setting]; } } diff --git a/src/pybind/mgr/dashboard/frontend/src/app/shared/api/iscsi.service.ts b/src/pybind/mgr/dashboard/frontend/src/app/shared/api/iscsi.service.ts index ff2794bf90f..6b31eaad23e 100644 --- a/src/pybind/mgr/dashboard/frontend/src/app/shared/api/iscsi.service.ts +++ b/src/pybind/mgr/dashboard/frontend/src/app/shared/api/iscsi.service.ts @@ -11,57 +11,6 @@ import { ApiModule } from './api.module'; export class IscsiService { constructor(private http: HttpClient) {} - targetAdvancedSettings = { - cmdsn_depth: { - help: '' - }, - dataout_timeout: { - help: '' - }, - first_burst_length: { - help: '' - }, - immediate_data: { - help: '' - }, - initial_r2t: { - help: '' - }, - max_burst_length: { - help: '' - }, - max_outstanding_r2t: { - help: '' - }, - max_recv_data_segment_length: { - help: '' - }, - max_xmit_data_segment_length: { - help: '' - }, - nopin_response_timeout: { - help: '' - }, - nopin_timeout: { - help: '' - } - }; - - imageAdvancedSettings = { - hw_max_sectors: { - help: '' - }, - max_data_area_mb: { - help: '' - }, - osd_op_timeout: { - help: '' - }, - qfull_timeout: { - help: '' - } - }; - listTargets() { return this.http.get(`api/iscsi/target`); }