diff --git a/packages/tailwindcss-language-server/package.json b/packages/tailwindcss-language-server/package.json index 867f54b5..1b366667 100644 --- a/packages/tailwindcss-language-server/package.json +++ b/packages/tailwindcss-language-server/package.json @@ -34,7 +34,15 @@ "access": "public" }, "devDependencies": { - "@parcel/watcher": "2.0.3", + "@parcel/watcher": "2.5.1", + "@parcel/watcher-darwin-x64": "2.5.1", + "@parcel/watcher-darwin-arm64": "2.5.1", + "@parcel/watcher-win32-x64": "2.5.1", + "@parcel/watcher-win32-arm64": "2.5.1", + "@parcel/watcher-linux-x64-glibc": "2.5.1", + "@parcel/watcher-linux-x64-musl": "2.5.1", + "@parcel/watcher-linux-arm64-glibc": "2.5.1", + "@parcel/watcher-linux-arm64-musl": "2.5.1", "@tailwindcss/aspect-ratio": "0.4.2", "@tailwindcss/container-queries": "0.1.0", "@tailwindcss/forms": "0.5.3", diff --git a/packages/tailwindcss-language-server/src/projects.ts b/packages/tailwindcss-language-server/src/projects.ts index a8012920..25aacd5f 100644 --- a/packages/tailwindcss-language-server/src/projects.ts +++ b/packages/tailwindcss-language-server/src/projects.ts @@ -1164,6 +1164,11 @@ export async function createProjectService( let elapsed = process.hrtime.bigint() - start console.log(`---- RELOADED IN ${(Number(elapsed) / 1e6).toFixed(2)}ms ----`) + + let isTestMode = params.initializationOptions?.testMode ?? false + if (!isTestMode) return + + connection.sendNotification('@/tailwindCSS/projectReloaded') }, state, diff --git a/packages/tailwindcss-language-server/src/testing/index.ts b/packages/tailwindcss-language-server/src/testing/index.ts index 91a07b81..0a11bb9d 100644 --- a/packages/tailwindcss-language-server/src/testing/index.ts +++ b/packages/tailwindcss-language-server/src/testing/index.ts @@ -1,13 +1,20 @@ -import { onTestFinished, test, TestOptions } from 'vitest' +import { onTestFinished, test, TestContext, TestOptions } from 'vitest' import * as os from 'node:os' import * as fs from 'node:fs/promises' import * as path from 'node:path' import * as proc from 'node:child_process' import dedent from 'dedent' -export interface TestUtils { +export interface TestUtils> { /** The "cwd" for this test */ root: string + + /** + * The input for this test — taken from the `inputs` in the test config + * + * @see {TestConfig} + */ + input?: TestInput } export interface StorageSymlink { @@ -21,29 +28,39 @@ export interface Storage { [filePath: string]: string | Uint8Array | StorageSymlink } -export interface TestConfig { +export interface TestConfig> { name: string + inputs?: TestInput[] + fs?: Storage debug?: boolean - prepare?(utils: TestUtils): Promise - handle(utils: TestUtils & Extras): void | Promise + prepare?(utils: TestUtils): Promise + handle(utils: TestUtils & Extras): void | Promise options?: TestOptions } -export function defineTest(config: TestConfig) { - return test(config.name, config.options ?? {}, async ({ expect }) => { - let utils = await setup(config) +export function defineTest(config: TestConfig) { + async function runTest(ctx: TestContext, input?: I) { + let utils = await setup(config, input) let extras = await config.prepare?.(utils) await config.handle({ ...utils, ...extras, }) - }) + } + + if (config.inputs) { + return test.for(config.inputs ?? [])(config.name, config.options ?? {}, (input, ctx) => + runTest(ctx, input), + ) + } + + return test(config.name, config.options ?? {}, runTest) } -async function setup(config: TestConfig): Promise { +async function setup(config: TestConfig, input: I): Promise> { let randomId = Math.random().toString(36).substring(7) let baseDir = path.resolve(process.cwd(), `../../.debug/${randomId}`) @@ -56,7 +73,7 @@ async function setup(config: TestConfig): Promise { await installDependencies(baseDir, config.fs) } - onTestFinished(async (result) => { + onTestFinished(async (ctx) => { // Once done, move all the files to a new location try { await fs.rename(baseDir, doneDir) @@ -66,7 +83,7 @@ async function setup(config: TestConfig): Promise { console.error('Failed to move test files to done directory') } - if (result.state === 'fail') return + if (ctx.task.result?.state === 'fail') return if (path.sep === '\\') return @@ -79,6 +96,7 @@ async function setup(config: TestConfig): Promise { return { root: baseDir, + input, } } diff --git a/packages/tailwindcss-language-server/src/tw.ts b/packages/tailwindcss-language-server/src/tw.ts index d5cfba20..682aefcd 100644 --- a/packages/tailwindcss-language-server/src/tw.ts +++ b/packages/tailwindcss-language-server/src/tw.ts @@ -52,7 +52,7 @@ import { readCssFile } from './util/css' import { ProjectLocator, type ProjectConfig } from './project-locator' import type { TailwindCssSettings } from '@tailwindcss/language-service/src/util/state' import { createResolver, Resolver } from './resolver' -import { retry } from './util/retry' +import { analyzeStylesheet } from './version-guesser.js' const TRIGGER_CHARACTERS = [ // class attributes @@ -382,6 +382,13 @@ export class TW { for (let [, project] of this.projects) { if (!project.state.v4) continue + if ( + change.type === FileChangeType.Deleted && + changeAffectsFile(normalizedFilename, [project.projectConfig.configPath]) + ) { + continue + } + if (!changeAffectsFile(normalizedFilename, project.dependencies())) continue needsSoftRestart = true @@ -405,6 +412,31 @@ export class TW { needsRestart = true break } + + // + else { + // If the main CSS file in a project is deleted and then re-created + // the server won't restart because the project is gone by now and + // there's no concept of a "config file" for us to compare with + // + // So we'll check if the stylesheet could *potentially* create + // a new project but we'll only do so if no projects were found + // + // If we did this all the time we'd potentially restart the server + // unncessarily a lot while the user is editing their stylesheets + if (this.projects.size > 0) continue + + let content = await readCssFile(change.file) + if (!content) continue + + let stylesheet = analyzeStylesheet(content) + if (!stylesheet.root) continue + + if (!stylesheet.versions.includes('4')) continue + + needsRestart = true + break + } } let isConfigFile = isConfigMatcher(normalizedFilename) @@ -489,6 +521,10 @@ export class TW { } } } else if (parcel.getBinding()) { + console.log( + '[Global] Your LSP client does not support watching files on behalf of the server', + ) + console.log('[Global] Using bundled file watcher: @parcel/watcher') let typeMap = { create: FileChangeType.Created, update: FileChangeType.Changed, @@ -515,6 +551,10 @@ export class TW { }, }) } else { + console.log( + '[Global] Your LSP client does not support watching files on behalf of the server', + ) + console.log('[Global] Using bundled file watcher: chokidar') let watch: typeof chokidar.watch = require('chokidar').watch let chokidarWatcher = watch( [`**/${CONFIG_GLOB}`, `**/${PACKAGE_LOCK_GLOB}`, `**/${CSS_GLOB}`, `**/${TSCONFIG_GLOB}`], @@ -1041,11 +1081,17 @@ export class TW { this.watched.length = 0 } - restart(): void { + async restart(): void { + let isTestMode = this.initializeParams.initializationOptions?.testMode ?? false + console.log('----------\nRESTARTING\n----------') this.dispose() this.initPromise = undefined - this.init() + await this.init() + + if (isTestMode) { + this.connection.sendNotification('@/tailwindCSS/serverRestarted') + } } async softRestart(): Promise { diff --git a/packages/tailwindcss-language-server/src/watcher/index.js b/packages/tailwindcss-language-server/src/watcher/index.js index ecf582e6..46cec8e8 100644 --- a/packages/tailwindcss-language-server/src/watcher/index.js +++ b/packages/tailwindcss-language-server/src/watcher/index.js @@ -13,21 +13,24 @@ const uv = (process.versions.uv || '').split('.')[0] const prebuilds = { 'darwin-arm64': { - 'node.napi.glibc.node': () => - require('@parcel/watcher/prebuilds/darwin-arm64/node.napi.glibc.node'), + 'node.napi.glibc.node': () => require('@parcel/watcher-darwin-arm64/watcher.node'), }, 'darwin-x64': { - 'node.napi.glibc.node': () => - require('@parcel/watcher/prebuilds/darwin-x64/node.napi.glibc.node'), + 'node.napi.glibc.node': () => require('@parcel/watcher-darwin-x64/watcher.node'), }, 'linux-x64': { - 'node.napi.glibc.node': () => - require('@parcel/watcher/prebuilds/linux-x64/node.napi.glibc.node'), - 'node.napi.musl.node': () => require('@parcel/watcher/prebuilds/linux-x64/node.napi.musl.node'), + 'node.napi.glibc.node': () => require('@parcel/watcher-linux-x64-glibc/watcher.node'), + 'node.napi.musl.node': () => require('@parcel/watcher-linux-x64-musl/watcher.node'), + }, + 'linux-arm64': { + 'node.napi.glibc.node': () => require('@parcel/watcher-linux-arm64-glibc/watcher.node'), + 'node.napi.musl.node': () => require('@parcel/watcher-linux-arm64-musl/watcher.node'), }, 'win32-x64': { - 'node.napi.glibc.node': () => - require('@parcel/watcher/prebuilds/win32-x64/node.napi.glibc.node'), + 'node.napi.glibc.node': () => require('@parcel/watcher-win32-x64/watcher.node'), + }, + 'win32-arm64': { + 'node.napi.glibc.node': () => require('@parcel/watcher-win32-arm64/watcher.node'), }, } diff --git a/packages/tailwindcss-language-server/tests/env/restart.test.ts b/packages/tailwindcss-language-server/tests/env/restart.test.ts new file mode 100644 index 00000000..25fcadb5 --- /dev/null +++ b/packages/tailwindcss-language-server/tests/env/restart.test.ts @@ -0,0 +1,166 @@ +import { expect } from 'vitest' +import * as fs from 'node:fs/promises' +import * as path from 'node:path' +import { css, defineTest } from '../../src/testing' +import dedent from 'dedent' +import { createClient } from '../utils/client' + +defineTest({ + name: 'The design system is reloaded when the CSS changes ($watcher)', + fs: { + 'app.css': css` + @import 'tailwindcss'; + + @theme { + --color-primary: #c0ffee; + } + `, + }, + prepare: async ({ root }) => ({ + client: await createClient({ + root, + capabilities(caps) { + caps.workspace!.didChangeWatchedFiles!.dynamicRegistration = false + }, + }), + }), + handle: async ({ root, client }) => { + let doc = await client.open({ + lang: 'html', + text: '
', + }) + + //
+ // ^ + let hover = await doc.hover({ line: 0, character: 13 }) + + expect(hover).toEqual({ + contents: { + language: 'css', + value: dedent` + .text-primary { + color: var(--color-primary) /* #c0ffee */; + } + `, + }, + range: { + start: { line: 0, character: 12 }, + end: { line: 0, character: 24 }, + }, + }) + + let didReload = new Promise((resolve) => { + client.conn.onNotification('@/tailwindCSS/projectReloaded', resolve) + }) + + // Update the CSS + await fs.writeFile( + path.resolve(root, 'app.css'), + css` + @import 'tailwindcss'; + + @theme { + --color-primary: #bada55; + } + `, + ) + + await didReload + + //
+ // ^ + let hover2 = await doc.hover({ line: 0, character: 13 }) + + expect(hover2).toEqual({ + contents: { + language: 'css', + value: dedent` + .text-primary { + color: var(--color-primary) /* #bada55 */; + } + `, + }, + range: { + start: { line: 0, character: 12 }, + end: { line: 0, character: 24 }, + }, + }) + }, +}) + +defineTest({ + name: 'Server is "restarted" when a config file is removed', + fs: { + 'app.css': css` + @import 'tailwindcss'; + + @theme { + --color-primary: #c0ffee; + } + `, + }, + prepare: async ({ root }) => ({ + client: await createClient({ + root, + capabilities(caps) { + caps.workspace!.didChangeWatchedFiles!.dynamicRegistration = false + }, + }), + }), + handle: async ({ root, client }) => { + let doc = await client.open({ + lang: 'html', + text: '
', + }) + + //
+ // ^ + let hover = await doc.hover({ line: 0, character: 13 }) + + expect(hover).toEqual({ + contents: { + language: 'css', + value: dedent` + .text-primary { + color: var(--color-primary) /* #c0ffee */; + } + `, + }, + range: { + start: { line: 0, character: 12 }, + end: { line: 0, character: 24 }, + }, + }) + + // Remove the CSS file + let didRestart = new Promise((resolve) => { + client.conn.onNotification('@/tailwindCSS/serverRestarted', resolve) + }) + await fs.unlink(path.resolve(root, 'app.css')) + await didRestart + + //
+ // ^ + let hover2 = await doc.hover({ line: 0, character: 13 }) + expect(hover2).toEqual(null) + + // Re-create the CSS file + let didRestartAgain = new Promise((resolve) => { + client.conn.onNotification('@/tailwindCSS/serverRestarted', resolve) + }) + await fs.writeFile( + path.resolve(root, 'app.css'), + css` + @import 'tailwindcss'; + `, + ) + await didRestartAgain + + await new Promise((resolve) => setTimeout(resolve, 500)) + + //
+ // ^ + let hover3 = await doc.hover({ line: 0, character: 13 }) + expect(hover3).toEqual(null) + }, +}) diff --git a/packages/tailwindcss-language-server/tests/utils/client.ts b/packages/tailwindcss-language-server/tests/utils/client.ts index 3a860230..760b805b 100644 --- a/packages/tailwindcss-language-server/tests/utils/client.ts +++ b/packages/tailwindcss-language-server/tests/utils/client.ts @@ -6,7 +6,6 @@ import { CompletionList, CompletionParams, Diagnostic, - DidChangeWorkspaceFoldersNotification, Disposable, DocumentLink, DocumentLinkRequest, @@ -179,6 +178,11 @@ export interface ClientOptions extends ConnectOptions { * and the Tailwind CSS version it detects */ features?: Feature[] + + /** + * Tweak the client capabilities presented to the server + */ + capabilities?(caps: ClientCapabilities): ClientCapabilities | Promise | void } export interface Client extends ClientWorkspace { @@ -394,6 +398,8 @@ export async function createClient(opts: ClientOptions): Promise { }, } + capabilities = (await opts.capabilities?.(capabilities)) ?? capabilities + trace('Client initializing') await conn.sendRequest(InitializeRequest.type, { processId: process.pid, diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 92fea067..a72f9353 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -33,8 +33,32 @@ importers: packages/tailwindcss-language-server: devDependencies: '@parcel/watcher': - specifier: 2.0.3 - version: 2.0.3 + specifier: 2.5.1 + version: 2.5.1 + '@parcel/watcher-darwin-arm64': + specifier: 2.5.1 + version: 2.5.1 + '@parcel/watcher-darwin-x64': + specifier: 2.5.1 + version: 2.5.1 + '@parcel/watcher-linux-arm64-glibc': + specifier: 2.5.1 + version: 2.5.1 + '@parcel/watcher-linux-arm64-musl': + specifier: 2.5.1 + version: 2.5.1 + '@parcel/watcher-linux-x64-glibc': + specifier: 2.5.1 + version: 2.5.1 + '@parcel/watcher-linux-x64-musl': + specifier: 2.5.1 + version: 2.5.1 + '@parcel/watcher-win32-arm64': + specifier: 2.5.1 + version: 2.5.1 + '@parcel/watcher-win32-x64': + specifier: 2.5.1 + version: 2.5.1 '@tailwindcss/aspect-ratio': specifier: 0.4.2 version: 0.4.2(tailwindcss@3.4.17) @@ -767,8 +791,86 @@ packages: resolution: {integrity: sha512-wBqcGsMELZna0jDblGd7UXgOby45TQaMWmbFwWX+SEotk4HV6zG2t6rT9siyLhPk4P6YYqgfL1UO8nMWDBVJXQ==} engines: {node: ^16.14.0 || >=18.0.0} - '@parcel/watcher@2.0.3': - resolution: {integrity: sha512-PHh5PArr3nYGYVj9z/NSfDmmKEBNrg2bzoFgxzjTRBBxPUKx039x3HF6VGLFIfrghjJxcYn/IeSpdVwfob7KFA==} + '@parcel/watcher-android-arm64@2.5.1': + resolution: {integrity: sha512-KF8+j9nNbUN8vzOFDpRMsaKBHZ/mcjEjMToVMJOhTozkDonQFFrRcfdLWn6yWKCmJKmdVxSgHiYvTCef4/qcBA==} + engines: {node: '>= 10.0.0'} + cpu: [arm64] + os: [android] + + '@parcel/watcher-darwin-arm64@2.5.1': + resolution: {integrity: sha512-eAzPv5osDmZyBhou8PoF4i6RQXAfeKL9tjb3QzYuccXFMQU0ruIc/POh30ePnaOyD1UXdlKguHBmsTs53tVoPw==} + engines: {node: '>= 10.0.0'} + cpu: [arm64] + os: [darwin] + + '@parcel/watcher-darwin-x64@2.5.1': + resolution: {integrity: sha512-1ZXDthrnNmwv10A0/3AJNZ9JGlzrF82i3gNQcWOzd7nJ8aj+ILyW1MTxVk35Db0u91oD5Nlk9MBiujMlwmeXZg==} + engines: {node: '>= 10.0.0'} + cpu: [x64] + os: [darwin] + + '@parcel/watcher-freebsd-x64@2.5.1': + resolution: {integrity: sha512-SI4eljM7Flp9yPuKi8W0ird8TI/JK6CSxju3NojVI6BjHsTyK7zxA9urjVjEKJ5MBYC+bLmMcbAWlZ+rFkLpJQ==} + engines: {node: '>= 10.0.0'} + cpu: [x64] + os: [freebsd] + + '@parcel/watcher-linux-arm-glibc@2.5.1': + resolution: {integrity: sha512-RCdZlEyTs8geyBkkcnPWvtXLY44BCeZKmGYRtSgtwwnHR4dxfHRG3gR99XdMEdQ7KeiDdasJwwvNSF5jKtDwdA==} + engines: {node: '>= 10.0.0'} + cpu: [arm] + os: [linux] + + '@parcel/watcher-linux-arm-musl@2.5.1': + resolution: {integrity: sha512-6E+m/Mm1t1yhB8X412stiKFG3XykmgdIOqhjWj+VL8oHkKABfu/gjFj8DvLrYVHSBNC+/u5PeNrujiSQ1zwd1Q==} + engines: {node: '>= 10.0.0'} + cpu: [arm] + os: [linux] + + '@parcel/watcher-linux-arm64-glibc@2.5.1': + resolution: {integrity: sha512-LrGp+f02yU3BN9A+DGuY3v3bmnFUggAITBGriZHUREfNEzZh/GO06FF5u2kx8x+GBEUYfyTGamol4j3m9ANe8w==} + engines: {node: '>= 10.0.0'} + cpu: [arm64] + os: [linux] + + '@parcel/watcher-linux-arm64-musl@2.5.1': + resolution: {integrity: sha512-cFOjABi92pMYRXS7AcQv9/M1YuKRw8SZniCDw0ssQb/noPkRzA+HBDkwmyOJYp5wXcsTrhxO0zq1U11cK9jsFg==} + engines: {node: '>= 10.0.0'} + cpu: [arm64] + os: [linux] + + '@parcel/watcher-linux-x64-glibc@2.5.1': + resolution: {integrity: sha512-GcESn8NZySmfwlTsIur+49yDqSny2IhPeZfXunQi48DMugKeZ7uy1FX83pO0X22sHntJ4Ub+9k34XQCX+oHt2A==} + engines: {node: '>= 10.0.0'} + cpu: [x64] + os: [linux] + + '@parcel/watcher-linux-x64-musl@2.5.1': + resolution: {integrity: sha512-n0E2EQbatQ3bXhcH2D1XIAANAcTZkQICBPVaxMeaCVBtOpBZpWJuf7LwyWPSBDITb7In8mqQgJ7gH8CILCURXg==} + engines: {node: '>= 10.0.0'} + cpu: [x64] + os: [linux] + + '@parcel/watcher-win32-arm64@2.5.1': + resolution: {integrity: sha512-RFzklRvmc3PkjKjry3hLF9wD7ppR4AKcWNzH7kXR7GUe0Igb3Nz8fyPwtZCSquGrhU5HhUNDr/mKBqj7tqA2Vw==} + engines: {node: '>= 10.0.0'} + cpu: [arm64] + os: [win32] + + '@parcel/watcher-win32-ia32@2.5.1': + resolution: {integrity: sha512-c2KkcVN+NJmuA7CGlaGD1qJh1cLfDnQsHjE89E60vUEMlqduHGCdCLJCID5geFVM0dOtA3ZiIO8BoEQmzQVfpQ==} + engines: {node: '>= 10.0.0'} + cpu: [ia32] + os: [win32] + + '@parcel/watcher-win32-x64@2.5.1': + resolution: {integrity: sha512-9lHBdJITeNR++EvSQVUcaZoWupyHfXe1jZvGZ06O/5MflPcuPLtEphScIBL+AiCWBO46tDSHzWyD0uDmmZqsgA==} + engines: {node: '>= 10.0.0'} + cpu: [x64] + os: [win32] + + '@parcel/watcher@2.5.1': + resolution: {integrity: sha512-dfUnCxiN9H4ap84DvD2ubjw+3vUNpstxa0TneY/Paat8a3R4uQZDLSvWjmznAY/DoahqTHl9V46HF/Zs3F29pg==} engines: {node: '>= 10.0.0'} '@pkgjs/parseargs@0.11.0': @@ -1372,6 +1474,11 @@ packages: resolution: {integrity: sha512-oSyFlqaTHCItVRGK5RmrmjB+CmaMOW7IaNA/kdxqhoa6d17j/5ce9O9eWXmV/KEdRwqpQA+Vqe8a8Bsybu4YnA==} engines: {node: '>=8'} + detect-libc@1.0.3: + resolution: {integrity: sha512-pGjwhsmsp4kL2RTz08wcOlGN83otlqHeD/Z5T8GXZB+/YcpQ/dgo+lbU8ZsGxV0HIvqqxo9l7mqYwyYMD9bKDg==} + engines: {node: '>=0.10'} + hasBin: true + detect-libc@2.0.3: resolution: {integrity: sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==} engines: {node: '>=8'} @@ -1910,15 +2017,11 @@ packages: resolution: {integrity: sha512-ThjYBfoDNr08AWx6hGaRbfPwxKV9kVzAzOzlLKbk2CuqXE2xnCh+cbAGnwM3t8Lq4v9rUB7VfondlkBckcJrVA==} engines: {node: '>=10'} - node-addon-api@3.2.1: - resolution: {integrity: sha512-mmcei9JghVNDYydghQmeDX8KoAm0FAiYyIcUt/N4nhyAipB17pllZQDOJD2fotxABnt4Mdz+dKTO7eftLg4d0A==} - node-addon-api@4.3.0: resolution: {integrity: sha512-73sE9+3UaLYYFmDsFZnqCInzPyh3MqIwZO9cw58yIqAZhONrrabrYyYe3TuIqtIiOuTXVhsGau8hcrhhwSsDIQ==} - node-gyp-build@4.8.1: - resolution: {integrity: sha512-OSs33Z9yWr148JZcbZd5WiAXhh/n9z8TxQcdMhIOlpN9AhWpLfvVFO73+m77bBABQMaY9XSvIa+qk0jlI7Gcaw==} - hasBin: true + node-addon-api@7.1.1: + resolution: {integrity: sha512-5m3bsyrjFWE1xf7nz7YXdN4udnVtXK6/Yfgn5qnahL6bCkf2yKt4k3nuTKAtT4r3IG8JNR2ncsIMdZuAzJjHQQ==} node-releases@2.0.18: resolution: {integrity: sha512-d9VeXT4SJ7ZeOqGX6R5EM022wpL+eWPooLI+5UpWn2jCT1aosUQEhQP214x33Wkwx3JQMvIm+tIoVOdodFS40g==} @@ -3027,10 +3130,57 @@ snapshots: dependencies: which: 4.0.0 - '@parcel/watcher@2.0.3': + '@parcel/watcher-android-arm64@2.5.1': + optional: true + + '@parcel/watcher-darwin-arm64@2.5.1': {} + + '@parcel/watcher-darwin-x64@2.5.1': {} + + '@parcel/watcher-freebsd-x64@2.5.1': + optional: true + + '@parcel/watcher-linux-arm-glibc@2.5.1': + optional: true + + '@parcel/watcher-linux-arm-musl@2.5.1': + optional: true + + '@parcel/watcher-linux-arm64-glibc@2.5.1': {} + + '@parcel/watcher-linux-arm64-musl@2.5.1': {} + + '@parcel/watcher-linux-x64-glibc@2.5.1': {} + + '@parcel/watcher-linux-x64-musl@2.5.1': {} + + '@parcel/watcher-win32-arm64@2.5.1': {} + + '@parcel/watcher-win32-ia32@2.5.1': + optional: true + + '@parcel/watcher-win32-x64@2.5.1': {} + + '@parcel/watcher@2.5.1': dependencies: - node-addon-api: 3.2.1 - node-gyp-build: 4.8.1 + detect-libc: 1.0.3 + is-glob: 4.0.3 + micromatch: 4.0.8 + node-addon-api: 7.1.1 + optionalDependencies: + '@parcel/watcher-android-arm64': 2.5.1 + '@parcel/watcher-darwin-arm64': 2.5.1 + '@parcel/watcher-darwin-x64': 2.5.1 + '@parcel/watcher-freebsd-x64': 2.5.1 + '@parcel/watcher-linux-arm-glibc': 2.5.1 + '@parcel/watcher-linux-arm-musl': 2.5.1 + '@parcel/watcher-linux-arm64-glibc': 2.5.1 + '@parcel/watcher-linux-arm64-musl': 2.5.1 + '@parcel/watcher-linux-x64-glibc': 2.5.1 + '@parcel/watcher-linux-x64-musl': 2.5.1 + '@parcel/watcher-win32-arm64': 2.5.1 + '@parcel/watcher-win32-ia32': 2.5.1 + '@parcel/watcher-win32-x64': 2.5.1 '@pkgjs/parseargs@0.11.0': optional: true @@ -3580,6 +3730,8 @@ snapshots: detect-indent@6.0.0: {} + detect-libc@1.0.3: {} + detect-libc@2.0.3: optional: true @@ -4122,12 +4274,10 @@ snapshots: semver: 7.7.1 optional: true - node-addon-api@3.2.1: {} - node-addon-api@4.3.0: optional: true - node-gyp-build@4.8.1: {} + node-addon-api@7.1.1: {} node-releases@2.0.18: {}