Add user update for admins
This commit is contained in:
parent
980246ea8f
commit
8094a89802
|
@ -4,7 +4,7 @@ import { AdminComponent } from './admin.component'
|
||||||
import { AdminRoutingModule } from './admin-routing.module'
|
import { AdminRoutingModule } from './admin-routing.module'
|
||||||
import { FriendsComponent, FriendAddComponent, FriendListComponent, FriendService } from './friends'
|
import { FriendsComponent, FriendAddComponent, FriendListComponent, FriendService } from './friends'
|
||||||
import { RequestSchedulersComponent, RequestSchedulersStatsComponent, RequestSchedulersService } from './request-schedulers'
|
import { RequestSchedulersComponent, RequestSchedulersStatsComponent, RequestSchedulersService } from './request-schedulers'
|
||||||
import { UsersComponent, UserAddComponent, UserListComponent, UserService } from './users'
|
import { UsersComponent, UserAddComponent, UserUpdateComponent, UserListComponent, UserService } from './users'
|
||||||
import { VideoAbusesComponent, VideoAbuseListComponent } from './video-abuses'
|
import { VideoAbusesComponent, VideoAbuseListComponent } from './video-abuses'
|
||||||
import { SharedModule } from '../shared'
|
import { SharedModule } from '../shared'
|
||||||
|
|
||||||
|
@ -26,6 +26,7 @@ import { SharedModule } from '../shared'
|
||||||
|
|
||||||
UsersComponent,
|
UsersComponent,
|
||||||
UserAddComponent,
|
UserAddComponent,
|
||||||
|
UserUpdateComponent,
|
||||||
UserListComponent,
|
UserListComponent,
|
||||||
|
|
||||||
VideoAbusesComponent,
|
VideoAbusesComponent,
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
export * from './shared'
|
export * from './shared'
|
||||||
export * from './user-add'
|
export * from './user-edit'
|
||||||
export * from './user-list'
|
export * from './user-list'
|
||||||
export * from './users.component'
|
export * from './users.component'
|
||||||
export * from './users.routes'
|
export * from './users.routes'
|
||||||
|
|
|
@ -5,7 +5,7 @@ import 'rxjs/add/operator/map'
|
||||||
import { BytesPipe } from 'angular-pipes/src/math/bytes.pipe'
|
import { BytesPipe } from 'angular-pipes/src/math/bytes.pipe'
|
||||||
|
|
||||||
import { AuthHttp, RestExtractor, RestDataSource, User } from '../../../shared'
|
import { AuthHttp, RestExtractor, RestDataSource, User } from '../../../shared'
|
||||||
import { UserCreate } from '../../../../../../shared'
|
import { UserCreate, UserUpdate } from '../../../../../../shared'
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class UserService {
|
export class UserService {
|
||||||
|
@ -23,6 +23,18 @@ export class UserService {
|
||||||
.catch(this.restExtractor.handleError)
|
.catch(this.restExtractor.handleError)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
updateUser (userId: number, userUpdate: UserUpdate) {
|
||||||
|
return this.authHttp.put(UserService.BASE_USERS_URL + userId, userUpdate)
|
||||||
|
.map(this.restExtractor.extractDataBool)
|
||||||
|
.catch(this.restExtractor.handleError)
|
||||||
|
}
|
||||||
|
|
||||||
|
getUser (userId: number) {
|
||||||
|
return this.authHttp.get(UserService.BASE_USERS_URL + userId)
|
||||||
|
.map(this.restExtractor.extractDataGet)
|
||||||
|
.catch(this.restExtractor.handleError)
|
||||||
|
}
|
||||||
|
|
||||||
getDataSource () {
|
getDataSource () {
|
||||||
return new RestDataSource(this.authHttp, UserService.BASE_USERS_URL, this.formatDataSource.bind(this))
|
return new RestDataSource(this.authHttp, UserService.BASE_USERS_URL, this.formatDataSource.bind(this))
|
||||||
}
|
}
|
||||||
|
|
|
@ -1 +0,0 @@
|
||||||
export * from './user-add.component'
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
export * from './user-add.component'
|
||||||
|
export * from './user-update.component'
|
|
@ -6,20 +6,20 @@ import { NotificationsService } from 'angular2-notifications'
|
||||||
|
|
||||||
import { UserService } from '../shared'
|
import { UserService } from '../shared'
|
||||||
import {
|
import {
|
||||||
FormReactive,
|
|
||||||
USER_USERNAME,
|
USER_USERNAME,
|
||||||
USER_EMAIL,
|
USER_EMAIL,
|
||||||
USER_PASSWORD,
|
USER_PASSWORD,
|
||||||
USER_VIDEO_QUOTA
|
USER_VIDEO_QUOTA
|
||||||
} from '../../../shared'
|
} from '../../../shared'
|
||||||
import { UserCreate } from '../../../../../../shared'
|
import { UserCreate } from '../../../../../../shared'
|
||||||
|
import { UserEdit } from './user-edit'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'my-user-add',
|
selector: 'my-user-add',
|
||||||
templateUrl: './user-add.component.html'
|
templateUrl: './user-edit.component.html'
|
||||||
})
|
})
|
||||||
export class UserAddComponent extends FormReactive implements OnInit {
|
export class UserAddComponent extends UserEdit implements OnInit {
|
||||||
error: string = null
|
error: string
|
||||||
|
|
||||||
form: FormGroup
|
form: FormGroup
|
||||||
formErrors = {
|
formErrors = {
|
||||||
|
@ -59,8 +59,8 @@ export class UserAddComponent extends FormReactive implements OnInit {
|
||||||
this.buildForm()
|
this.buildForm()
|
||||||
}
|
}
|
||||||
|
|
||||||
addUser () {
|
formValidated () {
|
||||||
this.error = null
|
this.error = undefined
|
||||||
|
|
||||||
const userCreate: UserCreate = this.form.value
|
const userCreate: UserCreate = this.form.value
|
||||||
|
|
||||||
|
@ -76,4 +76,12 @@ export class UserAddComponent extends FormReactive implements OnInit {
|
||||||
err => this.error = err.text
|
err => this.error = err.text
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
isCreation () {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
getFormButtonTitle () {
|
||||||
|
return 'Add user'
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -1,12 +1,13 @@
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="content-padding">
|
<div class="content-padding">
|
||||||
|
|
||||||
<h3>Add user</h3>
|
<h3 *ngIf="isCreation() === true">Add user</h3>
|
||||||
|
<h3 *ngIf="isCreation() === false">Edit user {{ username }}</h3>
|
||||||
|
|
||||||
<div *ngIf="error" class="alert alert-danger">{{ error }}</div>
|
<div *ngIf="error" class="alert alert-danger">{{ error }}</div>
|
||||||
|
|
||||||
<form role="form" (ngSubmit)="addUser()" [formGroup]="form">
|
<form role="form" (ngSubmit)="formValidated()" [formGroup]="form">
|
||||||
<div class="form-group">
|
<div class="form-group" *ngIf="isCreation()">
|
||||||
<label for="username">Username</label>
|
<label for="username">Username</label>
|
||||||
<input
|
<input
|
||||||
type="text" class="form-control" id="username" placeholder="john"
|
type="text" class="form-control" id="username" placeholder="john"
|
||||||
|
@ -28,7 +29,7 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group" *ngIf="isCreation()">
|
||||||
<label for="password">Password</label>
|
<label for="password">Password</label>
|
||||||
<input
|
<input
|
||||||
type="password" class="form-control" id="password"
|
type="password" class="form-control" id="password"
|
||||||
|
@ -42,17 +43,13 @@
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="videoQuota">Video quota</label>
|
<label for="videoQuota">Video quota</label>
|
||||||
<select class="form-control" id="videoQuota" formControlName="videoQuota">
|
<select class="form-control" id="videoQuota" formControlName="videoQuota">
|
||||||
<option value="-1">Unlimited</option>
|
<option *ngFor="let videoQuotaOption of videoQuotaOptions" [value]="videoQuotaOption.value">
|
||||||
<option value="100000000">100MB</option>
|
{{ videoQuotaOption.label }}
|
||||||
<option value="500000000">500MB</option>
|
</option>
|
||||||
<option value="1000000000">1GB</option>
|
|
||||||
<option value="5000000000">5GB</option>
|
|
||||||
<option value="20000000000">20GB</option>
|
|
||||||
<option value="50000000000">50GB</option>
|
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<input type="submit" value="Add user" class="btn btn-default" [disabled]="!form.valid">
|
<input type="submit" value="{{ getFormButtonTitle() }}" class="btn btn-default" [disabled]="!form.valid">
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
|
@ -0,0 +1,16 @@
|
||||||
|
import { FormReactive } from '../../../shared'
|
||||||
|
|
||||||
|
export abstract class UserEdit extends FormReactive {
|
||||||
|
videoQuotaOptions = [
|
||||||
|
{ value: -1, label: 'Unlimited' },
|
||||||
|
{ value: 100 * 1024 * 1024, label: '100MB' },
|
||||||
|
{ value: 5 * 1024 * 1024, label: '500MB' },
|
||||||
|
{ value: 1024 * 1024 * 1024, label: '1GB' },
|
||||||
|
{ value: 5 * 1024 * 1024 * 1024, label: '5GB' },
|
||||||
|
{ value: 20 * 1024 * 1024 * 1024, label: '20GB' },
|
||||||
|
{ value: 50 * 1024 * 1024 * 1024, label: '50GB' }
|
||||||
|
]
|
||||||
|
|
||||||
|
abstract isCreation (): boolean
|
||||||
|
abstract getFormButtonTitle (): string
|
||||||
|
}
|
|
@ -0,0 +1,106 @@
|
||||||
|
import { Component, OnDestroy, OnInit } from '@angular/core'
|
||||||
|
import { FormBuilder, FormGroup } from '@angular/forms'
|
||||||
|
import { ActivatedRoute, Router } from '@angular/router'
|
||||||
|
import { Subscription } from 'rxjs/Subscription'
|
||||||
|
|
||||||
|
import { NotificationsService } from 'angular2-notifications'
|
||||||
|
|
||||||
|
import { UserService } from '../shared'
|
||||||
|
import { USER_EMAIL, USER_VIDEO_QUOTA } from '../../../shared'
|
||||||
|
import { UserUpdate } from '../../../../../../shared/models/users/user-update.model'
|
||||||
|
import { User } from '../../../shared/users/user.model'
|
||||||
|
import { UserEdit } from './user-edit'
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'my-user-update',
|
||||||
|
templateUrl: './user-edit.component.html'
|
||||||
|
})
|
||||||
|
export class UserUpdateComponent extends UserEdit implements OnInit, OnDestroy {
|
||||||
|
error: string
|
||||||
|
userId: number
|
||||||
|
username: string
|
||||||
|
|
||||||
|
form: FormGroup
|
||||||
|
formErrors = {
|
||||||
|
'email': '',
|
||||||
|
'videoQuota': ''
|
||||||
|
}
|
||||||
|
validationMessages = {
|
||||||
|
'email': USER_EMAIL.MESSAGES,
|
||||||
|
'videoQuota': USER_VIDEO_QUOTA.MESSAGES
|
||||||
|
}
|
||||||
|
|
||||||
|
private paramsSub: Subscription
|
||||||
|
|
||||||
|
constructor (
|
||||||
|
private formBuilder: FormBuilder,
|
||||||
|
private route: ActivatedRoute,
|
||||||
|
private router: Router,
|
||||||
|
private notificationsService: NotificationsService,
|
||||||
|
private userService: UserService
|
||||||
|
) {
|
||||||
|
super()
|
||||||
|
}
|
||||||
|
|
||||||
|
buildForm () {
|
||||||
|
this.form = this.formBuilder.group({
|
||||||
|
email: [ '', USER_EMAIL.VALIDATORS ],
|
||||||
|
videoQuota: [ '-1', USER_VIDEO_QUOTA.VALIDATORS ]
|
||||||
|
})
|
||||||
|
|
||||||
|
this.form.valueChanges.subscribe(data => this.onValueChanged(data))
|
||||||
|
}
|
||||||
|
|
||||||
|
ngOnInit () {
|
||||||
|
this.buildForm()
|
||||||
|
|
||||||
|
this.paramsSub = this.route.params.subscribe(routeParams => {
|
||||||
|
const userId = routeParams['id']
|
||||||
|
this.userService.getUser(userId).subscribe(
|
||||||
|
user => this.onUserFetched(user),
|
||||||
|
|
||||||
|
err => this.error = err.text
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
ngOnDestroy () {
|
||||||
|
this.paramsSub.unsubscribe()
|
||||||
|
}
|
||||||
|
|
||||||
|
formValidated () {
|
||||||
|
this.error = undefined
|
||||||
|
|
||||||
|
const userUpdate: UserUpdate = this.form.value
|
||||||
|
|
||||||
|
// A select in HTML is always mapped as a string, we convert it to number
|
||||||
|
userUpdate.videoQuota = parseInt(this.form.value['videoQuota'], 10)
|
||||||
|
|
||||||
|
this.userService.updateUser(this.userId, userUpdate).subscribe(
|
||||||
|
() => {
|
||||||
|
this.notificationsService.success('Success', `User ${this.username} updated.`)
|
||||||
|
this.router.navigate([ '/admin/users/list' ])
|
||||||
|
},
|
||||||
|
|
||||||
|
err => this.error = err.text
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
isCreation () {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
getFormButtonTitle () {
|
||||||
|
return 'Update user'
|
||||||
|
}
|
||||||
|
|
||||||
|
private onUserFetched (userJson: User) {
|
||||||
|
this.userId = userJson.id
|
||||||
|
this.username = userJson.username
|
||||||
|
|
||||||
|
this.form.patchValue({
|
||||||
|
email: userJson.email,
|
||||||
|
videoQuota: userJson.videoQuota
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,7 +5,7 @@
|
||||||
|
|
||||||
<ng2-smart-table
|
<ng2-smart-table
|
||||||
[settings]="tableSettings" [source]="usersSource"
|
[settings]="tableSettings" [source]="usersSource"
|
||||||
(delete)="removeUser($event)"
|
(delete)="removeUser($event)" (edit)="editUser($event)"
|
||||||
></ng2-smart-table>
|
></ng2-smart-table>
|
||||||
|
|
||||||
<a class="add-user btn btn-success pull-right" [routerLink]="['/admin/users/add']">
|
<a class="add-user btn btn-success pull-right" [routerLink]="['/admin/users/add']">
|
||||||
|
|
|
@ -5,6 +5,7 @@ import { NotificationsService } from 'angular2-notifications'
|
||||||
import { ConfirmService } from '../../../core'
|
import { ConfirmService } from '../../../core'
|
||||||
import { RestDataSource, User, Utils } from '../../../shared'
|
import { RestDataSource, User, Utils } from '../../../shared'
|
||||||
import { UserService } from '../shared'
|
import { UserService } from '../shared'
|
||||||
|
import { Router } from '@angular/router'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'my-user-list',
|
selector: 'my-user-list',
|
||||||
|
@ -22,15 +23,18 @@ export class UserListComponent {
|
||||||
actions: {
|
actions: {
|
||||||
position: 'right',
|
position: 'right',
|
||||||
add: false,
|
add: false,
|
||||||
edit: false,
|
edit: true,
|
||||||
delete: true
|
delete: true
|
||||||
},
|
},
|
||||||
delete: {
|
delete: {
|
||||||
deleteButtonContent: Utils.getRowDeleteButton()
|
deleteButtonContent: Utils.getRowDeleteButton()
|
||||||
},
|
},
|
||||||
|
edit: {
|
||||||
|
editButtonContent: Utils.getRowEditButton()
|
||||||
|
},
|
||||||
pager: {
|
pager: {
|
||||||
display: true,
|
display: true,
|
||||||
perPage: 1
|
perPage: 10
|
||||||
},
|
},
|
||||||
columns: {
|
columns: {
|
||||||
id: {
|
id: {
|
||||||
|
@ -58,6 +62,7 @@ export class UserListComponent {
|
||||||
}
|
}
|
||||||
|
|
||||||
constructor (
|
constructor (
|
||||||
|
private router: Router,
|
||||||
private notificationsService: NotificationsService,
|
private notificationsService: NotificationsService,
|
||||||
private confirmService: ConfirmService,
|
private confirmService: ConfirmService,
|
||||||
private userService: UserService
|
private userService: UserService
|
||||||
|
@ -65,8 +70,12 @@ export class UserListComponent {
|
||||||
this.usersSource = this.userService.getDataSource()
|
this.usersSource = this.userService.getDataSource()
|
||||||
}
|
}
|
||||||
|
|
||||||
removeUser ({ data }) {
|
editUser ({ data }: { data: User }) {
|
||||||
const user: User = data
|
this.router.navigate([ '/admin', 'users', data.id, 'update' ])
|
||||||
|
}
|
||||||
|
|
||||||
|
removeUser ({ data }: { data: User }) {
|
||||||
|
const user = data
|
||||||
|
|
||||||
if (user.username === 'root') {
|
if (user.username === 'root') {
|
||||||
this.notificationsService.error('Error', 'You cannot delete root.')
|
this.notificationsService.error('Error', 'You cannot delete root.')
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { Routes } from '@angular/router'
|
import { Routes } from '@angular/router'
|
||||||
|
|
||||||
import { UsersComponent } from './users.component'
|
import { UsersComponent } from './users.component'
|
||||||
import { UserAddComponent } from './user-add'
|
import { UserAddComponent, UserUpdateComponent } from './user-edit'
|
||||||
import { UserListComponent } from './user-list'
|
import { UserListComponent } from './user-list'
|
||||||
|
|
||||||
export const UsersRoutes: Routes = [
|
export const UsersRoutes: Routes = [
|
||||||
|
@ -31,6 +31,15 @@ export const UsersRoutes: Routes = [
|
||||||
title: 'Add a user'
|
title: 'Add a user'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: ':id/update',
|
||||||
|
component: UserUpdateComponent,
|
||||||
|
data: {
|
||||||
|
meta: {
|
||||||
|
title: 'Update a user'
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,6 @@ export class AccountChangePasswordComponent extends FormReactive implements OnIn
|
||||||
|
|
||||||
constructor (
|
constructor (
|
||||||
private formBuilder: FormBuilder,
|
private formBuilder: FormBuilder,
|
||||||
private router: Router,
|
|
||||||
private notificationsService: NotificationsService,
|
private notificationsService: NotificationsService,
|
||||||
private userService: UserService
|
private userService: UserService
|
||||||
) {
|
) {
|
||||||
|
|
|
@ -11,7 +11,7 @@ import {
|
||||||
UserService,
|
UserService,
|
||||||
USER_PASSWORD
|
USER_PASSWORD
|
||||||
} from '../../shared'
|
} from '../../shared'
|
||||||
import { UserUpdate } from '../../../../../shared'
|
import { UserUpdateMe } from '../../../../../shared'
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'my-account-details',
|
selector: 'my-account-details',
|
||||||
|
@ -30,7 +30,6 @@ export class AccountDetailsComponent extends FormReactive implements OnInit {
|
||||||
constructor (
|
constructor (
|
||||||
private authService: AuthService,
|
private authService: AuthService,
|
||||||
private formBuilder: FormBuilder,
|
private formBuilder: FormBuilder,
|
||||||
private router: Router,
|
|
||||||
private notificationsService: NotificationsService,
|
private notificationsService: NotificationsService,
|
||||||
private userService: UserService
|
private userService: UserService
|
||||||
) {
|
) {
|
||||||
|
@ -51,14 +50,14 @@ export class AccountDetailsComponent extends FormReactive implements OnInit {
|
||||||
|
|
||||||
updateDetails () {
|
updateDetails () {
|
||||||
const displayNSFW = this.form.value['displayNSFW']
|
const displayNSFW = this.form.value['displayNSFW']
|
||||||
const details: UserUpdate = {
|
const details: UserUpdateMe = {
|
||||||
displayNSFW
|
displayNSFW
|
||||||
}
|
}
|
||||||
|
|
||||||
this.error = null
|
this.error = null
|
||||||
this.userService.updateDetails(details).subscribe(
|
this.userService.updateMyDetails(details).subscribe(
|
||||||
() => {
|
() => {
|
||||||
this.notificationsService.success('Success', 'Informations updated.')
|
this.notificationsService.success('Success', 'Information updated.')
|
||||||
|
|
||||||
this.authService.refreshUserInformations()
|
this.authService.refreshUserInformations()
|
||||||
},
|
},
|
||||||
|
|
|
@ -6,7 +6,7 @@ import 'rxjs/add/operator/map'
|
||||||
import { AuthService } from '../../core'
|
import { AuthService } from '../../core'
|
||||||
import { AuthHttp } from '../auth'
|
import { AuthHttp } from '../auth'
|
||||||
import { RestExtractor } from '../rest'
|
import { RestExtractor } from '../rest'
|
||||||
import { UserCreate, UserUpdate } from '../../../../../shared'
|
import { UserCreate, UserUpdateMe } from '../../../../../shared'
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class UserService {
|
export class UserService {
|
||||||
|
@ -22,13 +22,13 @@ export class UserService {
|
||||||
checkTokenValidity () {
|
checkTokenValidity () {
|
||||||
const url = UserService.BASE_USERS_URL + 'me'
|
const url = UserService.BASE_USERS_URL + 'me'
|
||||||
|
|
||||||
// AuthHttp will redirect us to the login page if the oken is not valid anymore
|
// AuthHttp will redirect us to the login page if the token is not valid anymore
|
||||||
this.authHttp.get(url).subscribe()
|
this.authHttp.get(url).subscribe()
|
||||||
}
|
}
|
||||||
|
|
||||||
changePassword (newPassword: string) {
|
changePassword (newPassword: string) {
|
||||||
const url = UserService.BASE_USERS_URL + this.authService.getUser().id
|
const url = UserService.BASE_USERS_URL + 'me'
|
||||||
const body: UserUpdate = {
|
const body: UserUpdateMe = {
|
||||||
password: newPassword
|
password: newPassword
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -37,8 +37,8 @@ export class UserService {
|
||||||
.catch((res) => this.restExtractor.handleError(res))
|
.catch((res) => this.restExtractor.handleError(res))
|
||||||
}
|
}
|
||||||
|
|
||||||
updateDetails (details: UserUpdate) {
|
updateMyDetails (details: UserUpdateMe) {
|
||||||
const url = UserService.BASE_USERS_URL + this.authService.getUser().id
|
const url = UserService.BASE_USERS_URL + 'me'
|
||||||
|
|
||||||
return this.authHttp.put(url, details)
|
return this.authHttp.put(url, details)
|
||||||
.map(this.restExtractor.extractDataBool)
|
.map(this.restExtractor.extractDataBool)
|
||||||
|
|
|
@ -9,4 +9,8 @@ export class Utils {
|
||||||
static getRowDeleteButton () {
|
static getRowDeleteButton () {
|
||||||
return '<span class="glyphicon glyphicon-remove glyphicon-black"></span>'
|
return '<span class="glyphicon glyphicon-remove glyphicon-black"></span>'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static getRowEditButton () {
|
||||||
|
return '<span class="glyphicon glyphicon-pencil glyphicon-black"></span>'
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,15 +9,22 @@ import {
|
||||||
ensureUserRegistrationAllowed,
|
ensureUserRegistrationAllowed,
|
||||||
usersAddValidator,
|
usersAddValidator,
|
||||||
usersUpdateValidator,
|
usersUpdateValidator,
|
||||||
|
usersUpdateMeValidator,
|
||||||
usersRemoveValidator,
|
usersRemoveValidator,
|
||||||
usersVideoRatingValidator,
|
usersVideoRatingValidator,
|
||||||
|
usersGetValidator,
|
||||||
paginationValidator,
|
paginationValidator,
|
||||||
setPagination,
|
setPagination,
|
||||||
usersSortValidator,
|
usersSortValidator,
|
||||||
setUsersSort,
|
setUsersSort,
|
||||||
token
|
token
|
||||||
} from '../../middlewares'
|
} from '../../middlewares'
|
||||||
import { UserVideoRate as FormattedUserVideoRate, UserCreate, UserUpdate } from '../../../shared'
|
import {
|
||||||
|
UserVideoRate as FormattedUserVideoRate,
|
||||||
|
UserCreate,
|
||||||
|
UserUpdate,
|
||||||
|
UserUpdateMe
|
||||||
|
} from '../../../shared'
|
||||||
|
|
||||||
const usersRouter = express.Router()
|
const usersRouter = express.Router()
|
||||||
|
|
||||||
|
@ -40,6 +47,11 @@ usersRouter.get('/',
|
||||||
listUsers
|
listUsers
|
||||||
)
|
)
|
||||||
|
|
||||||
|
usersRouter.get('/:id',
|
||||||
|
usersGetValidator,
|
||||||
|
getUser
|
||||||
|
)
|
||||||
|
|
||||||
usersRouter.post('/',
|
usersRouter.post('/',
|
||||||
authenticate,
|
authenticate,
|
||||||
ensureIsAdmin,
|
ensureIsAdmin,
|
||||||
|
@ -53,8 +65,15 @@ usersRouter.post('/register',
|
||||||
createUser
|
createUser
|
||||||
)
|
)
|
||||||
|
|
||||||
|
usersRouter.put('/me',
|
||||||
|
authenticate,
|
||||||
|
usersUpdateMeValidator,
|
||||||
|
updateMe
|
||||||
|
)
|
||||||
|
|
||||||
usersRouter.put('/:id',
|
usersRouter.put('/:id',
|
||||||
authenticate,
|
authenticate,
|
||||||
|
ensureIsAdmin,
|
||||||
usersUpdateValidator,
|
usersUpdateValidator,
|
||||||
updateUser
|
updateUser
|
||||||
)
|
)
|
||||||
|
@ -105,6 +124,10 @@ function getUserInformation (req: express.Request, res: express.Response, next:
|
||||||
.catch(err => next(err))
|
.catch(err => next(err))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getUser (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
|
return res.json(res.locals.user.toFormattedJSON())
|
||||||
|
}
|
||||||
|
|
||||||
function getUserVideoRating (req: express.Request, res: express.Response, next: express.NextFunction) {
|
function getUserVideoRating (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
const videoId = +req.params.videoId
|
const videoId = +req.params.videoId
|
||||||
const userId = +res.locals.oauth.token.User.id
|
const userId = +res.locals.oauth.token.User.id
|
||||||
|
@ -139,14 +162,15 @@ function removeUser (req: express.Request, res: express.Response, next: express.
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
function updateUser (req: express.Request, res: express.Response, next: express.NextFunction) {
|
function updateMe (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
const body: UserUpdate = req.body
|
const body: UserUpdateMe = req.body
|
||||||
|
|
||||||
|
// FIXME: user is not already a Sequelize instance?
|
||||||
db.User.loadByUsername(res.locals.oauth.token.user.username)
|
db.User.loadByUsername(res.locals.oauth.token.user.username)
|
||||||
.then(user => {
|
.then(user => {
|
||||||
if (body.password) user.password = body.password
|
if (body.password !== undefined) user.password = body.password
|
||||||
|
if (body.email !== undefined) user.email = body.email
|
||||||
if (body.displayNSFW !== undefined) user.displayNSFW = body.displayNSFW
|
if (body.displayNSFW !== undefined) user.displayNSFW = body.displayNSFW
|
||||||
if (body.videoQuota !== undefined) user.videoQuota = body.videoQuota
|
|
||||||
|
|
||||||
return user.save()
|
return user.save()
|
||||||
})
|
})
|
||||||
|
@ -154,6 +178,18 @@ function updateUser (req: express.Request, res: express.Response, next: express.
|
||||||
.catch(err => next(err))
|
.catch(err => next(err))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function updateUser (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
|
const body: UserUpdate = req.body
|
||||||
|
const user = res.locals.user
|
||||||
|
|
||||||
|
if (body.email !== undefined) user.email = body.email
|
||||||
|
if (body.videoQuota !== undefined) user.videoQuota = body.videoQuota
|
||||||
|
|
||||||
|
return user.save()
|
||||||
|
.then(() => res.sendStatus(204))
|
||||||
|
.catch(err => next(err))
|
||||||
|
}
|
||||||
|
|
||||||
function success (req: express.Request, res: express.Response, next: express.NextFunction) {
|
function success (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
res.end()
|
res.end()
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,16 +53,35 @@ function usersRemoveValidator (req: express.Request, res: express.Response, next
|
||||||
|
|
||||||
function usersUpdateValidator (req: express.Request, res: express.Response, next: express.NextFunction) {
|
function usersUpdateValidator (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
req.checkParams('id', 'Should have a valid id').notEmpty().isInt()
|
req.checkParams('id', 'Should have a valid id').notEmpty().isInt()
|
||||||
// Add old password verification
|
req.checkBody('email', 'Should have a valid email attribute').optional().isEmail()
|
||||||
req.checkBody('password', 'Should have a valid password').optional().isUserPasswordValid()
|
|
||||||
req.checkBody('displayNSFW', 'Should have a valid display Not Safe For Work attribute').optional().isUserDisplayNSFWValid()
|
|
||||||
req.checkBody('videoQuota', 'Should have a valid user quota').optional().isUserVideoQuotaValid()
|
req.checkBody('videoQuota', 'Should have a valid user quota').optional().isUserVideoQuotaValid()
|
||||||
|
|
||||||
logger.debug('Checking usersUpdate parameters', { parameters: req.body })
|
logger.debug('Checking usersUpdate parameters', { parameters: req.body })
|
||||||
|
|
||||||
|
checkErrors(req, res, () => {
|
||||||
|
checkUserExists(req.params.id, res, next)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function usersUpdateMeValidator (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
|
// Add old password verification
|
||||||
|
req.checkBody('password', 'Should have a valid password').optional().isUserPasswordValid()
|
||||||
|
req.checkBody('email', 'Should have a valid email attribute').optional().isEmail()
|
||||||
|
req.checkBody('displayNSFW', 'Should have a valid display Not Safe For Work attribute').optional().isUserDisplayNSFWValid()
|
||||||
|
|
||||||
|
logger.debug('Checking usersUpdate parameters', { parameters: req.body })
|
||||||
|
|
||||||
checkErrors(req, res, next)
|
checkErrors(req, res, next)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function usersGetValidator (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
|
req.checkParams('id', 'Should have a valid id').notEmpty().isInt()
|
||||||
|
|
||||||
|
checkErrors(req, res, () => {
|
||||||
|
checkUserExists(req.params.id, res, next)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
function usersVideoRatingValidator (req: express.Request, res: express.Response, next: express.NextFunction) {
|
function usersVideoRatingValidator (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
req.checkParams('videoId', 'Should have a valid video id').notEmpty().isVideoIdOrUUIDValid()
|
req.checkParams('videoId', 'Should have a valid video id').notEmpty().isVideoIdOrUUIDValid()
|
||||||
|
|
||||||
|
@ -106,6 +125,24 @@ export {
|
||||||
usersAddValidator,
|
usersAddValidator,
|
||||||
usersRemoveValidator,
|
usersRemoveValidator,
|
||||||
usersUpdateValidator,
|
usersUpdateValidator,
|
||||||
|
usersUpdateMeValidator,
|
||||||
usersVideoRatingValidator,
|
usersVideoRatingValidator,
|
||||||
ensureUserRegistrationAllowed
|
ensureUserRegistrationAllowed,
|
||||||
|
usersGetValidator
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---------------------------------------------------------------------------
|
||||||
|
|
||||||
|
function checkUserExists (id: number, res: express.Response, callback: () => void) {
|
||||||
|
db.User.loadById(id)
|
||||||
|
.then(user => {
|
||||||
|
if (!user) return res.status(404).send('User not found')
|
||||||
|
|
||||||
|
res.locals.user = user
|
||||||
|
callback()
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
logger.error('Error in user request validator.', err)
|
||||||
|
return res.sendStatus(500)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
export * from './user.model'
|
export * from './user.model'
|
||||||
export * from './user-create.model'
|
export * from './user-create.model'
|
||||||
export * from './user-update.model'
|
export * from './user-update.model'
|
||||||
|
export * from './user-update-me.model'
|
||||||
export * from './user-role.type'
|
export * from './user-role.type'
|
||||||
|
|
|
@ -0,0 +1,5 @@
|
||||||
|
export interface UserUpdateMe {
|
||||||
|
displayNSFW?: boolean
|
||||||
|
email?: string
|
||||||
|
password?: string
|
||||||
|
}
|
|
@ -1,5 +1,4 @@
|
||||||
export interface UserUpdate {
|
export interface UserUpdate {
|
||||||
displayNSFW?: boolean
|
email?: string
|
||||||
password?: string
|
|
||||||
videoQuota?: number
|
videoQuota?: number
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue