|
1 | 1 | import * as vite from 'vite';
|
2 | 2 | import { Rollup } from 'vite';
|
3 |
| -import { BundleList, Config, ObfuscationResult } from '../type'; |
4 |
| -import os from 'node:os'; |
5 |
| -import { isBoolean, isFileNameExcluded, isObject } from './is'; |
6 | 3 | import { Worker } from 'node:worker_threads';
|
7 | 4 | import path from 'node:path';
|
| 5 | +import os from 'node:os'; |
| 6 | +import { gzipSync } from 'node:zlib'; |
8 | 7 | import javascriptObfuscator from 'javascript-obfuscator';
|
9 |
| -import { CHUNK_PREFIX, VENDOR_MODULES } from './constants'; |
| 8 | + |
| 9 | +import type { BundleList, Config, FormatSizeResult, ObfuscationResult, SizeResult } from '../type'; |
| 10 | +import { isBoolean, isFileNameExcluded, isObject } from './is'; |
| 11 | +import { CHUNK_PREFIX, LOG_COLOR, SizeUnit, VENDOR_MODULES } from './constants'; |
10 | 12 |
|
11 | 13 | export class Log {
|
12 | 14 | private readonly _log: (msg: string) => void;
|
@@ -133,3 +135,91 @@ export function createWorkerTask(finalConfig: Config, chunk: BundleList) {
|
133 | 135 | });
|
134 | 136 | });
|
135 | 137 | }
|
| 138 | + |
| 139 | +export function formatSize(bytes: number): FormatSizeResult { |
| 140 | + const units = Object.values(SizeUnit); |
| 141 | + let i = 0; |
| 142 | + while (bytes >= 1024 && i < units.length - 1) { |
| 143 | + bytes /= 1024; |
| 144 | + i++; |
| 145 | + } |
| 146 | + return { |
| 147 | + value: Number(bytes.toFixed(2)), |
| 148 | + unit: units[i], |
| 149 | + }; |
| 150 | +} |
| 151 | + |
| 152 | +export class CodeSizeAnalyzer { |
| 153 | + private _log; |
| 154 | + private originalSize: SizeResult; |
| 155 | + private obfuscatedSize: SizeResult; |
| 156 | + private startTime: number; |
| 157 | + private endTime: number; |
| 158 | + |
| 159 | + constructor(log: Log) { |
| 160 | + this._log = log; |
| 161 | + this.originalSize = this.createEmptySizeResult(); |
| 162 | + this.obfuscatedSize = this.createEmptySizeResult(); |
| 163 | + this.startTime = 0; |
| 164 | + this.endTime = 0; |
| 165 | + } |
| 166 | + |
| 167 | + private createEmptySizeResult(): SizeResult { |
| 168 | + return { |
| 169 | + original: { value: 0, unit: SizeUnit.B }, |
| 170 | + gzip: { value: 0, unit: SizeUnit.B }, |
| 171 | + }; |
| 172 | + } |
| 173 | + |
| 174 | + start(originalBundleList: BundleList): void { |
| 175 | + this.startTime = performance.now(); |
| 176 | + this.originalSize = this.calculateBundleSize(originalBundleList); |
| 177 | + } |
| 178 | + |
| 179 | + end(obfuscatedBundleList: BundleList): void { |
| 180 | + this.obfuscatedSize = this.calculateBundleSize(obfuscatedBundleList); |
| 181 | + this.endTime = performance.now(); |
| 182 | + this.logResult(); |
| 183 | + } |
| 184 | + |
| 185 | + private calculateBundleSize(bundleList: BundleList): { original: FormatSizeResult; gzip: FormatSizeResult } { |
| 186 | + const { totalSize, gzipSize } = bundleList.reduce( |
| 187 | + (acc, [, bundleItem]) => { |
| 188 | + if (bundleItem.code) { |
| 189 | + const { code } = bundleItem; |
| 190 | + acc.totalSize += Buffer.byteLength(code, 'utf-8'); |
| 191 | + acc.gzipSize += gzipSync(code, { level: 9 }).byteLength; |
| 192 | + } |
| 193 | + return acc; |
| 194 | + }, |
| 195 | + { totalSize: 0, gzipSize: 0 }, |
| 196 | + ); |
| 197 | + |
| 198 | + return { |
| 199 | + original: formatSize(totalSize), |
| 200 | + gzip: formatSize(gzipSize), |
| 201 | + }; |
| 202 | + } |
| 203 | + |
| 204 | + private analyze(): string { |
| 205 | + const { originalSize, obfuscatedSize } = this; |
| 206 | + |
| 207 | + const consume = formatTime(this.endTime - this.startTime); |
| 208 | + |
| 209 | + const percentageIncrease = ( |
| 210 | + ((obfuscatedSize.original.value - originalSize.original.value) / originalSize.original.value) |
| 211 | + * 100 |
| 212 | + ).toFixed(2); |
| 213 | + |
| 214 | + const gzipPercentageIncrease = ( |
| 215 | + ((obfuscatedSize.gzip.value - originalSize.gzip.value) / originalSize.gzip.value) |
| 216 | + * 100 |
| 217 | + ).toFixed(2); |
| 218 | + |
| 219 | + return `✓ obfuscated in ${consume} | 📦 ${originalSize.original.value}${originalSize.original.unit} (gzip: ${originalSize.gzip.value}${originalSize.gzip.unit}) → 🔒 ${obfuscatedSize.original.value}${obfuscatedSize.original.unit} (gzip: ${obfuscatedSize.gzip.value}${obfuscatedSize.gzip.unit}) | 📈 ${percentageIncrease}% (gzip: ${gzipPercentageIncrease}%)`; |
| 220 | + } |
| 221 | + |
| 222 | + private logResult(): void { |
| 223 | + this._log.forceLog(LOG_COLOR.success + '%s\x1b[0m', this.analyze()); |
| 224 | + } |
| 225 | +} |
0 commit comments