Add ability to delete comments
This commit is contained in:
		
							parent
							
								
									cf117aaafc
								
							
						
					
					
						commit
						4cb6d45788
					
				
					 29 changed files with 437 additions and 52 deletions
				
			
		| 
						 | 
					@ -1,6 +1,6 @@
 | 
				
			||||||
import { Component, OnInit } from '@angular/core'
 | 
					import { Component, OnInit } from '@angular/core'
 | 
				
			||||||
import { Router } from '@angular/router'
 | 
					import { Router } from '@angular/router'
 | 
				
			||||||
import { AuthService, ServerService } from './core'
 | 
					import { AuthService, ServerService } from '@app/core'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@Component({
 | 
					@Component({
 | 
				
			||||||
  selector: 'my-app',
 | 
					  selector: 'my-app',
 | 
				
			||||||
| 
						 | 
					@ -50,10 +50,6 @@ export class AppComponent implements OnInit {
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  isInAdmin () {
 | 
					 | 
				
			||||||
    return this.router.url.indexOf('/admin/') !== -1
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  toggleMenu () {
 | 
					  toggleMenu () {
 | 
				
			||||||
    window.scrollTo(0, 0)
 | 
					    window.scrollTo(0, 0)
 | 
				
			||||||
    this.isMenuDisplayed = !this.isMenuDisplayed
 | 
					    this.isMenuDisplayed = !this.isMenuDisplayed
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,11 +1,11 @@
 | 
				
			||||||
import { Account as ServerAccount } from '../../../../../shared/models/actors/account.model'
 | 
					import { Account as ServerAccount } from '../../../../../shared/models/actors/account.model'
 | 
				
			||||||
import { Avatar } from '../../../../../shared/models/avatars/avatar.model'
 | 
					import { Avatar } from '../../../../../shared/models/avatars/avatar.model'
 | 
				
			||||||
import { environment } from '../../../environments/environment'
 | 
					 | 
				
			||||||
import { getAbsoluteAPIUrl } from '../misc/utils'
 | 
					import { getAbsoluteAPIUrl } from '../misc/utils'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class Account implements ServerAccount {
 | 
					export class Account implements ServerAccount {
 | 
				
			||||||
  id: number
 | 
					  id: number
 | 
				
			||||||
  uuid: string
 | 
					  uuid: string
 | 
				
			||||||
 | 
					  url: string
 | 
				
			||||||
  name: string
 | 
					  name: string
 | 
				
			||||||
  displayName: string
 | 
					  displayName: string
 | 
				
			||||||
  host: string
 | 
					  host: string
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -51,8 +51,6 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
.submit-container {
 | 
					.submit-container {
 | 
				
			||||||
  text-align: right;
 | 
					  text-align: right;
 | 
				
			||||||
  position: relative;
 | 
					 | 
				
			||||||
  bottom: $button-height;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
  .message-submit {
 | 
					  .message-submit {
 | 
				
			||||||
    display: inline-block;
 | 
					    display: inline-block;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,13 +3,14 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  <div class="comment">
 | 
					  <div class="comment">
 | 
				
			||||||
    <div class="comment-account-date">
 | 
					    <div class="comment-account-date">
 | 
				
			||||||
      <div class="comment-account">{{ comment.by }}</div>
 | 
					      <a target="_blank" [href]="comment.account.url" class="comment-account">{{ comment.by }}</a>
 | 
				
			||||||
      <div class="comment-date">{{ comment.createdAt | myFromNow }}</div>
 | 
					      <div class="comment-date">{{ comment.createdAt | myFromNow }}</div>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
    <div>{{ comment.text }}</div>
 | 
					    <div>{{ comment.text }}</div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <div class="comment-actions">
 | 
					    <div class="comment-actions">
 | 
				
			||||||
      <div *ngIf="isUserLoggedIn()" (click)="onWantToReply()" class="comment-action-reply">Reply</div>
 | 
					      <div *ngIf="isUserLoggedIn()" (click)="onWantToReply()" class="comment-action-reply">Reply</div>
 | 
				
			||||||
 | 
					      <div *ngIf="isRemovableByUser()" (click)="onWantToDelete()" class="comment-action-delete">Delete</div>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <my-video-comment-add
 | 
					    <my-video-comment-add
 | 
				
			||||||
| 
						 | 
					@ -28,7 +29,8 @@
 | 
				
			||||||
          [video]="video"
 | 
					          [video]="video"
 | 
				
			||||||
          [inReplyToCommentId]="inReplyToCommentId"
 | 
					          [inReplyToCommentId]="inReplyToCommentId"
 | 
				
			||||||
          [commentTree]="commentChild"
 | 
					          [commentTree]="commentChild"
 | 
				
			||||||
          (wantedToReply)="onWantedToReply($event)"
 | 
					          (wantedToReply)="onWantToReply($event)"
 | 
				
			||||||
 | 
					          (wantedToDelete)="onWantToDelete($event)"
 | 
				
			||||||
          (resetReply)="onResetReply()"
 | 
					          (resetReply)="onResetReply()"
 | 
				
			||||||
        ></my-video-comment>
 | 
					        ></my-video-comment>
 | 
				
			||||||
      </div>
 | 
					      </div>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -20,6 +20,9 @@
 | 
				
			||||||
      margin-bottom: 4px;
 | 
					      margin-bottom: 4px;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      .comment-account {
 | 
					      .comment-account {
 | 
				
			||||||
 | 
					        @include disable-default-a-behaviour;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        color: #000;
 | 
				
			||||||
        font-weight: $font-bold;
 | 
					        font-weight: $font-bold;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -31,10 +34,16 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    .comment-actions {
 | 
					    .comment-actions {
 | 
				
			||||||
      margin: 10px 0;
 | 
					      margin: 10px 0;
 | 
				
			||||||
 | 
					      display: flex;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      .comment-action-reply {
 | 
					      .comment-action-reply, .comment-action-delete {
 | 
				
			||||||
        color: #585858;
 | 
					        color: #585858;
 | 
				
			||||||
        cursor: pointer;
 | 
					        cursor: pointer;
 | 
				
			||||||
 | 
					        margin-right: 10px;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        &:hover {
 | 
				
			||||||
 | 
					          color: #000;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,5 +1,6 @@
 | 
				
			||||||
import { Component, EventEmitter, Input, Output } from '@angular/core'
 | 
					import { Component, EventEmitter, Input, Output } from '@angular/core'
 | 
				
			||||||
import { Account as AccountInterface } from '../../../../../../shared/models/actors'
 | 
					import { Account as AccountInterface } from '../../../../../../shared/models/actors'
 | 
				
			||||||
 | 
					import { UserRight } from '../../../../../../shared/models/users'
 | 
				
			||||||
import { VideoCommentThreadTree } from '../../../../../../shared/models/videos/video-comment.model'
 | 
					import { VideoCommentThreadTree } from '../../../../../../shared/models/videos/video-comment.model'
 | 
				
			||||||
import { AuthService } from '../../../core/auth'
 | 
					import { AuthService } from '../../../core/auth'
 | 
				
			||||||
import { Account } from '../../../shared/account/account.model'
 | 
					import { Account } from '../../../shared/account/account.model'
 | 
				
			||||||
| 
						 | 
					@ -17,7 +18,9 @@ export class VideoCommentComponent {
 | 
				
			||||||
  @Input() commentTree: VideoCommentThreadTree
 | 
					  @Input() commentTree: VideoCommentThreadTree
 | 
				
			||||||
  @Input() inReplyToCommentId: number
 | 
					  @Input() inReplyToCommentId: number
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  @Output() wantedToDelete = new EventEmitter<VideoComment>()
 | 
				
			||||||
  @Output() wantedToReply = new EventEmitter<VideoComment>()
 | 
					  @Output() wantedToReply = new EventEmitter<VideoComment>()
 | 
				
			||||||
 | 
					  @Output() threadCreated = new EventEmitter<VideoCommentThreadTree>()
 | 
				
			||||||
  @Output() resetReply = new EventEmitter()
 | 
					  @Output() resetReply = new EventEmitter()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  constructor (private authService: AuthService) {}
 | 
					  constructor (private authService: AuthService) {}
 | 
				
			||||||
| 
						 | 
					@ -32,6 +35,8 @@ export class VideoCommentComponent {
 | 
				
			||||||
        comment: this.comment,
 | 
					        comment: this.comment,
 | 
				
			||||||
        children: []
 | 
					        children: []
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      this.threadCreated.emit(this.commentTree)
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    this.commentTree.children.push({
 | 
					    this.commentTree.children.push({
 | 
				
			||||||
| 
						 | 
					@ -41,19 +46,18 @@ export class VideoCommentComponent {
 | 
				
			||||||
    this.resetReply.emit()
 | 
					    this.resetReply.emit()
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  onWantToReply () {
 | 
					  onWantToReply (comment?: VideoComment) {
 | 
				
			||||||
    this.wantedToReply.emit(this.comment)
 | 
					    this.wantedToReply.emit(comment || this.comment)
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  onWantToDelete (comment?: VideoComment) {
 | 
				
			||||||
 | 
					    this.wantedToDelete.emit(comment || this.comment)
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  isUserLoggedIn () {
 | 
					  isUserLoggedIn () {
 | 
				
			||||||
    return this.authService.isLoggedIn()
 | 
					    return this.authService.isLoggedIn()
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // Event from child comment
 | 
					 | 
				
			||||||
  onWantedToReply (comment: VideoComment) {
 | 
					 | 
				
			||||||
    this.wantedToReply.emit(comment)
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  onResetReply () {
 | 
					  onResetReply () {
 | 
				
			||||||
    this.resetReply.emit()
 | 
					    this.resetReply.emit()
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
| 
						 | 
					@ -61,4 +65,12 @@ export class VideoCommentComponent {
 | 
				
			||||||
  getAvatarUrl (account: AccountInterface) {
 | 
					  getAvatarUrl (account: AccountInterface) {
 | 
				
			||||||
    return Account.GET_ACCOUNT_AVATAR_URL(account)
 | 
					    return Account.GET_ACCOUNT_AVATAR_URL(account)
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  isRemovableByUser () {
 | 
				
			||||||
 | 
					    return this.isUserLoggedIn() &&
 | 
				
			||||||
 | 
					      (
 | 
				
			||||||
 | 
					        this.user.account.id === this.comment.account.id ||
 | 
				
			||||||
 | 
					        this.user.hasRight(UserRight.REMOVE_ANY_VIDEO_COMMENT)
 | 
				
			||||||
 | 
					      )
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -66,6 +66,15 @@ export class VideoCommentService {
 | 
				
			||||||
      .catch((res) => this.restExtractor.handleError(res))
 | 
					      .catch((res) => this.restExtractor.handleError(res))
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  deleteVideoComment (videoId: number | string, commentId: number) {
 | 
				
			||||||
 | 
					    const url = `${VideoCommentService.BASE_VIDEO_URL + videoId}/comments/${commentId}`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return this.authHttp
 | 
				
			||||||
 | 
					      .delete(url)
 | 
				
			||||||
 | 
					      .map(this.restExtractor.extractDataBool)
 | 
				
			||||||
 | 
					      .catch((res) => this.restExtractor.handleError(res))
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  private extractVideoComment (videoComment: VideoCommentServerModel) {
 | 
					  private extractVideoComment (videoComment: VideoCommentServerModel) {
 | 
				
			||||||
    return new VideoComment(videoComment)
 | 
					    return new VideoComment(videoComment)
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -27,6 +27,8 @@
 | 
				
			||||||
          [inReplyToCommentId]="inReplyToCommentId"
 | 
					          [inReplyToCommentId]="inReplyToCommentId"
 | 
				
			||||||
          [commentTree]="threadComments[comment.id]"
 | 
					          [commentTree]="threadComments[comment.id]"
 | 
				
			||||||
          (wantedToReply)="onWantedToReply($event)"
 | 
					          (wantedToReply)="onWantedToReply($event)"
 | 
				
			||||||
 | 
					          (wantedToDelete)="onWantedToDelete($event)"
 | 
				
			||||||
 | 
					          (threadCreated)="onThreadCreated($event)"
 | 
				
			||||||
          (resetReply)="onResetReply()"
 | 
					          (resetReply)="onResetReply()"
 | 
				
			||||||
        ></my-video-comment>
 | 
					        ></my-video-comment>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,6 +1,7 @@
 | 
				
			||||||
import { Component, Input, OnInit } from '@angular/core'
 | 
					import { Component, Input, OnInit } from '@angular/core'
 | 
				
			||||||
 | 
					import { ConfirmService } from '@app/core'
 | 
				
			||||||
import { NotificationsService } from 'angular2-notifications'
 | 
					import { NotificationsService } from 'angular2-notifications'
 | 
				
			||||||
import { VideoCommentThreadTree } from '../../../../../../shared/models/videos/video-comment.model'
 | 
					import { VideoComment as VideoCommentInterface, VideoCommentThreadTree } from '../../../../../../shared/models/videos/video-comment.model'
 | 
				
			||||||
import { AuthService } from '../../../core/auth'
 | 
					import { AuthService } from '../../../core/auth'
 | 
				
			||||||
import { ComponentPagination } from '../../../shared/rest/component-pagination.model'
 | 
					import { ComponentPagination } from '../../../shared/rest/component-pagination.model'
 | 
				
			||||||
import { User } from '../../../shared/users'
 | 
					import { User } from '../../../shared/users'
 | 
				
			||||||
| 
						 | 
					@ -32,6 +33,7 @@ export class VideoCommentsComponent implements OnInit {
 | 
				
			||||||
  constructor (
 | 
					  constructor (
 | 
				
			||||||
    private authService: AuthService,
 | 
					    private authService: AuthService,
 | 
				
			||||||
    private notificationsService: NotificationsService,
 | 
					    private notificationsService: NotificationsService,
 | 
				
			||||||
 | 
					    private confirmService: ConfirmService,
 | 
				
			||||||
    private videoCommentService: VideoCommentService
 | 
					    private videoCommentService: VideoCommentService
 | 
				
			||||||
  ) {}
 | 
					  ) {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -41,7 +43,7 @@ export class VideoCommentsComponent implements OnInit {
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  viewReplies (comment: VideoComment) {
 | 
					  viewReplies (comment: VideoCommentInterface) {
 | 
				
			||||||
    this.threadLoading[comment.id] = true
 | 
					    this.threadLoading[comment.id] = true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    this.videoCommentService.getVideoThreadComments(this.video.id, comment.id)
 | 
					    this.videoCommentService.getVideoThreadComments(this.video.id, comment.id)
 | 
				
			||||||
| 
						 | 
					@ -79,6 +81,44 @@ export class VideoCommentsComponent implements OnInit {
 | 
				
			||||||
    this.inReplyToCommentId = undefined
 | 
					    this.inReplyToCommentId = undefined
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  onThreadCreated (commentTree: VideoCommentThreadTree) {
 | 
				
			||||||
 | 
					    this.viewReplies(commentTree.comment)
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  onWantedToDelete (commentToDelete: VideoComment) {
 | 
				
			||||||
 | 
					    let message = 'Do you really want to delete this comment?'
 | 
				
			||||||
 | 
					    if (commentToDelete.totalReplies !== 0) message += `${commentToDelete.totalReplies} would be deleted too.`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.confirmService.confirm(message, 'Delete').subscribe(
 | 
				
			||||||
 | 
					      res => {
 | 
				
			||||||
 | 
					        if (res === false) return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        this.videoCommentService.deleteVideoComment(commentToDelete.videoId, commentToDelete.id)
 | 
				
			||||||
 | 
					          .subscribe(
 | 
				
			||||||
 | 
					            () => {
 | 
				
			||||||
 | 
					              // Delete the comment in the tree
 | 
				
			||||||
 | 
					              if (commentToDelete.inReplyToCommentId) {
 | 
				
			||||||
 | 
					                const thread = this.threadComments[commentToDelete.threadId]
 | 
				
			||||||
 | 
					                if (!thread) {
 | 
				
			||||||
 | 
					                  console.error(`Cannot find thread ${commentToDelete.threadId} of the comment to delete ${commentToDelete.id}`)
 | 
				
			||||||
 | 
					                  return
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                this.deleteLocalCommentThread(thread, commentToDelete)
 | 
				
			||||||
 | 
					                return
 | 
				
			||||||
 | 
					              }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					              // Delete the thread
 | 
				
			||||||
 | 
					              this.comments = this.comments.filter(c => c.id !== commentToDelete.id)
 | 
				
			||||||
 | 
					              this.componentPagination.totalItems--
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            err => this.notificationsService.error('Error', err.message)
 | 
				
			||||||
 | 
					          )
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  isUserLoggedIn () {
 | 
					  isUserLoggedIn () {
 | 
				
			||||||
    return this.authService.isLoggedIn()
 | 
					    return this.authService.isLoggedIn()
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
| 
						 | 
					@ -91,7 +131,7 @@ export class VideoCommentsComponent implements OnInit {
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  protected hasMoreComments () {
 | 
					  private hasMoreComments () {
 | 
				
			||||||
    // No results
 | 
					    // No results
 | 
				
			||||||
    if (this.componentPagination.totalItems === 0) return false
 | 
					    if (this.componentPagination.totalItems === 0) return false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -101,4 +141,15 @@ export class VideoCommentsComponent implements OnInit {
 | 
				
			||||||
    const maxPage = this.componentPagination.totalItems / this.componentPagination.itemsPerPage
 | 
					    const maxPage = this.componentPagination.totalItems / this.componentPagination.itemsPerPage
 | 
				
			||||||
    return maxPage > this.componentPagination.currentPage
 | 
					    return maxPage > this.componentPagination.currentPage
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  private deleteLocalCommentThread (parentComment: VideoCommentThreadTree, commentToDelete: VideoComment) {
 | 
				
			||||||
 | 
					    for (const commentChild of parentComment.children) {
 | 
				
			||||||
 | 
					      if (commentChild.comment.id === commentToDelete.id) {
 | 
				
			||||||
 | 
					        parentComment.children = parentComment.children.filter(c => c.comment.id !== commentToDelete.id)
 | 
				
			||||||
 | 
					        return
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      this.deleteLocalCommentThread(commentChild, commentToDelete)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -137,7 +137,7 @@ export class VideoWatchComponent implements OnInit, OnDestroy {
 | 
				
			||||||
  blacklistVideo (event: Event) {
 | 
					  blacklistVideo (event: Event) {
 | 
				
			||||||
    event.preventDefault()
 | 
					    event.preventDefault()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    this.confirmService.confirm('Do you really want to blacklist this video ?', 'Blacklist').subscribe(
 | 
					    this.confirmService.confirm('Do you really want to blacklist this video?', 'Blacklist').subscribe(
 | 
				
			||||||
      res => {
 | 
					      res => {
 | 
				
			||||||
        if (res === false) return
 | 
					        if (res === false) return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -14,6 +14,10 @@
 | 
				
			||||||
    "lib": [
 | 
					    "lib": [
 | 
				
			||||||
      "es2017",
 | 
					      "es2017",
 | 
				
			||||||
      "dom"
 | 
					      "dom"
 | 
				
			||||||
    ]
 | 
					    ],
 | 
				
			||||||
 | 
					    "baseUrl": "src",
 | 
				
			||||||
 | 
					    "paths": {
 | 
				
			||||||
 | 
					      "@app/*": [ "app/*" ]
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,14 +2,15 @@ import * as express from 'express'
 | 
				
			||||||
import { ResultList } from '../../../../shared/models'
 | 
					import { ResultList } from '../../../../shared/models'
 | 
				
			||||||
import { VideoCommentCreate } from '../../../../shared/models/videos/video-comment.model'
 | 
					import { VideoCommentCreate } from '../../../../shared/models/videos/video-comment.model'
 | 
				
			||||||
import { retryTransactionWrapper } from '../../../helpers/database-utils'
 | 
					import { retryTransactionWrapper } from '../../../helpers/database-utils'
 | 
				
			||||||
 | 
					import { logger } from '../../../helpers/logger'
 | 
				
			||||||
import { getFormattedObjects } from '../../../helpers/utils'
 | 
					import { getFormattedObjects } from '../../../helpers/utils'
 | 
				
			||||||
import { sequelizeTypescript } from '../../../initializers'
 | 
					import { sequelizeTypescript } from '../../../initializers'
 | 
				
			||||||
import { buildFormattedCommentTree, createVideoComment } from '../../../lib/video-comment'
 | 
					import { buildFormattedCommentTree, createVideoComment } from '../../../lib/video-comment'
 | 
				
			||||||
import { asyncMiddleware, authenticate, paginationValidator, setPagination, setVideoCommentThreadsSort } from '../../../middlewares'
 | 
					import { asyncMiddleware, authenticate, paginationValidator, setPagination, setVideoCommentThreadsSort } from '../../../middlewares'
 | 
				
			||||||
import { videoCommentThreadsSortValidator } from '../../../middlewares/validators'
 | 
					import { videoCommentThreadsSortValidator } from '../../../middlewares/validators'
 | 
				
			||||||
import {
 | 
					import {
 | 
				
			||||||
  addVideoCommentReplyValidator, addVideoCommentThreadValidator, listVideoCommentThreadsValidator,
 | 
					  addVideoCommentReplyValidator, addVideoCommentThreadValidator, listVideoCommentThreadsValidator, listVideoThreadCommentsValidator,
 | 
				
			||||||
  listVideoThreadCommentsValidator
 | 
					  removeVideoCommentValidator
 | 
				
			||||||
} from '../../../middlewares/validators/video-comments'
 | 
					} from '../../../middlewares/validators/video-comments'
 | 
				
			||||||
import { VideoModel } from '../../../models/video/video'
 | 
					import { VideoModel } from '../../../models/video/video'
 | 
				
			||||||
import { VideoCommentModel } from '../../../models/video/video-comment'
 | 
					import { VideoCommentModel } from '../../../models/video/video-comment'
 | 
				
			||||||
| 
						 | 
					@ -39,6 +40,11 @@ videoCommentRouter.post('/:videoId/comments/:commentId',
 | 
				
			||||||
  asyncMiddleware(addVideoCommentReplyValidator),
 | 
					  asyncMiddleware(addVideoCommentReplyValidator),
 | 
				
			||||||
  asyncMiddleware(addVideoCommentReplyRetryWrapper)
 | 
					  asyncMiddleware(addVideoCommentReplyRetryWrapper)
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					videoCommentRouter.delete('/:videoId/comments/:commentId',
 | 
				
			||||||
 | 
					  authenticate,
 | 
				
			||||||
 | 
					  asyncMiddleware(removeVideoCommentValidator),
 | 
				
			||||||
 | 
					  asyncMiddleware(removeVideoCommentRetryWrapper)
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ---------------------------------------------------------------------------
 | 
					// ---------------------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -131,3 +137,24 @@ function addVideoCommentReply (req: express.Request, res: express.Response, next
 | 
				
			||||||
    }, t)
 | 
					    }, t)
 | 
				
			||||||
  })
 | 
					  })
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function removeVideoCommentRetryWrapper (req: express.Request, res: express.Response, next: express.NextFunction) {
 | 
				
			||||||
 | 
					  const options = {
 | 
				
			||||||
 | 
					    arguments: [ req, res ],
 | 
				
			||||||
 | 
					    errorMessage: 'Cannot remove the video comment with many retries.'
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  await retryTransactionWrapper(removeVideoComment, options)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return res.type('json').status(204).end()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function removeVideoComment (req: express.Request, res: express.Response) {
 | 
				
			||||||
 | 
					  const videoCommentInstance: VideoCommentModel = res.locals.videoComment
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  await sequelizeTypescript.transaction(async t => {
 | 
				
			||||||
 | 
					    await videoCommentInstance.destroy({ transaction: t })
 | 
				
			||||||
 | 
					  })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  logger.info('Video comment %d deleted.', videoCommentInstance.id)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -5,7 +5,7 @@ import { isAnnounceActivityValid } from './announce'
 | 
				
			||||||
import { isActivityPubUrlValid } from './misc'
 | 
					import { isActivityPubUrlValid } from './misc'
 | 
				
			||||||
import { isDislikeActivityValid, isLikeActivityValid } from './rate'
 | 
					import { isDislikeActivityValid, isLikeActivityValid } from './rate'
 | 
				
			||||||
import { isUndoActivityValid } from './undo'
 | 
					import { isUndoActivityValid } from './undo'
 | 
				
			||||||
import { isVideoCommentCreateActivityValid } from './video-comments'
 | 
					import { isVideoCommentCreateActivityValid, isVideoCommentDeleteActivityValid } from './video-comments'
 | 
				
			||||||
import {
 | 
					import {
 | 
				
			||||||
  isVideoFlagValid,
 | 
					  isVideoFlagValid,
 | 
				
			||||||
  isVideoTorrentCreateActivityValid,
 | 
					  isVideoTorrentCreateActivityValid,
 | 
				
			||||||
| 
						 | 
					@ -70,7 +70,8 @@ function checkUpdateActivity (activity: any) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function checkDeleteActivity (activity: any) {
 | 
					function checkDeleteActivity (activity: any) {
 | 
				
			||||||
  return isVideoTorrentDeleteActivityValid(activity) ||
 | 
					  return isVideoTorrentDeleteActivityValid(activity) ||
 | 
				
			||||||
    isActorDeleteActivityValid(activity)
 | 
					    isActorDeleteActivityValid(activity) ||
 | 
				
			||||||
 | 
					    isVideoCommentDeleteActivityValid(activity)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function checkFollowActivity (activity: any) {
 | 
					function checkFollowActivity (activity: any) {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -18,10 +18,15 @@ function isVideoCommentObjectValid (comment: any) {
 | 
				
			||||||
    isActivityPubUrlValid(comment.url)
 | 
					    isActivityPubUrlValid(comment.url)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function isVideoCommentDeleteActivityValid (activity: any) {
 | 
				
			||||||
 | 
					  return isBaseActivityValid(activity, 'Delete')
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ---------------------------------------------------------------------------
 | 
					// ---------------------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export {
 | 
					export {
 | 
				
			||||||
  isVideoCommentCreateActivityValid
 | 
					  isVideoCommentCreateActivityValid,
 | 
				
			||||||
 | 
					  isVideoCommentDeleteActivityValid
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ---------------------------------------------------------------------------
 | 
					// ---------------------------------------------------------------------------
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -6,6 +6,7 @@ import { AccountModel } from '../../../models/account/account'
 | 
				
			||||||
import { ActorModel } from '../../../models/activitypub/actor'
 | 
					import { ActorModel } from '../../../models/activitypub/actor'
 | 
				
			||||||
import { VideoModel } from '../../../models/video/video'
 | 
					import { VideoModel } from '../../../models/video/video'
 | 
				
			||||||
import { VideoChannelModel } from '../../../models/video/video-channel'
 | 
					import { VideoChannelModel } from '../../../models/video/video-channel'
 | 
				
			||||||
 | 
					import { VideoCommentModel } from '../../../models/video/video-comment'
 | 
				
			||||||
import { getOrCreateActorAndServerAndModel } from '../actor'
 | 
					import { getOrCreateActorAndServerAndModel } from '../actor'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function processDeleteActivity (activity: ActivityDelete) {
 | 
					async function processDeleteActivity (activity: ActivityDelete) {
 | 
				
			||||||
| 
						 | 
					@ -24,9 +25,16 @@ async function processDeleteActivity (activity: ActivityDelete) {
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  {
 | 
					  {
 | 
				
			||||||
    let videoObject = await VideoModel.loadByUrlAndPopulateAccount(activity.id)
 | 
					    const videoCommentInstance = await VideoCommentModel.loadByUrlAndPopulateAccount(activity.id)
 | 
				
			||||||
    if (videoObject !== undefined) {
 | 
					    if (videoCommentInstance) {
 | 
				
			||||||
      return processDeleteVideo(actor, videoObject)
 | 
					      return processDeleteVideoComment(actor, videoCommentInstance)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  {
 | 
				
			||||||
 | 
					    const videoInstance = await VideoModel.loadByUrlAndPopulateAccount(activity.id)
 | 
				
			||||||
 | 
					    if (videoInstance) {
 | 
				
			||||||
 | 
					      return processDeleteVideo(actor, videoInstance)
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -101,3 +109,22 @@ async function deleteRemoteVideoChannel (videoChannelToRemove: VideoChannelModel
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  logger.info('Remote video channel with uuid %s removed.', videoChannelToRemove.Actor.uuid)
 | 
					  logger.info('Remote video channel with uuid %s removed.', videoChannelToRemove.Actor.uuid)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function processDeleteVideoComment (actor: ActorModel, videoComment: VideoCommentModel) {
 | 
				
			||||||
 | 
					  const options = {
 | 
				
			||||||
 | 
					    arguments: [ actor, videoComment ],
 | 
				
			||||||
 | 
					    errorMessage: 'Cannot remove the remote video comment with many retries.'
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  await retryTransactionWrapper(deleteRemoteVideoComment, options)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function deleteRemoteVideoComment (actor: ActorModel, videoComment: VideoCommentModel) {
 | 
				
			||||||
 | 
					  logger.debug('Removing remote video comment "%s".', videoComment.url)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return sequelizeTypescript.transaction(async t => {
 | 
				
			||||||
 | 
					    await videoComment.destroy({ transaction: t })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    logger.info('Remote video comment %s removed.', videoComment.url)
 | 
				
			||||||
 | 
					  })
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,6 +2,7 @@ import { Transaction } from 'sequelize'
 | 
				
			||||||
import { ActivityDelete } from '../../../../shared/models/activitypub'
 | 
					import { ActivityDelete } from '../../../../shared/models/activitypub'
 | 
				
			||||||
import { ActorModel } from '../../../models/activitypub/actor'
 | 
					import { ActorModel } from '../../../models/activitypub/actor'
 | 
				
			||||||
import { VideoModel } from '../../../models/video/video'
 | 
					import { VideoModel } from '../../../models/video/video'
 | 
				
			||||||
 | 
					import { VideoCommentModel } from '../../../models/video/video-comment'
 | 
				
			||||||
import { VideoShareModel } from '../../../models/video/video-share'
 | 
					import { VideoShareModel } from '../../../models/video/video-share'
 | 
				
			||||||
import { broadcastToFollowers } from './misc'
 | 
					import { broadcastToFollowers } from './misc'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -22,11 +23,24 @@ async function sendDeleteActor (byActor: ActorModel, t: Transaction) {
 | 
				
			||||||
  return broadcastToFollowers(data, byActor, [ byActor ], t)
 | 
					  return broadcastToFollowers(data, byActor, [ byActor ], t)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					async function sendDeleteVideoComment (videoComment: VideoCommentModel, t: Transaction) {
 | 
				
			||||||
 | 
					  const byActor = videoComment.Account.Actor
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const data = deleteActivityData(videoComment.url, byActor)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const actorsInvolved = await VideoShareModel.loadActorsByShare(videoComment.Video.id, t)
 | 
				
			||||||
 | 
					  actorsInvolved.push(videoComment.Video.VideoChannel.Account.Actor)
 | 
				
			||||||
 | 
					  actorsInvolved.push(byActor)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return broadcastToFollowers(data, byActor, actorsInvolved, t)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ---------------------------------------------------------------------------
 | 
					// ---------------------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export {
 | 
					export {
 | 
				
			||||||
  sendDeleteVideo,
 | 
					  sendDeleteVideo,
 | 
				
			||||||
  sendDeleteActor
 | 
					  sendDeleteActor,
 | 
				
			||||||
 | 
					  sendDeleteVideoComment
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ---------------------------------------------------------------------------
 | 
					// ---------------------------------------------------------------------------
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,9 +1,11 @@
 | 
				
			||||||
import * as express from 'express'
 | 
					import * as express from 'express'
 | 
				
			||||||
import { body, param } from 'express-validator/check'
 | 
					import { body, param } from 'express-validator/check'
 | 
				
			||||||
 | 
					import { UserRight } from '../../../shared'
 | 
				
			||||||
import { isIdOrUUIDValid, isIdValid } from '../../helpers/custom-validators/misc'
 | 
					import { isIdOrUUIDValid, isIdValid } from '../../helpers/custom-validators/misc'
 | 
				
			||||||
import { isValidVideoCommentText } from '../../helpers/custom-validators/video-comments'
 | 
					import { isValidVideoCommentText } from '../../helpers/custom-validators/video-comments'
 | 
				
			||||||
import { isVideoExist } from '../../helpers/custom-validators/videos'
 | 
					import { isVideoExist } from '../../helpers/custom-validators/videos'
 | 
				
			||||||
import { logger } from '../../helpers/logger'
 | 
					import { logger } from '../../helpers/logger'
 | 
				
			||||||
 | 
					import { UserModel } from '../../models/account/user'
 | 
				
			||||||
import { VideoModel } from '../../models/video/video'
 | 
					import { VideoModel } from '../../models/video/video'
 | 
				
			||||||
import { VideoCommentModel } from '../../models/video/video-comment'
 | 
					import { VideoCommentModel } from '../../models/video/video-comment'
 | 
				
			||||||
import { areValidationErrors } from './utils'
 | 
					import { areValidationErrors } from './utils'
 | 
				
			||||||
| 
						 | 
					@ -83,6 +85,24 @@ const videoCommentGetValidator = [
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
]
 | 
					]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const removeVideoCommentValidator = [
 | 
				
			||||||
 | 
					  param('videoId').custom(isIdOrUUIDValid).not().isEmpty().withMessage('Should have a valid videoId'),
 | 
				
			||||||
 | 
					  param('commentId').custom(isIdValid).not().isEmpty().withMessage('Should have a valid commentId'),
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  async (req: express.Request, res: express.Response, next: express.NextFunction) => {
 | 
				
			||||||
 | 
					    logger.debug('Checking removeVideoCommentValidator parameters.', { parameters: req.params })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (areValidationErrors(req, res)) return
 | 
				
			||||||
 | 
					    if (!await isVideoExist(req.params.videoId, res)) return
 | 
				
			||||||
 | 
					    if (!await isVideoCommentExist(req.params.commentId, res.locals.video, res)) return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Check if the user who did the request is able to delete the video
 | 
				
			||||||
 | 
					    if (!checkUserCanDeleteVideoComment(res.locals.oauth.token.User, res.locals.videoComment, res)) return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return next()
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ---------------------------------------------------------------------------
 | 
					// ---------------------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export {
 | 
					export {
 | 
				
			||||||
| 
						 | 
					@ -90,7 +110,8 @@ export {
 | 
				
			||||||
  listVideoThreadCommentsValidator,
 | 
					  listVideoThreadCommentsValidator,
 | 
				
			||||||
  addVideoCommentThreadValidator,
 | 
					  addVideoCommentThreadValidator,
 | 
				
			||||||
  addVideoCommentReplyValidator,
 | 
					  addVideoCommentReplyValidator,
 | 
				
			||||||
  videoCommentGetValidator
 | 
					  videoCommentGetValidator,
 | 
				
			||||||
 | 
					  removeVideoCommentValidator
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ---------------------------------------------------------------------------
 | 
					// ---------------------------------------------------------------------------
 | 
				
			||||||
| 
						 | 
					@ -160,3 +181,15 @@ function isVideoCommentsEnabled (video: VideoModel, res: express.Response) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  return true
 | 
					  return true
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function checkUserCanDeleteVideoComment (user: UserModel, videoComment: VideoCommentModel, res: express.Response) {
 | 
				
			||||||
 | 
					  const account = videoComment.Account
 | 
				
			||||||
 | 
					  if (user.hasRight(UserRight.REMOVE_ANY_VIDEO_COMMENT) === false && account.userId !== user.id) {
 | 
				
			||||||
 | 
					    res.status(403)
 | 
				
			||||||
 | 
					      .json({ error: 'Cannot remove video comment of another user' })
 | 
				
			||||||
 | 
					      .end()
 | 
				
			||||||
 | 
					    return false
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return true
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -253,7 +253,7 @@ function checkUserCanDeleteVideo (user: UserModel, video: VideoModel, res: expre
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // Check if the user can delete the video
 | 
					  // Check if the user can delete the video
 | 
				
			||||||
  // The user can delete it if s/he is an admin
 | 
					  // The user can delete it if he has the right
 | 
				
			||||||
  // Or if s/he is the video's account
 | 
					  // Or if s/he is the video's account
 | 
				
			||||||
  const account = video.VideoChannel.Account
 | 
					  const account = video.VideoChannel.Account
 | 
				
			||||||
  if (user.hasRight(UserRight.REMOVE_ANY_VIDEO) === false && account.userId !== user.id) {
 | 
					  if (user.hasRight(UserRight.REMOVE_ANY_VIDEO) === false && account.userId !== user.id) {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -271,6 +271,7 @@ export class ActorModel extends Model<ActorModel> {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return {
 | 
					    return {
 | 
				
			||||||
      id: this.id,
 | 
					      id: this.id,
 | 
				
			||||||
 | 
					      url: this.url,
 | 
				
			||||||
      uuid: this.uuid,
 | 
					      uuid: this.uuid,
 | 
				
			||||||
      host: this.getHost(),
 | 
					      host: this.getHost(),
 | 
				
			||||||
      score,
 | 
					      score,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -7,12 +7,14 @@ import { VideoCommentObject } from '../../../shared/models/activitypub/objects/v
 | 
				
			||||||
import { VideoComment } from '../../../shared/models/videos/video-comment.model'
 | 
					import { VideoComment } from '../../../shared/models/videos/video-comment.model'
 | 
				
			||||||
import { isActivityPubUrlValid } from '../../helpers/custom-validators/activitypub/misc'
 | 
					import { isActivityPubUrlValid } from '../../helpers/custom-validators/activitypub/misc'
 | 
				
			||||||
import { CONSTRAINTS_FIELDS } from '../../initializers'
 | 
					import { CONSTRAINTS_FIELDS } from '../../initializers'
 | 
				
			||||||
 | 
					import { sendDeleteVideoComment } from '../../lib/activitypub/send'
 | 
				
			||||||
import { AccountModel } from '../account/account'
 | 
					import { AccountModel } from '../account/account'
 | 
				
			||||||
import { ActorModel } from '../activitypub/actor'
 | 
					import { ActorModel } from '../activitypub/actor'
 | 
				
			||||||
import { AvatarModel } from '../avatar/avatar'
 | 
					import { AvatarModel } from '../avatar/avatar'
 | 
				
			||||||
import { ServerModel } from '../server/server'
 | 
					import { ServerModel } from '../server/server'
 | 
				
			||||||
import { getSort, throwIfNotValid } from '../utils'
 | 
					import { getSort, throwIfNotValid } from '../utils'
 | 
				
			||||||
import { VideoModel } from './video'
 | 
					import { VideoModel } from './video'
 | 
				
			||||||
 | 
					import { VideoChannelModel } from './video-channel'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
enum ScopeNames {
 | 
					enum ScopeNames {
 | 
				
			||||||
  WITH_ACCOUNT = 'WITH_ACCOUNT',
 | 
					  WITH_ACCOUNT = 'WITH_ACCOUNT',
 | 
				
			||||||
| 
						 | 
					@ -70,7 +72,25 @@ enum ScopeNames {
 | 
				
			||||||
    include: [
 | 
					    include: [
 | 
				
			||||||
      {
 | 
					      {
 | 
				
			||||||
        model: () => VideoModel,
 | 
					        model: () => VideoModel,
 | 
				
			||||||
        required: false
 | 
					        required: true,
 | 
				
			||||||
 | 
					        include: [
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            model: () => VideoChannelModel.unscoped(),
 | 
				
			||||||
 | 
					            required: true,
 | 
				
			||||||
 | 
					            include: [
 | 
				
			||||||
 | 
					              {
 | 
				
			||||||
 | 
					                model: () => AccountModel,
 | 
				
			||||||
 | 
					                required: true,
 | 
				
			||||||
 | 
					                include: [
 | 
				
			||||||
 | 
					                  {
 | 
				
			||||||
 | 
					                    model: () => ActorModel,
 | 
				
			||||||
 | 
					                    required: true
 | 
				
			||||||
 | 
					                  }
 | 
				
			||||||
 | 
					                ]
 | 
				
			||||||
 | 
					              }
 | 
				
			||||||
 | 
					            ]
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        ]
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    ]
 | 
					    ]
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
| 
						 | 
					@ -155,9 +175,10 @@ export class VideoCommentModel extends Model<VideoCommentModel> {
 | 
				
			||||||
  Account: AccountModel
 | 
					  Account: AccountModel
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @AfterDestroy
 | 
					  @AfterDestroy
 | 
				
			||||||
  static sendDeleteIfOwned (instance: VideoCommentModel) {
 | 
					  static async sendDeleteIfOwned (instance: VideoCommentModel) {
 | 
				
			||||||
    // TODO
 | 
					    if (instance.isOwned()) {
 | 
				
			||||||
    return undefined
 | 
					      await sendDeleteVideoComment(instance, undefined)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static loadById (id: number, t?: Sequelize.Transaction) {
 | 
					  static loadById (id: number, t?: Sequelize.Transaction) {
 | 
				
			||||||
| 
						 | 
					@ -198,6 +219,18 @@ export class VideoCommentModel extends Model<VideoCommentModel> {
 | 
				
			||||||
    return VideoCommentModel.findOne(query)
 | 
					    return VideoCommentModel.findOne(query)
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  static loadByUrlAndPopulateAccount (url: string, t?: Sequelize.Transaction) {
 | 
				
			||||||
 | 
					    const query: IFindOptions<VideoCommentModel> = {
 | 
				
			||||||
 | 
					      where: {
 | 
				
			||||||
 | 
					        url
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (t !== undefined) query.transaction = t
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return VideoCommentModel.scope([ ScopeNames.WITH_ACCOUNT ]).findOne(query)
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static listThreadsForApi (videoId: number, start: number, count: number, sort: string) {
 | 
					  static listThreadsForApi (videoId: number, start: number, count: number, sort: string) {
 | 
				
			||||||
    const query = {
 | 
					    const query = {
 | 
				
			||||||
      offset: start,
 | 
					      offset: start,
 | 
				
			||||||
| 
						 | 
					@ -237,6 +270,10 @@ export class VideoCommentModel extends Model<VideoCommentModel> {
 | 
				
			||||||
      })
 | 
					      })
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  isOwned () {
 | 
				
			||||||
 | 
					    return this.Account.isOwned()
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  toFormattedJSON () {
 | 
					  toFormattedJSON () {
 | 
				
			||||||
    return {
 | 
					    return {
 | 
				
			||||||
      id: this.id,
 | 
					      id: this.id,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -43,7 +43,8 @@ import { VideoTagModel } from './video-tag'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
enum ScopeNames {
 | 
					enum ScopeNames {
 | 
				
			||||||
  AVAILABLE_FOR_LIST = 'AVAILABLE_FOR_LIST',
 | 
					  AVAILABLE_FOR_LIST = 'AVAILABLE_FOR_LIST',
 | 
				
			||||||
  WITH_ACCOUNT = 'WITH_ACCOUNT',
 | 
					  WITH_ACCOUNT_API = 'WITH_ACCOUNT_API',
 | 
				
			||||||
 | 
					  WITH_ACCOUNT_DETAILS = 'WITH_ACCOUNT_DETAILS',
 | 
				
			||||||
  WITH_TAGS = 'WITH_TAGS',
 | 
					  WITH_TAGS = 'WITH_TAGS',
 | 
				
			||||||
  WITH_FILES = 'WITH_FILES',
 | 
					  WITH_FILES = 'WITH_FILES',
 | 
				
			||||||
  WITH_SHARES = 'WITH_SHARES',
 | 
					  WITH_SHARES = 'WITH_SHARES',
 | 
				
			||||||
| 
						 | 
					@ -62,7 +63,35 @@ enum ScopeNames {
 | 
				
			||||||
      privacy: VideoPrivacy.PUBLIC
 | 
					      privacy: VideoPrivacy.PUBLIC
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  [ScopeNames.WITH_ACCOUNT]: {
 | 
					  [ScopeNames.WITH_ACCOUNT_API]: {
 | 
				
			||||||
 | 
					    include: [
 | 
				
			||||||
 | 
					      {
 | 
				
			||||||
 | 
					        model: () => VideoChannelModel.unscoped(),
 | 
				
			||||||
 | 
					        required: true,
 | 
				
			||||||
 | 
					        include: [
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            attributes: [ 'name' ],
 | 
				
			||||||
 | 
					            model: () => AccountModel.unscoped(),
 | 
				
			||||||
 | 
					            required: true,
 | 
				
			||||||
 | 
					            include: [
 | 
				
			||||||
 | 
					              {
 | 
				
			||||||
 | 
					                attributes: [ 'serverId' ],
 | 
				
			||||||
 | 
					                model: () => ActorModel.unscoped(),
 | 
				
			||||||
 | 
					                required: true,
 | 
				
			||||||
 | 
					                include: [
 | 
				
			||||||
 | 
					                  {
 | 
				
			||||||
 | 
					                    model: () => ServerModel.unscoped(),
 | 
				
			||||||
 | 
					                    required: false
 | 
				
			||||||
 | 
					                  }
 | 
				
			||||||
 | 
					                ]
 | 
				
			||||||
 | 
					              }
 | 
				
			||||||
 | 
					            ]
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        ]
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    ]
 | 
				
			||||||
 | 
					  },
 | 
				
			||||||
 | 
					  [ScopeNames.WITH_ACCOUNT_DETAILS]: {
 | 
				
			||||||
    include: [
 | 
					    include: [
 | 
				
			||||||
      {
 | 
					      {
 | 
				
			||||||
        model: () => VideoChannelModel,
 | 
					        model: () => VideoChannelModel,
 | 
				
			||||||
| 
						 | 
					@ -146,6 +175,9 @@ enum ScopeNames {
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    {
 | 
					    {
 | 
				
			||||||
      fields: [ 'channelId' ]
 | 
					      fields: [ 'channelId' ]
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					      fields: [ 'id', 'privacy' ]
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  ]
 | 
					  ]
 | 
				
			||||||
})
 | 
					})
 | 
				
			||||||
| 
						 | 
					@ -461,7 +493,7 @@ export class VideoModel extends Model<VideoModel> {
 | 
				
			||||||
      order: [ getSort(sort) ]
 | 
					      order: [ getSort(sort) ]
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return VideoModel.scope([ ScopeNames.AVAILABLE_FOR_LIST, ScopeNames.WITH_ACCOUNT ])
 | 
					    return VideoModel.scope([ ScopeNames.AVAILABLE_FOR_LIST, ScopeNames.WITH_ACCOUNT_API ])
 | 
				
			||||||
      .findAndCountAll(query)
 | 
					      .findAndCountAll(query)
 | 
				
			||||||
      .then(({ rows, count }) => {
 | 
					      .then(({ rows, count }) => {
 | 
				
			||||||
        return {
 | 
					        return {
 | 
				
			||||||
| 
						 | 
					@ -496,7 +528,7 @@ export class VideoModel extends Model<VideoModel> {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (t !== undefined) query.transaction = t
 | 
					    if (t !== undefined) query.transaction = t
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return VideoModel.scope([ ScopeNames.WITH_ACCOUNT, ScopeNames.WITH_FILES ]).findOne(query)
 | 
					    return VideoModel.scope([ ScopeNames.WITH_ACCOUNT_DETAILS, ScopeNames.WITH_FILES ]).findOne(query)
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static loadByUUIDOrURL (uuid: string, url: string, t?: Sequelize.Transaction) {
 | 
					  static loadByUUIDOrURL (uuid: string, url: string, t?: Sequelize.Transaction) {
 | 
				
			||||||
| 
						 | 
					@ -520,7 +552,7 @@ export class VideoModel extends Model<VideoModel> {
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return VideoModel
 | 
					    return VideoModel
 | 
				
			||||||
      .scope([ ScopeNames.WITH_TAGS, ScopeNames.WITH_FILES, ScopeNames.WITH_ACCOUNT ])
 | 
					      .scope([ ScopeNames.WITH_TAGS, ScopeNames.WITH_FILES, ScopeNames.WITH_ACCOUNT_DETAILS ])
 | 
				
			||||||
      .findById(id, options)
 | 
					      .findById(id, options)
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -545,7 +577,7 @@ export class VideoModel extends Model<VideoModel> {
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return VideoModel
 | 
					    return VideoModel
 | 
				
			||||||
      .scope([ ScopeNames.WITH_TAGS, ScopeNames.WITH_FILES, ScopeNames.WITH_ACCOUNT ])
 | 
					      .scope([ ScopeNames.WITH_TAGS, ScopeNames.WITH_FILES, ScopeNames.WITH_ACCOUNT_DETAILS ])
 | 
				
			||||||
      .findOne(options)
 | 
					      .findOne(options)
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -563,7 +595,7 @@ export class VideoModel extends Model<VideoModel> {
 | 
				
			||||||
        ScopeNames.WITH_SHARES,
 | 
					        ScopeNames.WITH_SHARES,
 | 
				
			||||||
        ScopeNames.WITH_TAGS,
 | 
					        ScopeNames.WITH_TAGS,
 | 
				
			||||||
        ScopeNames.WITH_FILES,
 | 
					        ScopeNames.WITH_FILES,
 | 
				
			||||||
        ScopeNames.WITH_ACCOUNT,
 | 
					        ScopeNames.WITH_ACCOUNT_DETAILS,
 | 
				
			||||||
        ScopeNames.WITH_COMMENTS
 | 
					        ScopeNames.WITH_COMMENTS
 | 
				
			||||||
      ])
 | 
					      ])
 | 
				
			||||||
      .findOne(options)
 | 
					      .findOne(options)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,8 +3,9 @@
 | 
				
			||||||
import * as chai from 'chai'
 | 
					import * as chai from 'chai'
 | 
				
			||||||
import 'mocha'
 | 
					import 'mocha'
 | 
				
			||||||
import {
 | 
					import {
 | 
				
			||||||
  flushTests, killallServers, makeGetRequest, makePostBodyRequest, runServer, ServerInfo, setAccessTokensToServers,
 | 
					  createUser,
 | 
				
			||||||
  uploadVideo
 | 
					  flushTests, killallServers, makeDeleteRequest, makeGetRequest, makePostBodyRequest, runServer, ServerInfo, setAccessTokensToServers,
 | 
				
			||||||
 | 
					  uploadVideo, userLogin
 | 
				
			||||||
} from '../../utils'
 | 
					} from '../../utils'
 | 
				
			||||||
import { checkBadCountPagination, checkBadSortPagination, checkBadStartPagination } from '../../utils/requests/check-api-params'
 | 
					import { checkBadCountPagination, checkBadSortPagination, checkBadStartPagination } from '../../utils/requests/check-api-params'
 | 
				
			||||||
import { addVideoCommentThread } from '../../utils/videos/video-comments'
 | 
					import { addVideoCommentThread } from '../../utils/videos/video-comments'
 | 
				
			||||||
| 
						 | 
					@ -16,6 +17,7 @@ describe('Test video comments API validator', function () {
 | 
				
			||||||
  let pathComment: string
 | 
					  let pathComment: string
 | 
				
			||||||
  let server: ServerInfo
 | 
					  let server: ServerInfo
 | 
				
			||||||
  let videoUUID: string
 | 
					  let videoUUID: string
 | 
				
			||||||
 | 
					  let userAccessToken: string
 | 
				
			||||||
  let commentId: number
 | 
					  let commentId: number
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // ---------------------------------------------------------------
 | 
					  // ---------------------------------------------------------------
 | 
				
			||||||
| 
						 | 
					@ -40,6 +42,15 @@ describe('Test video comments API validator', function () {
 | 
				
			||||||
      commentId = res.body.comment.id
 | 
					      commentId = res.body.comment.id
 | 
				
			||||||
      pathComment = '/api/v1/videos/' + videoUUID + '/comments/' + commentId
 | 
					      pathComment = '/api/v1/videos/' + videoUUID + '/comments/' + commentId
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					      const user = {
 | 
				
			||||||
 | 
					        username: 'user1',
 | 
				
			||||||
 | 
					        password: 'my super password'
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      await createUser(server.url, server.accessToken, user.username, user.password)
 | 
				
			||||||
 | 
					      userAccessToken = await userLogin(server, user)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
  })
 | 
					  })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  describe('When listing video comment threads', function () {
 | 
					  describe('When listing video comment threads', function () {
 | 
				
			||||||
| 
						 | 
					@ -185,6 +196,30 @@ describe('Test video comments API validator', function () {
 | 
				
			||||||
    })
 | 
					    })
 | 
				
			||||||
  })
 | 
					  })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  describe('When removing video comments', function () {
 | 
				
			||||||
 | 
					    it('Should fail with a non authenticated user', async function () {
 | 
				
			||||||
 | 
					      await makeDeleteRequest({ url: server.url, path: pathComment, token: 'none', statusCodeExpected: 401 })
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    it('Should fail with another user', async function () {
 | 
				
			||||||
 | 
					      await makeDeleteRequest({ url: server.url, path: pathComment, token: userAccessToken, statusCodeExpected: 403 })
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    it('Should fail with an incorrect video', async function () {
 | 
				
			||||||
 | 
					      const path = '/api/v1/videos/ba708d62-e3d7-45d9-9d73-41b9097cc02d/comments/' + commentId
 | 
				
			||||||
 | 
					      await makeDeleteRequest({ url: server.url, path, token: server.accessToken, statusCodeExpected: 404 })
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    it('Should fail with an incorrect comment', async function () {
 | 
				
			||||||
 | 
					      const path = '/api/v1/videos/' + videoUUID + '/comments/124'
 | 
				
			||||||
 | 
					      await makeDeleteRequest({ url: server.url, path, token: server.accessToken, statusCodeExpected: 404 })
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    it('Should succeed with the correct parameters', async function () {
 | 
				
			||||||
 | 
					      await makeDeleteRequest({ url: server.url, path: pathComment, token: server.accessToken, statusCodeExpected: 204 })
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					  })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  describe('When a video has comments disabled', function () {
 | 
					  describe('When a video has comments disabled', function () {
 | 
				
			||||||
    before(async function () {
 | 
					    before(async function () {
 | 
				
			||||||
      const res = await uploadVideo(server.url, server.accessToken, { commentsEnabled: false })
 | 
					      const res = await uploadVideo(server.url, server.accessToken, { commentsEnabled: false })
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -13,7 +13,7 @@ import {
 | 
				
			||||||
  updateVideo, uploadVideo, userLogin, viewVideo, wait, webtorrentAdd
 | 
					  updateVideo, uploadVideo, userLogin, viewVideo, wait, webtorrentAdd
 | 
				
			||||||
} from '../../utils'
 | 
					} from '../../utils'
 | 
				
			||||||
import {
 | 
					import {
 | 
				
			||||||
  addVideoCommentReply, addVideoCommentThread, getVideoCommentThreads,
 | 
					  addVideoCommentReply, addVideoCommentThread, deleteVideoComment, getVideoCommentThreads,
 | 
				
			||||||
  getVideoThreadComments
 | 
					  getVideoThreadComments
 | 
				
			||||||
} from '../../utils/videos/video-comments'
 | 
					} from '../../utils/videos/video-comments'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -738,6 +738,37 @@ describe('Test multiple servers', function () {
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    })
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    it('Should delete the thread comments', async function () {
 | 
				
			||||||
 | 
					      this.timeout(10000)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      const res1 = await getVideoCommentThreads(servers[0].url, videoUUID, 0, 5)
 | 
				
			||||||
 | 
					      const threadId = res1.body.data.find(c => c.text === 'my super first comment').id
 | 
				
			||||||
 | 
					      await deleteVideoComment(servers[0].url, servers[0].accessToken, videoUUID, threadId)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      await wait(5000)
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    it('Should have the thread comments deleted on other servers too', async function () {
 | 
				
			||||||
 | 
					      for (const server of servers) {
 | 
				
			||||||
 | 
					        const res = await getVideoCommentThreads(server.url, videoUUID, 0, 5)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        expect(res.body.total).to.equal(1)
 | 
				
			||||||
 | 
					        expect(res.body.data).to.be.an('array')
 | 
				
			||||||
 | 
					        expect(res.body.data).to.have.lengthOf(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					          const comment: VideoComment = res.body.data[0]
 | 
				
			||||||
 | 
					          expect(comment).to.not.be.undefined
 | 
				
			||||||
 | 
					          expect(comment.inReplyToCommentId).to.be.null
 | 
				
			||||||
 | 
					          expect(comment.account.name).to.equal('root')
 | 
				
			||||||
 | 
					          expect(comment.account.host).to.equal('localhost:9003')
 | 
				
			||||||
 | 
					          expect(comment.totalReplies).to.equal(0)
 | 
				
			||||||
 | 
					          expect(dateIsValid(comment.createdAt as string)).to.be.true
 | 
				
			||||||
 | 
					          expect(dateIsValid(comment.updatedAt as string)).to.be.true
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    it('Should disable comments', async function () {
 | 
					    it('Should disable comments', async function () {
 | 
				
			||||||
      this.timeout(20000)
 | 
					      this.timeout(20000)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -9,7 +9,7 @@ import {
 | 
				
			||||||
  uploadVideo
 | 
					  uploadVideo
 | 
				
			||||||
} from '../../utils/index'
 | 
					} from '../../utils/index'
 | 
				
			||||||
import {
 | 
					import {
 | 
				
			||||||
  addVideoCommentReply, addVideoCommentThread, getVideoCommentThreads,
 | 
					  addVideoCommentReply, addVideoCommentThread, deleteVideoComment, getVideoCommentThreads,
 | 
				
			||||||
  getVideoThreadComments
 | 
					  getVideoThreadComments
 | 
				
			||||||
} from '../../utils/videos/video-comments'
 | 
					} from '../../utils/videos/video-comments'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -20,6 +20,7 @@ describe('Test video comments', function () {
 | 
				
			||||||
  let videoId
 | 
					  let videoId
 | 
				
			||||||
  let videoUUID
 | 
					  let videoUUID
 | 
				
			||||||
  let threadId
 | 
					  let threadId
 | 
				
			||||||
 | 
					  let replyToDeleteId: number
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  before(async function () {
 | 
					  before(async function () {
 | 
				
			||||||
    this.timeout(10000)
 | 
					    this.timeout(10000)
 | 
				
			||||||
| 
						 | 
					@ -61,6 +62,7 @@ describe('Test video comments', function () {
 | 
				
			||||||
    expect(comment.id).to.equal(comment.threadId)
 | 
					    expect(comment.id).to.equal(comment.threadId)
 | 
				
			||||||
    expect(comment.account.name).to.equal('root')
 | 
					    expect(comment.account.name).to.equal('root')
 | 
				
			||||||
    expect(comment.account.host).to.equal('localhost:9001')
 | 
					    expect(comment.account.host).to.equal('localhost:9001')
 | 
				
			||||||
 | 
					    expect(comment.account.url).to.equal('http://localhost:9001/accounts/root')
 | 
				
			||||||
    expect(comment.totalReplies).to.equal(0)
 | 
					    expect(comment.totalReplies).to.equal(0)
 | 
				
			||||||
    expect(dateIsValid(comment.createdAt as string)).to.be.true
 | 
					    expect(dateIsValid(comment.createdAt as string)).to.be.true
 | 
				
			||||||
    expect(dateIsValid(comment.updatedAt as string)).to.be.true
 | 
					    expect(dateIsValid(comment.updatedAt as string)).to.be.true
 | 
				
			||||||
| 
						 | 
					@ -132,6 +134,8 @@ describe('Test video comments', function () {
 | 
				
			||||||
    const secondChild = tree.children[1]
 | 
					    const secondChild = tree.children[1]
 | 
				
			||||||
    expect(secondChild.comment.text).to.equal('my second answer to thread 1')
 | 
					    expect(secondChild.comment.text).to.equal('my second answer to thread 1')
 | 
				
			||||||
    expect(secondChild.children).to.have.lengthOf(0)
 | 
					    expect(secondChild.children).to.have.lengthOf(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    replyToDeleteId = secondChild.comment.id
 | 
				
			||||||
  })
 | 
					  })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  it('Should create other threads', async function () {
 | 
					  it('Should create other threads', async function () {
 | 
				
			||||||
| 
						 | 
					@ -157,6 +161,38 @@ describe('Test video comments', function () {
 | 
				
			||||||
    expect(res.body.data[2].totalReplies).to.equal(0)
 | 
					    expect(res.body.data[2].totalReplies).to.equal(0)
 | 
				
			||||||
  })
 | 
					  })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  it('Should delete a reply', async function () {
 | 
				
			||||||
 | 
					    await deleteVideoComment(server.url, server.accessToken, videoId, replyToDeleteId)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const res = await getVideoThreadComments(server.url, videoUUID, threadId)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const tree: VideoCommentThreadTree = res.body
 | 
				
			||||||
 | 
					    expect(tree.comment.text).equal('my super first comment')
 | 
				
			||||||
 | 
					    expect(tree.children).to.have.lengthOf(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const firstChild = tree.children[0]
 | 
				
			||||||
 | 
					    expect(firstChild.comment.text).to.equal('my super answer to thread 1')
 | 
				
			||||||
 | 
					    expect(firstChild.children).to.have.lengthOf(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const childOfFirstChild = firstChild.children[0]
 | 
				
			||||||
 | 
					    expect(childOfFirstChild.comment.text).to.equal('my super answer to answer of thread 1')
 | 
				
			||||||
 | 
					    expect(childOfFirstChild.children).to.have.lengthOf(0)
 | 
				
			||||||
 | 
					  })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  it('Should delete a complete thread', async function () {
 | 
				
			||||||
 | 
					    await deleteVideoComment(server.url, server.accessToken, videoId, threadId)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const res = await getVideoCommentThreads(server.url, videoUUID, 0, 5, 'createdAt')
 | 
				
			||||||
 | 
					    expect(res.body.total).to.equal(2)
 | 
				
			||||||
 | 
					    expect(res.body.data).to.be.an('array')
 | 
				
			||||||
 | 
					    expect(res.body.data).to.have.lengthOf(2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    expect(res.body.data[0].text).to.equal('super thread 2')
 | 
				
			||||||
 | 
					    expect(res.body.data[0].totalReplies).to.equal(0)
 | 
				
			||||||
 | 
					    expect(res.body.data[1].text).to.equal('super thread 3')
 | 
				
			||||||
 | 
					    expect(res.body.data[1].totalReplies).to.equal(0)
 | 
				
			||||||
 | 
					  })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  after(async function () {
 | 
					  after(async function () {
 | 
				
			||||||
    killallServers([ server ])
 | 
					    killallServers([ server ])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,4 +1,5 @@
 | 
				
			||||||
import * as request from 'supertest'
 | 
					import * as request from 'supertest'
 | 
				
			||||||
 | 
					import { makeDeleteRequest } from '../'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function getVideoCommentThreads (url: string, videoId: number | string, start: number, count: number, sort?: string) {
 | 
					function getVideoCommentThreads (url: string, videoId: number | string, start: number, count: number, sort?: string) {
 | 
				
			||||||
  const path = '/api/v1/videos/' + videoId + '/comment-threads'
 | 
					  const path = '/api/v1/videos/' + videoId + '/comment-threads'
 | 
				
			||||||
| 
						 | 
					@ -54,11 +55,29 @@ function addVideoCommentReply (
 | 
				
			||||||
    .expect(expectedStatus)
 | 
					    .expect(expectedStatus)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function deleteVideoComment (
 | 
				
			||||||
 | 
					  url: string,
 | 
				
			||||||
 | 
					  token: string,
 | 
				
			||||||
 | 
					  videoId: number | string,
 | 
				
			||||||
 | 
					  commentId: number,
 | 
				
			||||||
 | 
					  statusCodeExpected = 204
 | 
				
			||||||
 | 
					) {
 | 
				
			||||||
 | 
					  const path = '/api/v1/videos/' + videoId + '/comments/' + commentId
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return makeDeleteRequest({
 | 
				
			||||||
 | 
					    url,
 | 
				
			||||||
 | 
					    path,
 | 
				
			||||||
 | 
					    token,
 | 
				
			||||||
 | 
					    statusCodeExpected
 | 
				
			||||||
 | 
					  })
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ---------------------------------------------------------------------------
 | 
					// ---------------------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export {
 | 
					export {
 | 
				
			||||||
  getVideoCommentThreads,
 | 
					  getVideoCommentThreads,
 | 
				
			||||||
  getVideoThreadComments,
 | 
					  getVideoThreadComments,
 | 
				
			||||||
  addVideoCommentThread,
 | 
					  addVideoCommentThread,
 | 
				
			||||||
  addVideoCommentReply
 | 
					  addVideoCommentReply,
 | 
				
			||||||
 | 
					  deleteVideoComment
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,6 +3,7 @@ import { Avatar } from '../avatars/avatar.model'
 | 
				
			||||||
export interface Account {
 | 
					export interface Account {
 | 
				
			||||||
  id: number
 | 
					  id: number
 | 
				
			||||||
  uuid: string
 | 
					  uuid: string
 | 
				
			||||||
 | 
					  url: string
 | 
				
			||||||
  name: string
 | 
					  name: string
 | 
				
			||||||
  displayName: string
 | 
					  displayName: string
 | 
				
			||||||
  host: string
 | 
					  host: string
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -6,5 +6,6 @@ export enum UserRight {
 | 
				
			||||||
  MANAGE_VIDEO_BLACKLIST,
 | 
					  MANAGE_VIDEO_BLACKLIST,
 | 
				
			||||||
  MANAGE_JOBS,
 | 
					  MANAGE_JOBS,
 | 
				
			||||||
  REMOVE_ANY_VIDEO,
 | 
					  REMOVE_ANY_VIDEO,
 | 
				
			||||||
  REMOVE_ANY_VIDEO_CHANNEL
 | 
					  REMOVE_ANY_VIDEO_CHANNEL,
 | 
				
			||||||
 | 
					  REMOVE_ANY_VIDEO_COMMENT
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -23,7 +23,8 @@ const userRoleRights: { [ id: number ]: UserRight[] } = {
 | 
				
			||||||
    UserRight.MANAGE_VIDEO_BLACKLIST,
 | 
					    UserRight.MANAGE_VIDEO_BLACKLIST,
 | 
				
			||||||
    UserRight.MANAGE_VIDEO_ABUSES,
 | 
					    UserRight.MANAGE_VIDEO_ABUSES,
 | 
				
			||||||
    UserRight.REMOVE_ANY_VIDEO,
 | 
					    UserRight.REMOVE_ANY_VIDEO,
 | 
				
			||||||
    UserRight.REMOVE_ANY_VIDEO_CHANNEL
 | 
					    UserRight.REMOVE_ANY_VIDEO_CHANNEL,
 | 
				
			||||||
 | 
					    UserRight.REMOVE_ANY_VIDEO_COMMENT
 | 
				
			||||||
  ],
 | 
					  ],
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  [UserRole.USER]: []
 | 
					  [UserRole.USER]: []
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,6 +3,7 @@ import { Video } from './video.model'
 | 
				
			||||||
export interface VideoChannel {
 | 
					export interface VideoChannel {
 | 
				
			||||||
  id: number
 | 
					  id: number
 | 
				
			||||||
  name: string
 | 
					  name: string
 | 
				
			||||||
 | 
					  url: string
 | 
				
			||||||
  description: string
 | 
					  description: string
 | 
				
			||||||
  isLocal: boolean
 | 
					  isLocal: boolean
 | 
				
			||||||
  createdAt: Date | string
 | 
					  createdAt: Date | string
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue