UNPKG

nativescript

Version:

Command-line interface for building NativeScript projects

190 lines (189 loc) • 8.53 kB
"use strict"; var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.IOSDeviceLiveSyncService = void 0; const constants = require("../../constants"); const minimatch_1 = require("minimatch"); const _ = require("lodash"); const device_livesync_service_base_1 = require("./device-livesync-service-base"); const decorators_1 = require("../../common/decorators"); const semver = require("semver"); let currentPageReloadId = 0; class IOSDeviceLiveSyncService extends device_livesync_service_base_1.DeviceLiveSyncServiceBase { constructor($logger, $iOSSocketRequestExecutor, $devicePlatformsConstants, $lockService, platformsDataService, $platformCommandHelper, device) { super(platformsDataService, device); this.$logger = $logger; this.$iOSSocketRequestExecutor = $iOSSocketRequestExecutor; this.$devicePlatformsConstants = $devicePlatformsConstants; this.$lockService = $lockService; this.platformsDataService = platformsDataService; this.$platformCommandHelper = $platformCommandHelper; this.device = device; } canRefreshWithNotification(projectData, liveSyncInfo) { if (liveSyncInfo && liveSyncInfo.forceRefreshWithSocket) { return false; } if (this.device.isEmulator) { return false; } const currentRuntimeVersion = this.$platformCommandHelper.getCurrentPlatformVersion(this.$devicePlatformsConstants.iOS, projectData); const canRefresh = semver.gte(semver.coerce(currentRuntimeVersion), IOSDeviceLiveSyncService.MIN_RUNTIME_VERSION_WITH_REFRESH_NOTIFICATION); return canRefresh; } async setupSocketIfNeeded(projectData) { if (this.socket) { return true; } const appId = projectData.projectIdentifiers.ios; try { const ensureAppStarted = true; this.socket = await this.device.getDebugSocket(appId, projectData.projectName, projectData.projectDir, ensureAppStarted); } catch (err) { this.$logger.trace(`Error while connecting to the debug socket. Error is:`, err); } if (!this.socket) { return false; } this.attachEventHandlers(); return true; } async removeFiles(deviceAppData, localToDevicePaths) { await Promise.all(_.map(localToDevicePaths, (localToDevicePathData) => this.device.fileSystem.deleteFile(localToDevicePathData.getDevicePath(), deviceAppData.appIdentifier))); } async shouldRestart(projectData, liveSyncInfo) { let shouldRestart = false; const deviceAppData = liveSyncInfo.deviceAppData; const localToDevicePaths = liveSyncInfo.modifiedFilesData; if (liveSyncInfo.isFullSync || liveSyncInfo.waitForDebugger) { shouldRestart = true; } else { const canExecuteFastSync = this.canExecuteFastSyncForPaths(liveSyncInfo, localToDevicePaths, projectData, deviceAppData.platform); const isRefreshConnectionSetup = this.canRefreshWithNotification(projectData, liveSyncInfo) || (!this.device.isOnlyWiFiConnected && this.socket); if (!canExecuteFastSync || !isRefreshConnectionSetup) { shouldRestart = true; } } return shouldRestart; } async tryRefreshApplication(projectData, liveSyncInfo) { let didRefresh = true; const localToDevicePaths = liveSyncInfo.modifiedFilesData; let scriptRelatedFiles = []; constants.LIVESYNC_EXCLUDED_FILE_PATTERNS.forEach((pattern) => (scriptRelatedFiles = _.concat(scriptRelatedFiles, localToDevicePaths.filter((file) => (0, minimatch_1.minimatch)(file.getDevicePath(), pattern, { nocase: true }))))); const scriptFiles = _.filter(localToDevicePaths, (localToDevicePath) => _.endsWith(localToDevicePath.getDevicePath(), ".js")); const otherFiles = _.difference(localToDevicePaths, _.concat(scriptFiles, scriptRelatedFiles)); try { if (otherFiles.length) { didRefresh = await this.refreshApplicationCore(projectData, liveSyncInfo); } } catch (e) { didRefresh = false; } return didRefresh; } async refreshApplicationCore(projectData, liveSyncInfo) { let didRefresh = true; if (this.canRefreshWithNotification(projectData, liveSyncInfo)) { didRefresh = await this.refreshWithNotification(projectData); } else { if (await this.setupSocketIfNeeded(projectData)) { await this.reloadPage(); } else { didRefresh = false; } } return didRefresh; } async refreshWithNotification(projectData) { let didRefresh = false; await this.$lockService.executeActionWithLock(async () => { didRefresh = await this.$iOSSocketRequestExecutor.executeRefreshRequest(this.device, 5, projectData.projectIdentifiers.ios); }, `ios-device-livesync-${this.device.deviceInfo.identifier}-${projectData.projectIdentifiers.ios}.lock`); return didRefresh; } async restartApplication(projectData, liveSyncInfo) { await this.device.applicationManager.restartApplication({ appId: liveSyncInfo.deviceAppData.appIdentifier, projectName: projectData.projectName, waitForDebugger: liveSyncInfo.waitForDebugger, projectDir: projectData.projectDir, }); if (liveSyncInfo.useHotModuleReload) { await this.setupSocketIfNeeded(projectData); } } async reloadPage() { const message = JSON.stringify({ method: "Page.reload", params: { ignoreCache: false, }, id: ++currentPageReloadId, }); await this.sendMessage(message); } attachEventHandlers() { this.socket.on("close", (hadError) => { this.$logger.trace(`Socket closed, hadError is ${hadError}.`); this.socket = null; }); this.socket.on("error", (error) => { this.$logger.trace(`Socket error received: ${error}`); }); this.socket.on("data", (data) => { this.$logger.trace(`Socket sent data: ${data.toString()}`); }); } async sendMessage(message) { try { await new Promise((resolve, reject) => { let isResolved = false; const length = Buffer.byteLength(message, "utf16le"); const payload = Buffer.allocUnsafe(length + 4); payload.writeInt32BE(length, 0); payload.write(message, 4, length, "utf16le"); const errorCallback = (error) => { if (!isResolved) { isResolved = true; reject(error); } }; this.socket.once("error", errorCallback); this.socket.write(payload, "utf16le", () => { this.socket.removeListener("error", errorCallback); if (!isResolved) { isResolved = true; resolve(); } }); }); } catch (error) { this.$logger.trace("Error while sending message:", error); await this.destroySocket(); } } async destroySocket() { if (this.socket) { await this.device.destroyAllSockets(); this.socket = null; } } } exports.IOSDeviceLiveSyncService = IOSDeviceLiveSyncService; IOSDeviceLiveSyncService.MIN_RUNTIME_VERSION_WITH_REFRESH_NOTIFICATION = "6.1.0"; __decorate([ (0, decorators_1.performanceLog)() ], IOSDeviceLiveSyncService.prototype, "removeFiles", null);