/* * Copyright (c) 2023 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import path from 'path'; import { EventEmitter } from 'events'; import * as ts from 'typescript'; import { projectConfig, globalProgram } from '../../../main'; import { serviceChecker, languageService, printDiagnostic, fastBuildLogger, emitBuildInfo, runArkTSLinter, targetESVersionChanged, collectFileToIgnoreDiagnostics } from '../../ets_checker'; import { TS_WATCH_END_MSG } from '../../pre_define'; import { setChecker, startTimeStatisticsLocation, stopTimeStatisticsLocation, CompilationTimeStatistics } from '../../utils'; import { configureSyscapInfo, configurePermission } from '../system_api/api_check_utils'; export let tsWatchEmitter: EventEmitter | undefined = undefined; export let tsWatchEndPromise: Promise; export function etsChecker() { let executedOnce: boolean = false; return { name: 'etsChecker', buildStart() { const compilationTime: CompilationTimeStatistics = new CompilationTimeStatistics(this.share, 'etsChecker', 'buildStart'); if (process.env.watchMode === 'true' && process.env.triggerTsWatch === 'true') { tsWatchEmitter = new EventEmitter(); tsWatchEndPromise = new Promise(resolve => { tsWatchEmitter.on(TS_WATCH_END_MSG, () => { resolve(); }); }); } if (this.share.projectConfig.deviceTypes) { configureSyscapInfo(this.share.projectConfig); } if (this.share.projectConfig.permission) { configurePermission(this.share.projectConfig); } if (this.share.projectConfig.integratedHsp) { projectConfig.integratedHsp = this.share.projectConfig.integratedHsp; projectConfig.resetBundleName = this.share.projectConfig.integratedHsp; } Object.assign(projectConfig, this.share.projectConfig); Object.assign(this.share.projectConfig, { compileHar: projectConfig.compileHar, compileShared: projectConfig.compileShared, moduleRootPath: projectConfig.moduleRootPath, buildPath: projectConfig.buildPath, isCrossplatform: projectConfig.isCrossplatform, requestPermissions: projectConfig.requestPermissions, definePermissions: projectConfig.definePermissions, syscapIntersectionSet: projectConfig.syscapIntersectionSet, syscapUnionSet: projectConfig.syscapUnionSet, deviceTypesMessage: projectConfig.deviceTypesMessage, compileSdkVersion: projectConfig.compileSdkVersion, compatibleSdkVersion: projectConfig.compatibleSdkVersion, runtimeOS: projectConfig.runtimeOS }); const logger = this.share.getLogger('etsChecker'); const rootFileNames: string[] = []; const resolveModulePaths: string[] = []; rootFileNamesCollect(rootFileNames); if (this.share && this.share.projectConfig && this.share.projectConfig.resolveModulePaths && Array.isArray(this.share.projectConfig.resolveModulePaths)) { resolveModulePaths.push(...this.share.projectConfig.resolveModulePaths); } if (process.env.watchMode === 'true') { !executedOnce && serviceChecker(rootFileNames, logger, resolveModulePaths, compilationTime, this.share); startTimeStatisticsLocation(compilationTime ? compilationTime.diagnosticTime : undefined); if (executedOnce) { const timePrinterInstance = ts.ArkTSLinterTimePrinter.getInstance(); timePrinterInstance.setArkTSTimePrintSwitch(false); timePrinterInstance.appendTime(ts.TimePhase.START); globalProgram.builderProgram = languageService.getBuilderProgram(/*withLinterProgram*/ true); globalProgram.program = globalProgram.builderProgram.getProgram(); timePrinterInstance.appendTime(ts.TimePhase.GET_PROGRAM); collectFileToIgnoreDiagnostics(rootFileNames); runArkTSLinter(); } executedOnce = true; const allDiagnostics: ts.Diagnostic[] = globalProgram.builderProgram .getSyntacticDiagnostics() .concat(globalProgram.builderProgram.getSemanticDiagnostics()); stopTimeStatisticsLocation(compilationTime ? compilationTime.diagnosticTime : undefined); emitBuildInfo(); allDiagnostics.forEach((diagnostic: ts.Diagnostic) => { printDiagnostic(diagnostic); }); fastBuildLogger.debug(TS_WATCH_END_MSG); tsWatchEmitter.emit(TS_WATCH_END_MSG); } else { serviceChecker(rootFileNames, logger, resolveModulePaths, compilationTime, this.share); } setChecker(); }, shouldInvalidCache(): boolean { // The generated js file might be different in some cases when we change the targetESVersion, // so we need to regenerate them all when targetESVersion is changed. return targetESVersionChanged; } }; } function rootFileNamesCollect(rootFileNames: string[]): void { const entryFiles: string[] = projectConfig.widgetCompile ? Object.values(projectConfig.cardEntryObj) : Object.values(projectConfig.entryObj); entryFiles.forEach((fileName: string) => { rootFileNames.push(path.resolve(fileName)); }); }