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

Invite users to an org or space #3377

Merged
merged 58 commits into from
Feb 19, 2019
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
58 commits
Select commit Hold shift + click to select a range
8c86ef7
Merge remote-tracking branch 'origin/v2-master' into invite-users
nwmac Jan 25, 2019
366869b
WIP
nwmac Jan 25, 2019
04ad893
WIP - Add stepper with new step, routing and list button (tbd)
richard-cox Jan 25, 2019
b59bb53
WIP - Added stacked input & mostly fleshed out
richard-cox Jan 29, 2019
46ef39a
WIP - Flesh out step 1 with instructions, special case of space level
richard-cox Jan 29, 2019
28ea51c
Add email library
nwmac Jan 30, 2019
93cd3ed
v2/v3 api docs updates
richard-cox Jan 25, 2019
9c2b5dc
wire in perms, cleared many todos
richard-cox Jan 30, 2019
0bbc74b
Merge remote-tracking branch 'origin/invite-users' into invite-users-2
richard-cox Jan 31, 2019
d79cb13
WIP backend
nwmac Jan 31, 2019
292a79d
Merge remote-tracking branch 'origin/invite-users' into invite-users-2
richard-cox Jan 31, 2019
30b7396
Wire in request
richard-cox Jan 31, 2019
70f002f
Remove other steps, fix & tidy up
richard-cox Feb 1, 2019
66546a8
Fix dupe entries
richard-cox Feb 1, 2019
3602acf
Return + focus
richard-cox Feb 1, 2019
8a3d171
Tweaks
richard-cox Feb 4, 2019
1da601c
Tweaks
richard-cox Feb 4, 2019
a95f61e
Fix tests
richard-cox Feb 4, 2019
b9c1454
Merge remote-tracking branch 'origin/v2-master' into invite-users
richard-cox Feb 4, 2019
8585502
CC and lint fixes
richard-cox Feb 4, 2019
e9d53d9
Basic docs, test fixes
richard-cox Feb 4, 2019
4c909f7
MD lint fixes
richard-cox Feb 5, 2019
aece246
Only show invite configure button if cf admin, tweak message for non-…
richard-cox Feb 5, 2019
bc27492
Fix `other` suite e2e tests
richard-cox Feb 5, 2019
887aab0
Improve wait for users table
richard-cox Feb 5, 2019
42599f6
Fix backend test
richard-cox Feb 5, 2019
2944bc6
Fix unit test, allow more time for e2e loading users
richard-cox Feb 5, 2019
ad8eb90
Unit test tweak
richard-cox Feb 6, 2019
fa1855c
Doc fix, tweak for unit tests
richard-cox Feb 6, 2019
66219a7
To Revert - e2e debug
richard-cox Feb 6, 2019
adcc443
Revert "To Revert - e2e debug"
richard-cox Feb 7, 2019
8ceae43
Fix forever loading users list on systems with no smpt config
richard-cox Feb 7, 2019
0afc77e
CC Fixes
richard-cox Feb 7, 2019
501b77c
Add close option to config invite client snack bar
richard-cox Feb 7, 2019
e3fab3b
Add confirmation and error snackbar to disable user invite
richard-cox Feb 7, 2019
8551357
Check scopes and use token for creating user in CF
nwmac Feb 8, 2019
6cd885b
Merge remote-tracking branch 'origin/v2-master' into invite-users
nwmac Feb 12, 2019
93eead1
Update user invite to Echo V3 API
nwmac Feb 12, 2019
ca17c21
Unit test fix
nwmac Feb 12, 2019
086dcfe
Fix front-end unit test
nwmac Feb 12, 2019
9f72548
Add permission check
nwmac Feb 13, 2019
0b2ba30
Three changes
richard-cox Feb 12, 2019
cbf358f
Move invite users button to above user tables
richard-cox Feb 13, 2019
69c06a7
Fix doc readme cc warnings
richard-cox Feb 13, 2019
d442ee3
Minor tidy-ups
nwmac Feb 14, 2019
3fef4f6
Merge branch 'v2-master' into invite-users
KlapTrap Feb 15, 2019
40058aa
Fix imports
KlapTrap Feb 15, 2019
283b5a0
Update docs
KlapTrap Feb 15, 2019
159a5ef
Update the docs guide
KlapTrap Feb 18, 2019
db04bb3
Update invite users text
KlapTrap Feb 18, 2019
ecea0ad
Minor change to the invite table header
KlapTrap Feb 18, 2019
af6e698
Small text change
KlapTrap Feb 18, 2019
a55a88a
Minor text change
KlapTrap Feb 18, 2019
b108d71
Merge remote-tracking branch 'origin/v2-master' into invite-users
richard-cox Feb 18, 2019
73793bc
Merge fixes
richard-cox Feb 18, 2019
93d8719
Update fetch all users max pagination
richard-cox Feb 18, 2019
2090f70
CC Fixed and copy change
KlapTrap Feb 18, 2019
656e3ff
CC Fix
KlapTrap Feb 18, 2019
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
Prev Previous commit
Next Next commit
WIP - Add stepper with new step, routing and list button (tbd)
  • Loading branch information
