2024-03-09 10:45:21 +00:00
import Logger from '@joplin/utils/Logger' ;
import JoplinError from './JoplinError' ;
import { ErrorCode } from './errors' ;
import { bytesToHuman } from '@joplin/utils/bytes' ;
const logger = Logger . create ( 'downloadController' ) ;
export interface DownloadController {
totalBytes : number ;
imagesCount : number ;
maxImagesCount : number ;
imageCountExpected : number ;
printStats ( imagesCountExpected : number ) : void ;
2024-04-05 11:16:49 +00:00
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
2024-03-09 10:45:21 +00:00
handleChunk ( request : any ) : ( chunk : any ) = > void ;
limitMessage ( ) : string ;
}
export class LimitedDownloadController implements DownloadController {
private totalBytes_ = 0 ;
// counts before the downloaded has finished, so at the end if the totalBytes > maxTotalBytesAllowed
// it means that imageCount will be higher than the total downloaded during the process
private imagesCount_ = 0 ;
// how many images links the content has
private imageCountExpected_ = 0 ;
2024-05-17 08:42:16 +00:00
private requestId = '' ;
2024-03-09 10:45:21 +00:00
private maxTotalBytes = 0 ;
public readonly maxImagesCount : number ;
2024-05-17 08:42:16 +00:00
public constructor ( maxTotalBytes : number , maxImagesCount : number , requestId : string ) {
2024-03-09 10:45:21 +00:00
this . maxTotalBytes = maxTotalBytes ;
this . maxImagesCount = maxImagesCount ;
2024-05-17 08:42:16 +00:00
this . requestId = requestId ;
2024-03-09 10:45:21 +00:00
}
public set totalBytes ( value : number ) {
if ( this . totalBytes_ >= this . maxTotalBytes ) {
2024-05-17 08:42:16 +00:00
throw new JoplinError ( ` ${ this . requestId } : Total bytes stored ( ${ this . totalBytes_ } ) has exceeded the amount established ( ${ this . maxTotalBytes } ) ` , ErrorCode . DownloadLimiter ) ;
2024-03-09 10:45:21 +00:00
}
this . totalBytes_ = value ;
}
public get totalBytes() {
return this . totalBytes_ ;
}
public set imagesCount ( value : number ) {
if ( this . imagesCount_ > this . maxImagesCount ) {
2024-05-17 08:42:16 +00:00
throw new JoplinError ( ` ${ this . requestId } : Total images to be stored ( ${ this . imagesCount_ } ) has exceeded the amount established ( ${ this . maxImagesCount } ) ` , ErrorCode . DownloadLimiter ) ;
2024-03-09 10:45:21 +00:00
}
this . imagesCount_ = value ;
}
public get imagesCount() {
return this . imagesCount_ ;
}
public set imageCountExpected ( value : number ) {
this . imageCountExpected_ = value ;
}
public get imageCountExpected() {
return this . imageCountExpected_ ;
}
2024-04-05 11:16:49 +00:00
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
2024-03-09 10:45:21 +00:00
public handleChunk ( request : any ) {
2024-04-05 11:16:49 +00:00
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
2024-03-09 10:45:21 +00:00
return ( chunk : any ) = > {
try {
this . totalBytes += chunk . length ;
} catch ( error ) {
request . destroy ( error ) ;
}
} ;
}
public printStats() {
2024-05-17 08:42:16 +00:00
const totalBytes = ` Total downloaded: ${ bytesToHuman ( this . totalBytes ) } . Maximum: ${ bytesToHuman ( this . maxTotalBytes ) } ` ;
2024-03-09 10:45:21 +00:00
const totalImages = ` Images initiated for download: ${ this . imagesCount_ } . Maximum: ${ this . maxImagesCount } . Expected: ${ this . imageCountExpected } ` ;
2024-05-17 08:42:16 +00:00
logger . info ( ` ${ this . requestId } : ${ totalBytes } ` ) ;
logger . info ( ` ${ this . requestId } : ${ totalImages } ` ) ;
2024-03-09 10:45:21 +00:00
}
public limitMessage() {
if ( this . imagesCount_ > this . maxImagesCount ) {
return ` The maximum image count of ${ this . maxImagesCount } has been exceeded. Image count in your content: ${ this . imageCountExpected } ` ;
}
if ( this . totalBytes >= this . maxTotalBytes ) {
return ` The maximum content size ${ bytesToHuman ( this . maxTotalBytes ) } has been exceeded. Content size: ( ${ bytesToHuman ( this . totalBytes ) } ) ` ;
}
return '' ;
}
}