richard-cox committed Jan 25, 2019
commit 04ad893fe813c7b076dc0f4ee96ce5a7e34d44a0
14 changes: 10 additions & 4 deletions src/frontend/app/features/cloud-foundry/cloud-foundry.module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import { RouterModule } from '@angular/router';
import { NgxChartsModule } from '@swimlane/ngx-charts';

import { CoreModule } from '../../core/core.module';
import { CustomImportModule } from '../../custom-import.module';
import {
CFEndpointsListConfigService,
} from '../../shared/components/list/list-types/cf-endpoints/cf-endpoints-list-config.service';
Expand Down Expand Up @@ -77,12 +78,19 @@ import {
import {
CloudFoundryOrganizationsComponent,
} from './tabs/cloud-foundry-organizations/cloud-foundry-organizations.component';
import { CloudFoundryRoutesComponent } from './tabs/cloud-foundry-routes/cloud-foundry-routes.component';
import {
CloudFoundrySecurityGroupsComponent,
} from './tabs/cloud-foundry-security-groups/cloud-foundry-security-groups.component';
import { CloudFoundryStacksComponent } from './tabs/cloud-foundry-stacks/cloud-foundry-stacks.component';
import { CloudFoundrySummaryTabComponent } from './tabs/cloud-foundry-summary-tab/cloud-foundry-summary-tab.component';
import { CloudFoundryUsersComponent } from './tabs/cloud-foundry-users/cloud-foundry-users.component';
import {
UserInviteConfigurationDialogComponent,
} from './user-invites/configuration-dialog/user-invite-configuration-dialog.component';
import { UserInviteService } from './user-invites/user-invite.service';
import { InviteUsersCreateComponent } from './users/invite-users/invite-users-create/invite-users-create.component';
import { InviteUsersComponent } from './users/invite-users/invite-users.component';
import { CfRolesService } from './users/manage-users/cf-roles.service';
import { UsersRolesConfirmComponent } from './users/manage-users/manage-users-confirm/manage-users-confirm.component';
import { UsersRolesModifyComponent } from './users/manage-users/manage-users-modify/manage-users-modify.component';
Expand All @@ -91,10 +99,6 @@ import {
} from './users/manage-users/manage-users-modify/space-roles-list-wrapper/space-roles-list-wrapper.component';
import { UsersRolesSelectComponent } from './users/manage-users/manage-users-select/manage-users-select.component';
import { UsersRolesComponent } from './users/manage-users/manage-users.component';
import { CustomImportModule } from '../../custom-import.module';
import { CloudFoundryRoutesComponent } from './tabs/cloud-foundry-routes/cloud-foundry-routes.component';
import { UserInviteConfigurationDialogComponent } from './user-invites/configuration-dialog/user-invite-configuration-dialog.component';
import { UserInviteService } from './user-invites/user-invite.service';

@NgModule({
imports: [
Expand Down Expand Up @@ -148,6 +152,8 @@ import { UserInviteService } from './user-invites/user-invite.service';
UsersRolesConfirmComponent,
CloudFoundryRoutesComponent,
UserInviteConfigurationDialogComponent,
InviteUsersComponent,
InviteUsersCreateComponent
],
providers: [
CFEndpointsListConfigService,
Expand Down
19 changes: 15 additions & 4 deletions src/frontend/app/features/cloud-foundry/cloud-foundry.routing.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,9 @@
import { NgModule } from '@angular/core';
import { RouterModule, Routes } from '@angular/router';

import { DynamicExtensionRoutes } from '../../core/extension/dynamic-extension-routes';
import { StratosActionType } from '../../core/extension/extension-service';
import { PageNotFoundComponentComponent } from '../../core/page-not-found-component/page-not-found-component.component';
import { AddOrganizationComponent } from './add-organization/add-organization.component';
import { AddSpaceComponent } from './add-space/add-space.component';
import { CliInfoCloudFoundryComponent } from './cli-info-cloud-foundry/cli-info-cloud-foundry.component';
Expand Down Expand Up @@ -59,17 +62,15 @@ import {
import {
CloudFoundryOrganizationsComponent,
} from './tabs/cloud-foundry-organizations/cloud-foundry-organizations.component';
import { CloudFoundryRoutesComponent } from './tabs/cloud-foundry-routes/cloud-foundry-routes.component';
import {
CloudFoundrySecurityGroupsComponent,
} from './tabs/cloud-foundry-security-groups/cloud-foundry-security-groups.component';
import { CloudFoundryStacksComponent } from './tabs/cloud-foundry-stacks/cloud-foundry-stacks.component';
import { CloudFoundrySummaryTabComponent } from './tabs/cloud-foundry-summary-tab/cloud-foundry-summary-tab.component';
import { CloudFoundryUsersComponent } from './tabs/cloud-foundry-users/cloud-foundry-users.component';
import { InviteUsersComponent } from './users/invite-users/invite-users.component';
import { UsersRolesComponent } from './users/manage-users/manage-users.component';
import { DynamicExtensionRoutes } from '../../core/extension/dynamic-extension-routes';
import { PageNotFoundComponentComponent } from '../../core/page-not-found-component/page-not-found-component.component';
import { StratosActionType } from '../../core/extension/extension-service';
import { CloudFoundryRoutesComponent } from './tabs/cloud-foundry-routes/cloud-foundry-routes.component';

/* tslint:enable:max-line-length */
const usersRoles = [
Expand All @@ -83,11 +84,21 @@ const usersRoles = [
component: UsersRolesComponent,
pathMatch: 'full'
},
{
path: 'organizations/:orgId/users/invite',
component: InviteUsersComponent,
pathMatch: 'full'
},
{
path: 'organizations/:orgId/spaces/:spaceId/users/manage',
component: UsersRolesComponent,
pathMatch: 'full'
},
{
path: 'organizations/:orgId/spaces/:spaceId/users/invite',
component: InviteUsersComponent,
pathMatch: 'full'
},
];

const cloudFoundry: Routes = [{
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
<p>
invite-users-create works!
</p>
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
import { async, ComponentFixture, TestBed } from '@angular/core/testing';

import { InviteUsersCreateComponent } from './invite-users-create.component';

describe('InviteUsersCreateComponent', () => {
let component: InviteUsersCreateComponent;
let fixture: ComponentFixture<InviteUsersCreateComponent>;

beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [ InviteUsersCreateComponent ]
})
.compileComponents();
}));

beforeEach(() => {
fixture = TestBed.createComponent(InviteUsersCreateComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});

it('should create', () => {
expect(component).toBeTruthy();
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
import { Component, OnDestroy, OnInit } from '@angular/core';
import { Observable, of as observableOf } from 'rxjs';

@Component({
selector: 'app-invite-users-create',
templateUrl: './invite-users-create.component.html',
styleUrls: ['./invite-users-create.component.scss']
})
export class InviteUsersCreateComponent implements OnInit, OnDestroy {

valid$: Observable<boolean> = observableOf(false);

constructor() { }

ngOnInit() {
}

ngOnDestroy() {
// if (this.orgGuidChangedSub) {
// this.orgGuidChangedSub.unsubscribe();
// }
// this.destroySpacesList();
// if (this.snackBarRef) {
// this.snackBarRef.dismiss();
// this.snackBarRef = null;
// }
}


onEnter = () => {
// if (!this.snackBarRef) {
// this.usersWithWarning$.pipe(first()).subscribe((usersWithWarning => {
// if (usersWithWarning && usersWithWarning.length) {
// this.snackBarRef = this.snackBar.open(`Not all roles are shown for user/s - ${usersWithWarning.join(', ')}. To avoid this please
// navigate to a specific organization or space`, 'Dismiss');
// }
// }));
// }
}

onLeave = (isNext: boolean) => {
// if (!isNext && this.snackBarRef) {
// this.snackBarRef.dismiss();
// this.snackBarRef = null;
// }
}

onNext = () => {
// this.store.dispatch(new UsersRolesSetUsers(this.cfUserService.activeRouteCfOrgSpace.cfGuid, [user.entity]));

return observableOf({ success: false });
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
<app-page-header>Invite Users</app-page-header>
<app-steppers cancel="{{defaultCancelUrl}}" [nextButtonProgress]="false">
<app-step title="Create Users" [onLeave]="create.onLeave" [onEnter]="create.onEnter" [onNext]="create.onNext" [valid]="create.valid$ | async">
<app-invite-users-create #create></app-invite-users-create>
</app-step>
<app-step title="Update Roles" [blocked]="modify.cfRolesService.loading$ | async" [onLeave]="modify.onLeave" [onEnter]="modify.onEnter" [onNext]="modify.onNext" [valid]="modify.valid$ | async">
<app-manage-users-modify #modify></app-manage-users-modify>
</app-step>
<app-step title="Confirm Roles" [onEnter]="confirm.onEnter" [canClose]="!applyStarted" [disablePrevious]="applyStarted" [destructiveStep]="!applyStarted" [onNext]="startApply" [finishButtonText]="applyStarted ? 'Close' : 'Apply'">
<app-manage-users-confirm #confirm></app-manage-users-confirm>
</app-step>
</app-steppers>
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
import { async, ComponentFixture, TestBed } from '@angular/core/testing';

import { InviteUsersComponent } from './invite-users.component';

describe('InviteUsersComponent', () => {
let component: InviteUsersComponent;
let fixture: ComponentFixture<InviteUsersComponent>;

beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [ InviteUsersComponent ]
})
.compileComponents();
}));

beforeEach(() => {
fixture = TestBed.createComponent(InviteUsersComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});

it('should create', () => {
expect(component).toBeTruthy();
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
import { Component, OnInit } from '@angular/core';
import { ActivatedRoute } from '@angular/router';
import { Store } from '@ngrx/store';
import { of as observableOf } from 'rxjs';

import { StepOnNextFunction } from '../../../../shared/components/stepper/step/step.component';
import { CfUserService } from '../../../../shared/data-services/cf-user.service';
import { UsersRolesExecuteChanges } from '../../../../store/actions/users-roles.actions';
import { AppState } from '../../../../store/app-state';
import { ActiveRouteCfOrgSpace } from '../../cf-page.types';

@Component({
selector: 'app-invite-users',
templateUrl: './invite-users.component.html',
styleUrls: ['./invite-users.component.scss']
})
export class InviteUsersComponent implements OnInit {

defaultCancelUrl: string;
applyStarted = true;

constructor(
private store: Store<AppState>,
private activeRouteCfOrgSpace: ActiveRouteCfOrgSpace,
private cfUserService: CfUserService,
private route: ActivatedRoute) {
this.defaultCancelUrl = this.createReturnUrl(activeRouteCfOrgSpace);

}

ngOnInit() {
}

createReturnUrl(activeRouteCfOrgSpace: ActiveRouteCfOrgSpace): string {
return !this.activeRouteCfOrgSpace.spaceGuid ? `/organizations/${activeRouteCfOrgSpace.orgGuid}/users` : `/spaces/${activeRouteCfOrgSpace.spaceGuid}/users`;
}

startApply: StepOnNextFunction = () => {
if (this.applyStarted) {
return observableOf({ success: true, redirect: true });
}
this.applyStarted = true;
this.store.dispatch(new UsersRolesExecuteChanges());
return observableOf({ success: true, ignoreSuccess: true });
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -31,5 +31,15 @@ export class CfOrgUsersListConfigService extends CfUserListConfigService {
(user: CfUser): boolean => cfUserService.hasRolesInOrg(user, activeRouteCfOrgSpace.orgGuid, false),
cfOrgService.org$
);

// TODO: RC Permissions (as per config shown in cf summary)
this.getGlobalActions = () => [{
action: () => {
router.navigate([this.createManagerUsersUrl(`/users/invite`)]);
},
icon: 'add', // TODO: RC
label: 'Invite', // TODO: RC
description: 'Invite users to this organization'
}];
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -104,15 +104,15 @@ export class CfUserListConfigService extends ListConfig<APIResource<CfUser>> {
description: `Manage roles`,
};

private createManagerUsersUrl(): string {
protected createManagerUsersUrl(stepperPath: string = `/users/manage`): string {
let route = `/cloud-foundry/${this.cfUserService.activeRouteCfOrgSpace.cfGuid}`;
if (this.activeRouteCfOrgSpace.orgGuid) {
route += `/organizations/${this.activeRouteCfOrgSpace.orgGuid}`;
if (this.activeRouteCfOrgSpace.spaceGuid) {
route += `/spaces/${this.activeRouteCfOrgSpace.spaceGuid}`;
}
}
route += `/users/manage`;
route += stepperPath;
return route;
}

Expand Down