wdio-obsidian-service 1.1.0 → 1.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -2,14 +2,15 @@
2
2
  # WDIO Obsidian Service
3
3
 
4
4
  `wdio-obsidian-service` lets you test [Obsidian](https://obsidian.md) plugins end-to-end using
5
- [WebdriverIO](https://webdriver.io). The service handles:
6
- - Downloading and installing Obsidian
7
- - Testing your plugin on different Obsidian app versions and installer/electron versions
8
- - Opening and switching between test vaults during your tests
9
- - Downloading Chromedriver matching the Obsidian electron version
10
- - Sandboxing Obsidian so tests don't interfere with your system Obsidian installation or each other
11
- - Running tests in parallel
12
- - Provides helper functions for common testing tasks
5
+ [WebdriverIO](https://webdriver.io). The service can:
6
+ - Download and install Obsidian
7
+ - Test your plugin on different Obsidian app versions and installer/electron versions
8
+ - Download Chromedriver matching the Obsidian electron version
9
+ - Sandbox Obsidian so tests don't interfere with your system Obsidian installation or each other
10
+ - Run tests in parallel
11
+ - Open and switch between test vaults during your tests
12
+ - Provide helper functions for common testing tasks
13
+ - Run tests in GitHub CI
13
14
 
14
15
  ## Installation and Setup
15
16
  If you want to get going quickly, you can use the
@@ -18,7 +19,7 @@ all the setup you need to build and end-to-end test Obsidian plugins, including
18
19
 
19
20
  See also: [WebdriverIO | Getting Started](https://webdriver.io/docs/gettingstarted).
20
21
 
21
- To setup wdio-obsidian-service manually, run the WebdriverIO Starter Toolkit:
22
+ To set up wdio-obsidian-service manually, run the WebdriverIO Starter Toolkit:
22
23
  ```bash
23
24
  npm init wdio@latest .
24
25
  ```
@@ -27,10 +28,10 @@ Delete the generated `pageobjects` dir for now, or replace it with a stub for la
27
28
 
28
29
  Then install `wdio-obsidian-service` and other deps:
29
30
  ```bash
30
- npm install --save-dev wdio-obsidian-service wdio-obsidian-reporter mocha @types/mocha chai @types/chai @types/node
31
+ npm install --save-dev wdio-obsidian-service wdio-obsidian-reporter mocha @types/mocha
31
32
  ```
32
33
 
33
- And add this `tsconfig.json`:
34
+ And add this to `tsconfig.json`:
34
35
  ```json
35
36
  {
36
37
  "compilerOptions": {
@@ -44,7 +45,7 @@ And add this `tsconfig.json`:
44
45
  }
45
46
  ```
46
47
 
47
- Setup your `wdio.conf.ts` like so:
48
+ Set up your `wdio.conf.ts` like so:
48
49
  ```ts
49
50
  import * as path from "path"
50
51
 
@@ -92,10 +93,9 @@ export const config: WebdriverIO.Config = {
92
93
  }
93
94
  ```
94
95
 
95
- Create a file `test/specs/test.e2e.ts` with something like:
96
+ And create a file `test/specs/test.e2e.ts` with something like:
96
97
  ```ts
97
98
  import { browser } from '@wdio/globals'
98
- import { expect } from 'chai';
99
99
 
100
100
  describe('Test my plugin', function() {
101
101
  before(async function() {
@@ -109,15 +109,14 @@ describe('Test my plugin', function() {
109
109
  "sample-plugin:open-sample-modal-simple",
110
110
  );
111
111
  const modalEl = browser.$(".modal-container .modal-content");
112
- expect(await modalEl.isExisting()).to.equal(true);
113
- expect(await modalEl.getText()).to.equal("Woah!");
112
+ await expect(modalEl).toExist();
113
+ await expect(modalEl).toHaveText("Woah!");
114
114
  })
115
115
  })
116
116
  ```
117
117
 
118
118
  `wdio-obsidian-service` has a few helper functions that can be useful in your wdio conf, such as `obsidianBetaAvailable`
119
- which checks if there's a current Obsidian beta and you have the credentials to download it. E.g. to test your
120
- `minAppVersion`, `latest`, and `latest-beta` if it's available use:
119
+ which checks if there's a current Obsidian beta and you have the credentials to download it. E.g. to test on your `minAppVersion`, `latest`, and `latest-beta` if it's available, use:
121
120
  ```ts
122
121
  import { obsidianBetaAvailable } from "wdio-obsidian-service";
123
122
  const cacheDir = path.resolve(".obsidian-cache");
@@ -159,6 +158,15 @@ Obsidian installer. Windows firewall will sometimes complain about NodeJS, you c
159
158
  Currently `wdio-obsidian-service` only works for Obsidian Desktop. Testing Obsidian Mobile may be added in the future
160
159
  using WDIO + Appium.
161
160
 
161
+ ### Test Frameworks
162
+ WebdriverIO can run tests using [Mocha](https://mochajs.org), [Jasmine](https://jasmine.github.io), and
163
+ [Cucumber](https://cucumber.io/). Mocha is the easiest to set up and is used in all the wdio-obsidian-service examples.
164
+ Mocha can also run your unit tests, typically with the addition of an assertion library like
165
+ [Chai](https://www.chaijs.com). You can't run WebdriverIO using [Jest](https://jestjs.io), but if you already have Jest
166
+ unit tests (or just prefer Jest) you can easily continue using Jest for your unit tests and Mocha just for your e2e
167
+ tests. The built-in WebdriverIO [expect](https://webdriver.io/docs/api/expect-webdriverio) is very similar to Jest
168
+ matchers, so should be familiar to use.
169
+
162
170
  ## Usage
163
171
 
164
172
  ### Obsidian App vs Installer Versions
@@ -186,12 +194,13 @@ To set the installer version use `'wdio:obsidianOptions'.installerVersion`. It c
186
194
  - "earliest": run the oldest Obsidian installer compatible with `appVersion`
187
195
 
188
196
  You can see more configuration options for the capabilities
189
- [here](https://jesse-r-s-hines.github.io/wdio-obsidian-service/wdio-obsidian-service/ObsidianCapabilityOptions.html)
197
+ [here](https://jesse-r-s-hines.github.io/wdio-obsidian-service/wdio-obsidian-service/ObsidianCapabilityOptions.html).
190
198
 
191
199
  ### Opening and Switching between Vaults
192
200
  If all your tests use the same vault, you can set the vault in the `wdio:obsidianOptions` capabilities section. If you
193
- want to switch between vaults you can use `reloadObsidian` or `resetVault` during your tests. These can also be useful
194
- for reseting state between tests to avoid tests affecting each other (such as in Mocha `before` and `beforeEach` hooks).
201
+ need to switch between vaults during your test you can use the `reloadObsidian` or `resetVault` functions. These can
202
+ also be useful for reseting state between tests to avoid tests affecting each other (such as in Mocha `before` and
203
+ `beforeEach` hooks).
195
204
 
196
205
  `browser.reloadObsidian` completely reboots Obsidian with a fresh copy of the vault. This will clear all state, but is
197
206
  quite slow so avoid calling it too often.
@@ -222,7 +231,7 @@ API docs, including all configuration options and helper functions, are availabl
222
231
  [here](https://jesse-r-s-hines.github.io/wdio-obsidian-service/wdio-obsidian-service/README.html).
223
232
 
224
233
  ### GitHub CI Workflows
225
- The sample plugin has workflows setup to release and test your plugin, which you can see
234
+ The sample plugin has workflows set up to release and test your plugin, which you can see
226
235
  [here](https://github.com/jesse-r-s-hines/wdio-obsidian-service-sample-plugin#github-workflows).
227
236
 
228
237
  ### obsidian-launcher CLI
package/dist/index.d.ts CHANGED
@@ -78,43 +78,10 @@ declare class ObsidianPage {
78
78
  resetVault(...vaults: (string | Record<string, string>)[]): Promise<void>;
79
79
  }
80
80
  /**
81
- * Instance of ObsidianPage with helper methods for writing Obsidian tests
81
+ * Instance of {@link ObsidianPage} with helper methods for writing Obsidian tests.
82
82
  */
83
83
  declare const obsidianPage: ObsidianPage;
84
84
 
85
- /** Installed plugins, mapped by their id converted to camelCase */
86
- interface InstalledPlugins extends Record<string, obsidian.Plugin> {
87
- }
88
- /**
89
- * Argument passed to the `executeObsidian` browser command.
90
- */
91
- interface ExecuteObsidianArg {
92
- /**
93
- * There is a global "app" instance, but that may be removed in the future so you can use this to access it from
94
- * tests. See https://docs.obsidian.md/Plugins/Releasing/Plugin+guidelines#Avoid+using+global+app+instance
95
- */
96
- app: obsidian.App;
97
- /**
98
- * The full obsidian API. See https://github.com/obsidianmd/obsidian-api/blob/master/obsidian.d.ts
99
- */
100
- obsidian: typeof obsidian;
101
- /**
102
- * Object containing all installed plugins mapped by their id. Plugin ids are converted to converted to camelCase
103
- * for ease of destructuring.
104
- *
105
- * You can add types for your plugin(s) here with:
106
- * ```ts
107
- * import type MyPlugin from "../src/main.js"
108
- * declare module "wdio-obsidian-service" {
109
- * interface InstalledPlugins {
110
- * myPlugin: MyPlugin,
111
- * }
112
- * }
113
- * ```
114
- */
115
- plugins: InstalledPlugins;
116
- require: NodeJS.Require;
117
- }
118
85
  declare const browserCommands: {
119
86
  /**
120
87
  * Returns the Obsidian app version this test is running under.
@@ -130,8 +97,8 @@ declare const browserCommands: {
130
97
  * - app: Obsidian app instance
131
98
  * - obsidian: Full Obsidian API
132
99
  * - plugins: Object of all installed plugins, mapped by plugin id converted to camelCase.
133
- * - require: The customized require function Obsidian makes available to plugins. This is also made available
134
- * globally, so you can just use `require` directly instead of from ExecuteObsidianArg if you prefer.
100
+ * - require: The customized require function Obsidian makes available to plugins. This is also available globally,
101
+ * so you can just use `require` directly instead of from {@link ExecuteObsidianArg} if you prefer.
135
102
  *
136
103
  * Like `brower.execute`, you can pass other extra arguments to the function.
137
104
  *
@@ -185,7 +152,7 @@ type PlainObsidianBrowserCommands = typeof browserCommands;
185
152
  /**
186
153
  * Extra commands added to the WDIO Browser instance.
187
154
  *
188
- * See also: https://webdriver.io/docs/api/browser#custom-commands
155
+ * See also: https://webdriver.io/docs/api/browser
189
156
  * @interface
190
157
  */
191
158
  type ObsidianBrowserCommands = PlainObsidianBrowserCommands & {
@@ -213,25 +180,45 @@ type ObsidianBrowserCommands = PlainObsidianBrowserCommands & {
213
180
  theme?: string;
214
181
  }): Promise<string>;
215
182
  };
216
-
217
- declare const OBSIDIAN_CAPABILITY_KEY: "wdio:obsidianOptions";
218
- interface ObsidianServiceOptions {
183
+ /**
184
+ * Argument passed to the `executeObsidian` browser command.
185
+ */
186
+ interface ExecuteObsidianArg {
219
187
  /**
220
- * Override the `obsidian-versions.json` used by the service. Can be a file URL.
221
- * This is only really useful for this package's own internal tests.
188
+ * There is a global "app" instance, but that may be removed in the future so you can use this to access it from
189
+ * tests. See https://docs.obsidian.md/Plugins/Releasing/Plugin+guidelines#Avoid+using+global+app+instance
222
190
  */
223
- versionsUrl?: string;
191
+ app: obsidian.App;
224
192
  /**
225
- * Override the `community-plugins.json` used by the service. Can be a file URL.
226
- * This is only really useful for this package's own internal tests.
193
+ * The full obsidian API. See https://github.com/obsidianmd/obsidian-api/blob/master/obsidian.d.ts
227
194
  */
228
- communityPluginsUrl?: string;
195
+ obsidian: typeof obsidian;
229
196
  /**
230
- * Override the `community-css-themes.json` used by the service. Can be a file URL.
231
- * This is only really useful for this package's own internal tests.
197
+ * Object containing all installed plugins mapped by their id. Plugin ids are converted to camelCase for ease of
198
+ * destructuring.
199
+ *
200
+ * If you want to add typings for your plugin(s) you can use something like this in a `.d.ts`:
201
+ * ```ts
202
+ * import type MyPlugin from "../src/main.js"
203
+ * declare module "wdio-obsidian-service" {
204
+ * interface InstalledPlugins {
205
+ * myPlugin: MyPlugin,
206
+ * }
207
+ * }
208
+ * ```
232
209
  */
233
- communityThemesUrl?: string;
210
+ plugins: InstalledPlugins;
211
+ /**
212
+ * The customized require function Obsidian makes available to plugins. This is also available globally, so you can
213
+ * just use `require` directly instead of from `ExecuteObsidianArg` if you prefer.
214
+ */
215
+ require: NodeJS.Require;
216
+ }
217
+ /** Installed plugins, mapped by their id converted to camelCase */
218
+ interface InstalledPlugins extends Record<string, obsidian.Plugin> {
234
219
  }
220
+
221
+ declare const OBSIDIAN_CAPABILITY_KEY: "wdio:obsidianOptions";
235
222
  interface ObsidianCapabilityOptions {
236
223
  /**
237
224
  * Version of Obsidian to download and run.
@@ -241,7 +228,7 @@ interface ObsidianCapabilityOptions {
241
228
  * - "latest-beta": Run the latest beta Obsidian version (or latest is there is no current beta)
242
229
  * - To download Obsidian beta versions you'll need to have an Obsidian account with Catalyst and set the
243
230
  * `OBSIDIAN_USERNAME` and `OBSIDIAN_PASSWORD` environment variables. 2FA needs to be disabled.
244
- * - "earliest": Run the `minAppVersion` set in set in your `manifest.json`
231
+ * - "earliest": Run the `minAppVersion` set in your `manifest.json`
245
232
  *
246
233
  * Defaults to "latest".
247
234
  *
@@ -260,8 +247,8 @@ interface ObsidianCapabilityOptions {
260
247
  * See also: [Obsidian App vs Installer Versions](../README.md#obsidian-app-vs-installer-versions)
261
248
  *
262
249
  * Can be set to a specific version string or one of:
263
- * - "latest": Run the latest Obsidian installer.
264
- * - "earliest": Run the oldest Obsidian installer compatible with the specified Obsidian app version.
250
+ * - "latest": Run the latest Obsidian installer compatible with `appVersion`.
251
+ * - "earliest": Run the oldest Obsidian installer compatible with `appVersion`.
265
252
  *
266
253
  * Defaults to "earliest".
267
254
  */
@@ -301,6 +288,23 @@ interface ObsidianCapabilityOptions {
301
288
  */
302
289
  appPath?: string;
303
290
  }
291
+ interface ObsidianServiceOptions {
292
+ /**
293
+ * Override the `obsidian-versions.json` used by the service. Can be a file URL.
294
+ * This is only really useful for this package's own internal tests.
295
+ */
296
+ versionsUrl?: string;
297
+ /**
298
+ * Override the `community-plugins.json` used by the service. Can be a file URL.
299
+ * This is only really useful for this package's own internal tests.
300
+ */
301
+ communityPluginsUrl?: string;
302
+ /**
303
+ * Override the `community-css-themes.json` used by the service. Can be a file URL.
304
+ * This is only really useful for this package's own internal tests.
305
+ */
306
+ communityThemesUrl?: string;
307
+ }
304
308
  declare global {
305
309
  namespace WebdriverIO {
306
310
  interface Capabilities {
@@ -391,11 +395,17 @@ declare const launcher: typeof ObsidianLauncherService;
391
395
  declare function obsidianBetaAvailable(cacheDir?: string): Promise<boolean>;
392
396
  /**
393
397
  * Resolves Obsidian app and installer version strings to absolute versions.
394
- * @param appVersion Obsidian version string or "latest", "latest-beta" or "earliest". "earliest" will use the
395
- * minAppVersion set in your manifest.json.
396
- * @param installerVersion Obsidian version string or "latest" or "earliest". "earliest" will use the oldest
397
- * installer version compatible with the appVersion.
398
+ * @param appVersion Obsidian version string or one of
399
+ * - "latest": Get the current latest non-beta Obsidian version
400
+ * - "latest-beta": Get the current latest beta Obsidian version (or latest is there is no current beta)
401
+ * - "earliest": Get the `minAppVersion` set in your `manifest.json`
402
+ * @param installerVersion Obsidian version string or one of
403
+ * - "latest": Get the latest Obsidian installer compatible with `appVersion`
404
+ * - "earliest": Get the oldest Obsidian installer compatible with `appVersion`
398
405
  * @param cacheDir Obsidian cache dir, defaults to `.obsidian-cache`.
406
+ *
407
+ * See also: [Obsidian App vs Installer Versions](../README.md#obsidian-app-vs-installer-versions)
408
+ *
399
409
  * @returns [appVersion, installerVersion] with any "latest" etc. resolved to specific versions.
400
410
  */
401
411
  declare function resolveObsidianVersions(appVersion: string, installerVersion: string, cacheDir?: string): Promise<[string, string]>;
package/dist/index.js CHANGED
@@ -250,8 +250,8 @@ var browserCommands = {
250
250
  * - app: Obsidian app instance
251
251
  * - obsidian: Full Obsidian API
252
252
  * - plugins: Object of all installed plugins, mapped by plugin id converted to camelCase.
253
- * - require: The customized require function Obsidian makes available to plugins. This is also made available
254
- * globally, so you can just use `require` directly instead of from ExecuteObsidianArg if you prefer.
253
+ * - require: The customized require function Obsidian makes available to plugins. This is also available globally,
254
+ * so you can just use `require` directly instead of from {@link ExecuteObsidianArg} if you prefer.
255
255
  *
256
256
  * Like `brower.execute`, you can pass other extra arguments to the function.
257
257
  *
@@ -482,6 +482,12 @@ var ObsidianWorkerService = class {
482
482
  await browser2.executeObsidian(async ({ app }) => {
483
483
  await new Promise((resolve2) => app.workspace.onLayoutReady(resolve2));
484
484
  });
485
+ } else {
486
+ await browser2.execute(async () => {
487
+ if (document.readyState === "loading") {
488
+ return new Promise((resolve2) => document.addEventListener("DOMContentLoaded", () => resolve2()));
489
+ }
490
+ });
485
491
  }
486
492
  }
487
493
  /**
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/service.ts","../src/types.ts","../src/pageobjects/obsidianPage.ts","../src/browserCommands.ts","../src/utils.ts"],"sourcesContent":["/**\n * @module\n * @document ../README.md\n */\nimport ObsidianLauncher from \"obsidian-launcher\";\nimport { ObsidianLauncherService, ObsidianWorkerService } from \"./service.js\";\n/** @hidden */\nexport default ObsidianWorkerService;\n/** @hidden */\nexport const launcher = ObsidianLauncherService;\n\nexport type { ObsidianCapabilityOptions, ObsidianServiceOptions } from \"./types.js\";\nexport type { ObsidianBrowserCommands, ExecuteObsidianArg, InstalledPlugins } from \"./browserCommands.js\";\nexport { default as obsidianPage } from \"./pageobjects/obsidianPage.js\";\nexport type { ObsidianPage } from \"./pageobjects/obsidianPage.js\";\nexport type { PluginEntry, DownloadedPluginEntry, ThemeEntry, DownloadedThemeEntry } from \"obsidian-launcher\";\n\nexport { minSupportedObsidianVersion } from \"./service.js\";\n\n// Some convenience helpers for use in wdio.conf.(m)ts\n\n/**\n * Returns true if there is a current Obsidian beta and we have the credentials to download it, or its already in cache.\n * @param cacheDir Obsidian cache dir, defaults to `.obsidian-cache`.\n */\nexport async function obsidianBetaAvailable(cacheDir?: string) {\n const launcher = new ObsidianLauncher({cacheDir: cacheDir});\n const versionInfo = await launcher.getVersionInfo(\"latest-beta\");\n return versionInfo.isBeta && await launcher.isAvailable(versionInfo.version);\n}\n\n/**\n * Resolves Obsidian app and installer version strings to absolute versions.\n * @param appVersion Obsidian version string or \"latest\", \"latest-beta\" or \"earliest\". \"earliest\" will use the \n * minAppVersion set in your manifest.json.\n * @param installerVersion Obsidian version string or \"latest\" or \"earliest\". \"earliest\" will use the oldest\n * installer version compatible with the appVersion.\n * @param cacheDir Obsidian cache dir, defaults to `.obsidian-cache`.\n * @returns [appVersion, installerVersion] with any \"latest\" etc. resolved to specific versions.\n */\nexport async function resolveObsidianVersions(\n appVersion: string, installerVersion: string, cacheDir?: string,\n): Promise<[string, string]> {\n const launcher = new ObsidianLauncher({cacheDir: cacheDir});\n return await launcher.resolveVersions(appVersion, installerVersion);\n\n}\n","import fs from \"fs\"\nimport fsAsync from \"fs/promises\"\nimport path from \"path\"\nimport { SevereServiceError } from 'webdriverio'\nimport type { Capabilities, Options, Services } from '@wdio/types'\nimport logger from '@wdio/logger'\nimport { fileURLToPath } from \"url\"\nimport ObsidianLauncher, { DownloadedPluginEntry, DownloadedThemeEntry } from \"obsidian-launcher\"\nimport browserCommands from \"./browserCommands.js\"\nimport { ObsidianCapabilityOptions, ObsidianServiceOptions, OBSIDIAN_CAPABILITY_KEY } from \"./types.js\"\nimport obsidianPage from \"./pageobjects/obsidianPage.js\"\nimport { sleep } from \"./utils.js\"\nimport semver from \"semver\"\nimport _ from \"lodash\"\n\n\nconst log = logger(\"wdio-obsidian-service\");\n\nfunction getDefaultCacheDir() {\n return path.resolve(process.env.WEBDRIVER_CACHE_DIR ?? process.env.OBSIDIAN_CACHE ?? \"./.obsidian-cache\")\n}\n\n/**\n * Minimum Obsidian version that wdio-obsidian-service supports.\n */\nexport const minSupportedObsidianVersion: string = \"1.0.3\"\n\n\n/**\n * wdio launcher service.\n * Use in wdio.conf.ts like so:\n * ```ts\n * services: ['obsidian'],\n * ```\n * @hidden\n */\nexport class ObsidianLauncherService implements Services.ServiceInstance {\n private obsidianLauncher: ObsidianLauncher\n private readonly helperPluginPath: string\n\n constructor (\n public options: ObsidianServiceOptions,\n public capabilities: WebdriverIO.Capabilities,\n public config: Options.Testrunner\n ) {\n this.obsidianLauncher = new ObsidianLauncher({\n cacheDir: config.cacheDir ?? getDefaultCacheDir(),\n versionsUrl: options.versionsUrl,\n communityPluginsUrl: options.communityPluginsUrl, communityThemesUrl: options.communityThemesUrl,\n });\n this.helperPluginPath = path.resolve(path.join(fileURLToPath(import.meta.url), '../../helper-plugin'));\n }\n\n /**\n * Validates wdio:obsidianOptions and downloads Obsidian, plugins, and themes.\n */\n async onPrepare(config: Options.Testrunner, capabilities: Capabilities.TestrunnerCapabilities) {\n if (!Array.isArray(capabilities)) {\n capabilities = Object.values(capabilities as Capabilities.RequestedMultiremoteCapabilities).map(\n (multiremoteOption) => (multiremoteOption as Capabilities.WithRequestedCapabilities).capabilities,\n );\n }\n\n const obsidianCapabilities = capabilities.flatMap((cap) => {\n if ((\"browserName\" in cap) && cap.browserName === \"obsidian\") {\n return [cap as WebdriverIO.Capabilities];\n } else {\n return [];\n }\n });\n\n try {\n for (const cap of obsidianCapabilities) {\n const obsidianOptions = cap[OBSIDIAN_CAPABILITY_KEY] ?? {};\n \n const vault = obsidianOptions.vault != undefined ? path.resolve(obsidianOptions.vault) : undefined;\n \n const [appVersion, installerVersion] = await this.obsidianLauncher.resolveVersions(\n cap.browserVersion ?? cap[OBSIDIAN_CAPABILITY_KEY]?.appVersion ?? \"latest\",\n obsidianOptions.installerVersion ?? \"earliest\",\n );\n const installerVersionInfo = await this.obsidianLauncher.getVersionInfo(installerVersion);\n if (semver.lt(appVersion, minSupportedObsidianVersion)) {\n throw Error(`Minimum supported Obsidian version is ${minSupportedObsidianVersion}`)\n }\n\n let installerPath = obsidianOptions.binaryPath;\n if (!installerPath) {\n installerPath = await this.obsidianLauncher.downloadInstaller(installerVersion);\n }\n let appPath = obsidianOptions.appPath;\n if (!appPath) {\n appPath = await this.obsidianLauncher.downloadApp(appVersion);\n }\n let chromedriverPath = cap['wdio:chromedriverOptions']?.binary\n // wdio can download chromedriver for versions greater than 115 automatically\n if (!chromedriverPath && Number(installerVersionInfo.chromeVersion!.split(\".\")[0]) <= 115) {\n chromedriverPath = await this.obsidianLauncher.downloadChromedriver(installerVersion);\n }\n\n let plugins = obsidianOptions.plugins ?? [\".\"];\n plugins.push(this.helperPluginPath); // Always install the helper plugin\n plugins = await this.obsidianLauncher.downloadPlugins(plugins);\n\n const themes = await this.obsidianLauncher.downloadThemes(obsidianOptions.themes ?? []);\n\n if (obsidianOptions.vault != undefined && !fs.existsSync(obsidianOptions.vault)) {\n throw Error(`Vault \"${obsidianOptions.vault}\" doesn't exist`)\n }\n\n const args = [\n // Workaround for SUID issue on AppImages. See https://github.com/electron/electron/issues/42510\n ...(process.platform == 'linux' ? [\"--no-sandbox\"] : []),\n ...(cap['goog:chromeOptions']?.args ?? [])\n ];\n\n cap.browserName = \"chrome\";\n cap.browserVersion = installerVersionInfo.chromeVersion;\n cap[OBSIDIAN_CAPABILITY_KEY] = {\n ...obsidianOptions,\n plugins: plugins,\n themes: themes,\n binaryPath: installerPath,\n appPath: appPath,\n vault: vault,\n appVersion: appVersion, // Resolve the versions\n installerVersion: installerVersion,\n };\n cap['goog:chromeOptions'] = {\n binary: installerPath,\n windowTypes: [\"app\", \"webview\"],\n ...cap['goog:chromeOptions'],\n args: args,\n }\n cap['wdio:chromedriverOptions'] = {\n // allowedIps is not included in the types, but gets passed as --allowed-ips to chromedriver.\n // It defaults to [\"0.0.0.0\"] which makes Windows Firewall complain, and we don't need remote\n // connections anyways.\n allowedIps: [],\n ...cap['wdio:chromedriverOptions'],\n binary: chromedriverPath,\n } as any\n cap[\"wdio:enforceWebDriverClassic\"] = true;\n }\n } catch (e: any) {\n // By default wdio just logs service errors, throwing this makes it bail if anything goes wrong.\n throw new SevereServiceError(`Failed to download and setup Obsidian. Caused by: ${e.stack}\\n`+\n ` ------The above causes:-----`);\n }\n }\n}\n\n\n/**\n * wdio worker service.\n * Use in wdio.conf.ts like so:\n * ```ts\n * services: ['obsidian'],\n * ```\n * @hidden\n */\nexport class ObsidianWorkerService implements Services.ServiceInstance {\n private obsidianLauncher: ObsidianLauncher\n /** Directories to clean up after the tests */\n private tmpDirs: string[]\n\n constructor (\n public options: ObsidianServiceOptions,\n public capabilities: WebdriverIO.Capabilities,\n public config: Options.Testrunner\n ) {\n this.obsidianLauncher = new ObsidianLauncher({\n cacheDir: config.cacheDir ?? getDefaultCacheDir(),\n versionsUrl: options.versionsUrl,\n communityPluginsUrl: options.communityPluginsUrl, communityThemesUrl: options.communityThemesUrl,\n });\n this.tmpDirs = [];\n }\n\n /**\n * Setup vault and config dir for a sandboxed Obsidian instance.\n */\n private async setupObsidian(obsidianOptions: ObsidianCapabilityOptions) {\n let vault = obsidianOptions.vault;\n if (vault != undefined) {\n log.info(`Opening vault ${obsidianOptions.vault}`);\n vault = await this.obsidianLauncher.setupVault({\n vault,\n copy: true,\n plugins: obsidianOptions.plugins,\n themes: obsidianOptions.themes,\n });\n this.tmpDirs.push(vault);\n } else {\n log.info(`Opening Obsidian without a vault`)\n }\n\n const configDir = await this.obsidianLauncher.setupConfigDir({\n appVersion: obsidianOptions.appVersion!, installerVersion: obsidianOptions.installerVersion!,\n appPath: obsidianOptions.appPath!,\n vault: vault,\n });\n this.tmpDirs.push(configDir);\n\n return configDir;\n }\n\n /**\n * Wait for Obsidian to fully boot.\n */\n private async waitForReady(browser: WebdriverIO.Browser) {\n if (browser.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].vault != undefined) {\n await browser.waitUntil( // wait until the helper plugin is loaded\n () => browser.execute(() => !!(window as any).wdioObsidianService),\n {timeout: 30 * 1000, interval: 100},\n );\n await browser.executeObsidian(async ({app}) => {\n await new Promise<void>((resolve) => app.workspace.onLayoutReady(resolve) );\n })\n }\n }\n\n /**\n * Handles vault and sandboxed config directory setup.\n */\n async beforeSession(config: Options.Testrunner, capabilities: WebdriverIO.Capabilities) {\n if (!capabilities[OBSIDIAN_CAPABILITY_KEY]) return;\n\n const configDir = await this.setupObsidian(capabilities[OBSIDIAN_CAPABILITY_KEY]);\n\n capabilities['goog:chromeOptions']!.args = [\n `--user-data-dir=${configDir}`,\n ...(capabilities['goog:chromeOptions']!.args ?? [])\n ];\n }\n\n /**\n * Returns a plugin list with only the selected plugin ids enabled.\n */\n private selectPlugins(currentPlugins: DownloadedPluginEntry[], selection?: string[]) {\n if (selection !== undefined) {\n const unknownPlugins = _.difference(selection, currentPlugins.map(p => p.id));\n if (unknownPlugins.length > 0) {\n throw Error(`Unknown plugin ids: ${unknownPlugins.join(', ')}`)\n }\n return currentPlugins.map(p => ({\n ...p,\n enabled: selection.includes(p.id) || p.id == \"wdio-obsidian-service-plugin\",\n }));\n } else {\n return currentPlugins;\n }\n }\n\n /**\n * Returns a theme list with only the selected theme enabled.\n */\n private selectThemes(currentThemes: DownloadedThemeEntry[], selection?: string) {\n if (selection !== undefined) {\n if (selection != \"default\" && currentThemes.every((t: any) => t.name != selection)) {\n throw Error(`Unknown theme: ${selection}`);\n }\n return currentThemes.map((t: any) => ({...t, enabled: selection != 'default' && t.name === selection}));\n } else {\n return currentThemes;\n }\n }\n\n /**\n * Setup custom browser commands.\n */\n async before(capabilities: WebdriverIO.Capabilities, specs: never, browser: WebdriverIO.Browser) {\n // There's a slow event listener link on the browser \"command\" event when you reloadSession that causes some\n // warnings. This will silence them. TODO: Make issue or PR to wdio to fix this.\n browser.setMaxListeners(1000);\n\n if (!capabilities[OBSIDIAN_CAPABILITY_KEY]) return;\n\n const service = this; // eslint-disable-line @typescript-eslint/no-this-alias\n const reloadObsidian: typeof browser['reloadObsidian'] = async function(\n this: WebdriverIO.Browser,\n {vault, plugins, theme} = {},\n ) {\n const oldObsidianOptions = this.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY];\n let newCapabilities: WebdriverIO.Capabilities\n\n if (vault) {\n const newObsidianOptions = {\n ...oldObsidianOptions,\n vault: path.resolve(vault),\n plugins: service.selectPlugins(oldObsidianOptions.plugins, plugins),\n themes: service.selectThemes(oldObsidianOptions.themes, theme),\n }\n \n const configDir = await service.setupObsidian(newObsidianOptions);\n \n const newArgs = [\n `--user-data-dir=${configDir}`,\n ...this.requestedCapabilities['goog:chromeOptions'].args.filter((arg: string) => {\n const match = arg.match(/^--user-data-dir=(.*)$/);\n return !match || !service.tmpDirs.includes(match[1]);\n }),\n ]\n \n newCapabilities = {\n [OBSIDIAN_CAPABILITY_KEY]: newObsidianOptions,\n 'goog:chromeOptions': {\n ...this.requestedCapabilities['goog:chromeOptions'],\n args: newArgs,\n },\n };\n } else {\n // preserve vault and config dir\n newCapabilities = {};\n // Since we aren't recreating the vault, we'll need to reset plugins and themes here if specified.\n if (plugins) {\n const enabledPlugins = await browser.executeObsidian(({app}) =>\n [...(app as any).plugins.enabledPlugins].sort()\n )\n for (const pluginId of _.difference(enabledPlugins, plugins, ['wdio-obsidian-service-plugin'])) {\n await obsidianPage.disablePlugin(pluginId);\n }\n for (const pluginId of _.difference(plugins, enabledPlugins)) {\n await obsidianPage.enablePlugin(pluginId);\n }\n }\n if (theme) {\n await obsidianPage.setTheme(theme);\n }\n // Obsidian debounces saves to the config dir, and so changes to configuration made in the tests may not\n // get saved to disk before the reboot. Here I manually trigger save for plugins and themes, but other\n // configurations might not get saved. I haven't found a better way to flush everything than just\n // waiting a bit. Wdio has ways to mock the clock, which might work, but it's only supported when using\n // BiDi, which I can't get working on Obsidian.\n await browser.executeObsidian(async ({app}) => await Promise.all([\n (app as any).plugins.saveConfig(),\n (app.vault as any).saveConfig(),\n ]))\n await sleep(2000);\n }\n\n const sessionId = await browser.reloadSession({\n // if browserName is set, reloadSession tries to restart the driver entirely, so unset those\n ..._.omit(this.requestedCapabilities, ['browserName', 'browserVersion']),\n ...newCapabilities,\n });\n await service.waitForReady(this);\n return sessionId;\n }\n\n await browser.addCommand(\"reloadObsidian\", reloadObsidian);\n\n for (const [name, cmd] of Object.entries(browserCommands)) {\n await browser.addCommand(name, cmd);\n }\n\n await service.waitForReady(browser);\n }\n\n /**\n * Cleanup\n */\n async afterSession() {\n for (const tmpDir of this.tmpDirs) {\n await fsAsync.rm(tmpDir, { recursive: true, force: true });\n }\n }\n}\n","import type { ObsidianBrowserCommands } from \"./browserCommands.js\";\nimport type { PluginEntry, ThemeEntry } from \"obsidian-launcher\";\n\nexport const OBSIDIAN_CAPABILITY_KEY = \"wdio:obsidianOptions\" as const;\n\nexport interface ObsidianServiceOptions {\n /**\n * Override the `obsidian-versions.json` used by the service. Can be a file URL.\n * This is only really useful for this package's own internal tests.\n */\n versionsUrl?: string,\n /**\n * Override the `community-plugins.json` used by the service. Can be a file URL.\n * This is only really useful for this package's own internal tests.\n */\n communityPluginsUrl?: string,\n /**\n * Override the `community-css-themes.json` used by the service. Can be a file URL.\n * This is only really useful for this package's own internal tests.\n */\n communityThemesUrl?: string,\n}\n\nexport interface ObsidianCapabilityOptions {\n /**\n * Version of Obsidian to download and run.\n * \n * Can be set to a specific version or one of:\n * - \"latest\": Run the latest non-beta Obsidian version\n * - \"latest-beta\": Run the latest beta Obsidian version (or latest is there is no current beta)\n * - To download Obsidian beta versions you'll need to have an Obsidian account with Catalyst and set the \n * `OBSIDIAN_USERNAME` and `OBSIDIAN_PASSWORD` environment variables. 2FA needs to be disabled.\n * - \"earliest\": Run the `minAppVersion` set in set in your `manifest.json`\n * \n * Defaults to \"latest\".\n * \n * You can also use the wdio capability `browserVersion` field to set the Obsidian version.\n * \n * See also: [Obsidian App vs Installer Versions](../README.md#obsidian-app-vs-installer-versions)\n */\n appVersion?: string\n\n /**\n * Version of the Obsidian installer to download and run.\n * \n * Obsidian is distributed in two parts, the app which contains the JS, and the installer which is the binary with\n * electron. Obsidian's auto update only updates the app, so users on the same Obsidian version can be running\n * different Electron versions. You can use this to test your plugin against different installer/electron versions.\n * \n * See also: [Obsidian App vs Installer Versions](../README.md#obsidian-app-vs-installer-versions)\n *\n * Can be set to a specific version string or one of:\n * - \"latest\": Run the latest Obsidian installer.\n * - \"earliest\": Run the oldest Obsidian installer compatible with the specified Obsidian app version.\n * \n * Defaults to \"earliest\".\n */\n installerVersion?: string,\n\n /**\n * List of plugins to install.\n * \n * Each entry is a path to the local plugin to install, e.g. [\".\"] or [\"dist\"] depending on your build setup. You\n * can also pass objects. If you pass an object it can contain one of either `path` (to install a local plugin),\n * `repo` (to install a plugin from github), or `id` (to install a community plugin). You can set `enabled: false`\n * to install the plugin but start it disabled. You can enable the plugin later using `reloadObsidian` or the\n * `enablePlugin` command.\n */\n plugins?: PluginEntry[],\n\n /**\n * List of themes to install.\n * \n * Each entry is a path to the local theme to install. You can also pass an object. If you pass an object it can\n * contain one of either `path` (to install a local theme), `repo` (to install a theme from github), or `name` (to\n * install a community theme). You can set `enabled: false` to install the theme, but start it disabled. You can\n * only have one enabled theme, so if you pass multiple you'll have to disable all but one.\n */\n themes?: ThemeEntry[],\n\n /**\n * The path to the vault to open.\n * \n * The vault will be copied, so any changes made in your tests won't affect the original. If omitted, no vault will\n * be opened and you'll need to call `browser.reloadObsidian` to open a vault during your tests.\n */\n vault?: string,\n\n /**\n * Path to the Obsidian binary to use. If omitted it will be downloaded automatically.\n */\n binaryPath?: string,\n\n /**\n * Path to the app asar to load into obsidian. If omitted it will be downloaded automatically.\n */\n appPath?: string,\n}\n\n\ndeclare global {\n namespace WebdriverIO {\n interface Capabilities {\n [OBSIDIAN_CAPABILITY_KEY]?: ObsidianCapabilityOptions,\n }\n\n interface Browser extends ObsidianBrowserCommands {}\n }\n}\n","import * as path from \"path\"\nimport * as fs from \"fs\"\nimport * as fsAsync from \"fs/promises\"\nimport { OBSIDIAN_CAPABILITY_KEY } from \"../types.js\";\nimport { TFile } from \"obsidian\";\nimport _ from \"lodash\";\n\n/**\n * Class with various helper methods for writing Obsidian tests using the\n * [page object pattern](https://webdriver.io/docs/pageobjects).\n * \n * You can get an instance of this class either by running\n * ```ts\n * const obsidianPage = await browser.getObsidianPage();\n * ```\n * or just importing it directly with\n * ```ts\n * import { obsidianPage } from \"wdio-obsidian-service\";\n * ```\n * \n * @hideconstructor\n */\nclass ObsidianPage {\n /**\n * Returns the path to the vault opened in Obsidian.\n * \n * wdio-obsidian-service copies your vault before running tests, so this is the path to the temporary copy.\n */\n async getVaultPath(): Promise<string|undefined> {\n if (browser.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].vault == undefined) {\n return undefined; // no vault open\n } else { // return the actual path to the vault\n return await browser.executeObsidian(({app, obsidian}) => {\n if (app.vault.adapter instanceof obsidian.FileSystemAdapter) {\n return app.vault.adapter.getBasePath()\n } else { // TODO handle CapacitorAdapater\n throw new Error(`Unrecognized DataAdapater type`)\n };\n })\n }\n }\n\n /**\n * Return the path to the Obsidian config dir (\".obsidian\" unless you've changed it explicitly)\n */\n async getConfigDir(): Promise<string> {\n return await browser.executeObsidian(({app}) => {\n return app.vault.configDir;\n })\n }\n\n /**\n * Enables a plugin by ID\n */\n async enablePlugin(pluginId: string): Promise<void> {\n await browser.executeObsidian(\n async ({app}, pluginId) => await (app as any).plugins.enablePluginAndSave(pluginId),\n pluginId,\n );\n }\n\n /**\n * Disables a plugin by ID\n */\n async disablePlugin(pluginId: string): Promise<void> {\n await browser.executeObsidian(\n async ({app}, pluginId) => await (app as any).plugins.disablePluginAndSave(pluginId),\n pluginId,\n );\n }\n\n /**\n * Sets the theme. Pass \"default\" to reset to the Obsidian theme.\n */\n async setTheme(themeName: string): Promise<void> {\n themeName = themeName == 'default' ? '' : themeName;\n await browser.executeObsidian(\n async ({app}, themeName) => await (app as any).customCss.setTheme(themeName),\n themeName,\n )\n }\n\n /**\n * Opens a file in a new tab.\n */\n async openFile(path: string) {\n await browser.executeObsidian(async ({app, obsidian}, path) => {\n const file = app.vault.getAbstractFileByPath(path);\n if (file instanceof obsidian.TFile) {\n await app.workspace.getLeaf('tab').openFile(file);\n } else {\n throw Error(`No file ${path} exists`);\n }\n }, path)\n }\n\n /**\n * Loads a saved workspace layout from `.obsidian/workspaces.json` by name. Use the core \"Workspaces\"\n * plugin to create the layouts. You can also pass the layout object directly.\n */\n async loadWorkspaceLayout(layout: any): Promise<void> {\n if (typeof layout == \"string\") {\n // read from .obsidian/workspaces.json like the built-in workspaces plugin does\n const vaultPath = (await this.getVaultPath())!;\n const configDir = await this.getConfigDir();\n const workspacesPath = path.join(vaultPath, configDir, 'workspaces.json');\n const layoutName = layout;\n try {\n const fileContent = await fsAsync.readFile(workspacesPath, 'utf-8');\n layout = JSON.parse(fileContent)?.workspaces?.[layoutName];\n } catch {\n throw new Error(`No workspace ${layoutName} found in ${configDir}/workspaces.json`);\n }\n }\n\n await browser.executeObsidian(async ({app}, layout) => {\n await app.workspace.changeLayout(layout)\n }, layout)\n }\n\n /**\n * Resets the vault files to the original state by deleting/creating/modifying vault files in place without\n * reloading Obsidian.\n * \n * This will only reset regular vault files, it won't touch anything under `.obsidian`, and it won't reset any\n * config and app state you've set in Obsidian. But if all you need is to reset the vault files, this can be used as\n * a faster alternative to reloadObsidian.\n * \n * If no vault is passed, it resets the vault back to the oringal vault opened by the tests. You can also pass a\n * path to a different vault, and it will sync the current vault to match that one (similar to \"rsync\"). Or,\n * instead of passing a vault path you can pass an object mapping vault file paths to file content. E.g.\n * ```ts\n * obsidianPage.resetVault({\n * 'path/in/vault.md': \"Hello World\",\n * })\n * ```\n * \n * You can also pass multiple vaults and the files will be merged. This can be useful if you want to add a few small\n * modifications to the base vault. e.g:\n * ```ts\n * obsidianPage.resetVault('./path/to/vault', {\n * \"books/leviathan-wakes.md\": \"...\",\n * })\n * ```\n */\n async resetVault(...vaults: (string|Record<string, string>)[]) {\n const origVaultPath: string = browser.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].vault;\n vaults = vaults.length == 0 ? [origVaultPath] : vaults;\n const configDir = await this.getConfigDir();\n\n async function readVaultFiles(vault: string): Promise<Map<string, fs.Stats>> {\n const files = await fsAsync.readdir(vault, { recursive: true, withFileTypes: true });\n const paths = files\n .filter(f => f.isFile())\n .map(f => path.relative(vault, path.join(f.parentPath, f.name)).split(path.sep).join(\"/\"))\n .filter(f => !f.startsWith(configDir + \"/\"));\n const promises = paths.map(async (p) => [p, await fsAsync.stat(path.join(vault, p))] as const);\n return new Map(await Promise.all(promises));\n }\n\n function getFolders(files: Iterable<string>): Set<string> {\n return new Set([...files].map(p => path.dirname(p)).filter(p => p != '.'));\n }\n\n // merge the vaults\n const newFiles: Map<string, {stat?: fs.Stats, sourcePath?: string, sourceContent?: string}> = new Map();\n for (let vault of vaults) {\n if (typeof vault == \"string\") {\n vault = path.resolve(vault);\n for (const [file, stat] of await readVaultFiles(vault)) {\n newFiles.set(file, {stat, sourcePath: path.join(vault, file)});\n }\n } else {\n for (const [file, sourceContent] of Object.entries(vault)) {\n newFiles.set(file, {sourceContent});\n }\n }\n }\n\n // calculate the changes needed to the current vault\n const newFolders = getFolders(newFiles.keys());\n const currFiles = await readVaultFiles((await obsidianPage.getVaultPath())!);\n const currFolders = getFolders(currFiles.keys());\n\n type FileUpdateInstruction = {\n action: string, path: string,\n sourcePath?: string, sourceContent?: string,\n };\n const instructions: FileUpdateInstruction[] = [];\n\n // delete files\n for (const currFile of currFiles.keys()) {\n if (!newFiles.has(currFile)) {\n instructions.push({action: \"delete-file\", path: currFile});\n }\n }\n // delete folders, sort so children are before parents\n for (const currFolder of [...currFolders].sort().reverse()) {\n if (!newFolders.has(currFolder)) {\n instructions.push({action: \"delete-folder\", path: currFolder});\n }\n }\n // create folders, sort so parents are before children\n for (const newFolder of [...newFolders].sort()) {\n if (!currFolders.has(newFolder)) {\n instructions.push({action: \"create-folder\", path: newFolder});\n }\n }\n // create/modify files\n for (let [newFile, newFileInfo] of newFiles.entries()) {\n const {stat: newStat, sourcePath, sourceContent} = newFileInfo;\n const args = {path: newFile, sourcePath, sourceContent};\n const currStat = currFiles.get(newFile);\n if (!currStat) {\n instructions.push({action: \"create-file\", ...args});\n } else if ( // check if file has changed (setupVault preserves mtimes)\n !newStat ||\n currStat.mtime.getTime() != newStat.mtime.getTime() ||\n currStat.size != newStat.size\n ) {\n instructions.push({action: \"modify-file\", ...args});\n }\n }\n\n await browser.executeObsidian(async ({app, require}, instructions) => {\n // the require is getting transpiled by tsup, so use it from args instead of globally\n const fs = require('fs');\n \n for (const {action, path, sourcePath, sourceContent} of instructions) {\n const isHidden = path.split(\"/\").some(p => p.startsWith(\".\"));\n if (action == \"delete-file\") {\n if (isHidden) {\n await app.vault.adapter.remove(path);\n } else {\n await app.vault.delete(app.vault.getAbstractFileByPath(path)!);\n }\n } else if (action == \"delete-folder\") {\n if (isHidden) {\n await app.vault.adapter.rmdir(path, true);\n } else {\n await app.vault.delete(app.vault.getAbstractFileByPath(path)!, true);\n }\n } else if (action == \"create-folder\") {\n if (isHidden) {\n await app.vault.adapter.mkdir(path);\n } else {\n await app.vault.createFolder(path);\n }\n } else if (action == \"create-file\" || action == \"modify-file\") {\n const content = sourceContent ?? await fs.readFileSync(sourcePath!, 'utf-8');\n if (isHidden) {\n await app.vault.adapter.write(path, content);\n } else if (action == \"modify-file\") {\n await app.vault.modify(app.vault.getAbstractFileByPath(path) as TFile, content);\n } else { // action == \"create-file\"\n await app.vault.create(path, content);\n }\n } else {\n throw Error(`Unknown action ${action}`)\n }\n }\n }, instructions);\n }\n}\n\n/**\n * Instance of ObsidianPage with helper methods for writing Obsidian tests\n */\nconst obsidianPage = new ObsidianPage()\nexport default obsidianPage;\nexport { ObsidianPage };\n","import { OBSIDIAN_CAPABILITY_KEY } from \"./types.js\";\nimport type * as obsidian from \"obsidian\"\nimport obsidianPage, { ObsidianPage } from \"./pageobjects/obsidianPage.js\"\n\n/** Installed plugins, mapped by their id converted to camelCase */\nexport interface InstalledPlugins extends Record<string, obsidian.Plugin> {\n}\n\n/**\n * Argument passed to the `executeObsidian` browser command.\n */\nexport interface ExecuteObsidianArg {\n /**\n * There is a global \"app\" instance, but that may be removed in the future so you can use this to access it from\n * tests. See https://docs.obsidian.md/Plugins/Releasing/Plugin+guidelines#Avoid+using+global+app+instance\n */\n app: obsidian.App,\n\n /**\n * The full obsidian API. See https://github.com/obsidianmd/obsidian-api/blob/master/obsidian.d.ts\n */\n obsidian: typeof obsidian,\n\n /**\n * Object containing all installed plugins mapped by their id. Plugin ids are converted to converted to camelCase\n * for ease of destructuring.\n * \n * You can add types for your plugin(s) here with:\n * ```ts\n * import type MyPlugin from \"../src/main.js\"\n * declare module \"wdio-obsidian-service\" {\n * interface InstalledPlugins {\n * myPlugin: MyPlugin,\n * }\n * }\n * ```\n */\n plugins: InstalledPlugins,\n\n require: NodeJS.Require,\n}\n\nconst browserCommands = {\n /**\n * Returns the Obsidian app version this test is running under.\n */\n async getObsidianVersion(this: WebdriverIO.Browser): Promise<string> {\n return this.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].appVersion;\n },\n\n /**\n * Returns the Obsidian installer version this test is running under.\n */\n async getObsidianInstallerVersion(this: WebdriverIO.Browser): Promise<string> {\n return this.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].installerVersion;\n },\n\n /**\n * Wrapper around browser.execute that passes the Obsidian API to the function. The first argument to the function\n * is an object containing keys:\n * - app: Obsidian app instance\n * - obsidian: Full Obsidian API\n * - plugins: Object of all installed plugins, mapped by plugin id converted to camelCase.\n * - require: The customized require function Obsidian makes available to plugins. This is also made available\n * globally, so you can just use `require` directly instead of from ExecuteObsidianArg if you prefer.\n * \n * Like `brower.execute`, you can pass other extra arguments to the function.\n * \n * See also: https://webdriver.io/docs/api/browser/execute\n * \n * Example usage\n * ```ts\n * const file = browser.executeObsidian(({app, obsidian}, path) => {\n * return app.vault.getMarkdownFiles().find(f => f.path == path);\n * })\n * ```\n * \n * Note: The same caveats as `browser.execute` apply. The function will be stringified and then run inside Obsidian,\n * so you can't capture any local variables. E.g.\n * \n * This *won't* work:\n * ```ts\n * import { FileView } from Obsidian\n * browser.executeObsidian(({app, obsidian}) => {\n * if (leaf.view instanceof FileView) {\n * ...\n * }\n * })\n * ```\n * do this instead:\n * ```ts\n * browser.executeObsidian(({app, obsidian}) => {\n * if (leaf.view instanceof obsidian.FileView) {\n * ...\n * }\n * })\n * ```\n */\n async executeObsidian<Return, Params extends unknown[]>(\n this: WebdriverIO.Browser,\n func: (obs: ExecuteObsidianArg, ...params: Params) => Return,\n ...params: Params\n ): Promise<Return> {\n if (this.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].vault == undefined) {\n throw Error(\"No vault open\")\n }\n return await this.execute<Return, Params>(`\n const require = window.wdioObsidianService.require;\n return (${func.toString()}).call(null, {...window.wdioObsidianService}, ...arguments)\n `, ...params)\n },\n\n /**\n * Executes an Obsidian command by id.\n * @param id Id of the command to run.\n */\n async executeObsidianCommand(this: WebdriverIO.Browser, id: string) {\n const result = await this.executeObsidian(({app}, id) => (app as any).commands.executeCommandById(id), id);\n if (!result) {\n throw Error(`Obsidian command ${id} not found or failed.`);\n }\n },\n\n /**\n * Returns the Workspace page object with convenience helper functions.\n * You can also just import the page object directly with\n * ```ts\n * import { obsidianPage } from \"wdio-obsidian-service\"\n * ```\n */\n async getObsidianPage(this: WebdriverIO.Browser): Promise<ObsidianPage> {\n return obsidianPage;\n },\n} as const\n\n/** Define this type separately so we can @inline it in typedoc */\ntype PlainObsidianBrowserCommands = typeof browserCommands;\n\n/**\n * Extra commands added to the WDIO Browser instance.\n * \n * See also: https://webdriver.io/docs/api/browser#custom-commands\n * @interface\n */\nexport type ObsidianBrowserCommands = PlainObsidianBrowserCommands & {\n // This command is implemented in the service hooks.\n /**\n * Relaunch obsidian. Can be used to switch to a new vault, change the plugin list, or just to reboot\n * Obsidian.\n * \n * As this does a full reboot of Obsidian, avoid calling this too often so you don't slow your tests down.\n * You can also set the vault in the `wdio.conf.(m)ts` capabilities section which may be useful if all your\n * tests use the same vault.\n * \n * @param params.vault Path to the vault to open. The vault will be copied, so any changes made in your tests won't\n * be persited to the original. If omitted, it will reboot Obsidian with the current vault, without\n * creating a new copy of the vault.\n * @param params.plugins List of plugin ids to enable. If omitted it will keep current plugin list. Note, all the\n * plugins must be defined in your wdio.conf.ts capabilities. You can also use the enablePlugin and \n * disablePlugin commands to change plugins without relaunching Obsidian.\n * @param params.theme Name of the theme to enable. If omitted it will keep the current theme. Pass \"default\" to\n * switch back to the default theme. Like with plugins, the theme must be defined in wdio.conf.ts.\n * @returns Returns the new sessionId (same as browser.reloadSession()).\n */\n reloadObsidian(params?: {\n vault?: string,\n plugins?: string[], theme?: string,\n }): Promise<string>;\n};\nexport default browserCommands;\n","export async function sleep(ms: number): Promise<void> {\n return new Promise(resolve => setTimeout(resolve, ms));\n}\n"],"mappings":"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","names":["ObsidianLauncher","fsAsync","path","pluginId","themeName","path","layout","stat","require","instructions","fs","id","resolve","path","browser","resolve","fsAsync","launcher","ObsidianLauncher"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/service.ts","../src/types.ts","../src/pageobjects/obsidianPage.ts","../src/browserCommands.ts","../src/utils.ts"],"sourcesContent":["/**\n * @module\n * @document ../README.md\n */\nimport ObsidianLauncher from \"obsidian-launcher\";\nimport { ObsidianLauncherService, ObsidianWorkerService } from \"./service.js\";\n/** @hidden */\nexport default ObsidianWorkerService;\n/** @hidden */\nexport const launcher = ObsidianLauncherService;\n\nexport type { ObsidianCapabilityOptions, ObsidianServiceOptions } from \"./types.js\";\nexport type { ObsidianBrowserCommands, ExecuteObsidianArg, InstalledPlugins } from \"./browserCommands.js\";\nexport { default as obsidianPage } from \"./pageobjects/obsidianPage.js\";\nexport type { ObsidianPage } from \"./pageobjects/obsidianPage.js\";\nexport type { PluginEntry, DownloadedPluginEntry, ThemeEntry, DownloadedThemeEntry } from \"obsidian-launcher\";\n\nexport { minSupportedObsidianVersion } from \"./service.js\";\n\n// Some convenience helpers for use in wdio.conf.(m)ts\n\n/**\n * Returns true if there is a current Obsidian beta and we have the credentials to download it, or its already in cache.\n * @param cacheDir Obsidian cache dir, defaults to `.obsidian-cache`.\n */\nexport async function obsidianBetaAvailable(cacheDir?: string) {\n const launcher = new ObsidianLauncher({cacheDir: cacheDir});\n const versionInfo = await launcher.getVersionInfo(\"latest-beta\");\n return versionInfo.isBeta && await launcher.isAvailable(versionInfo.version);\n}\n\n/**\n * Resolves Obsidian app and installer version strings to absolute versions.\n * @param appVersion Obsidian version string or one of \n * - \"latest\": Get the current latest non-beta Obsidian version\n * - \"latest-beta\": Get the current latest beta Obsidian version (or latest is there is no current beta)\n * - \"earliest\": Get the `minAppVersion` set in your `manifest.json`\n * @param installerVersion Obsidian version string or one of \n * - \"latest\": Get the latest Obsidian installer compatible with `appVersion`\n * - \"earliest\": Get the oldest Obsidian installer compatible with `appVersion`\n * @param cacheDir Obsidian cache dir, defaults to `.obsidian-cache`.\n * \n * See also: [Obsidian App vs Installer Versions](../README.md#obsidian-app-vs-installer-versions)\n *\n * @returns [appVersion, installerVersion] with any \"latest\" etc. resolved to specific versions.\n */\nexport async function resolveObsidianVersions(\n appVersion: string, installerVersion: string, cacheDir?: string,\n): Promise<[string, string]> {\n const launcher = new ObsidianLauncher({cacheDir: cacheDir});\n return await launcher.resolveVersions(appVersion, installerVersion);\n}\n","import fs from \"fs\"\nimport fsAsync from \"fs/promises\"\nimport path from \"path\"\nimport { SevereServiceError } from 'webdriverio'\nimport type { Capabilities, Options, Services } from '@wdio/types'\nimport logger from '@wdio/logger'\nimport { fileURLToPath } from \"url\"\nimport ObsidianLauncher, { DownloadedPluginEntry, DownloadedThemeEntry } from \"obsidian-launcher\"\nimport browserCommands from \"./browserCommands.js\"\nimport { ObsidianCapabilityOptions, ObsidianServiceOptions, OBSIDIAN_CAPABILITY_KEY } from \"./types.js\"\nimport obsidianPage from \"./pageobjects/obsidianPage.js\"\nimport { sleep } from \"./utils.js\"\nimport semver from \"semver\"\nimport _ from \"lodash\"\n\n\nconst log = logger(\"wdio-obsidian-service\");\n\nfunction getDefaultCacheDir() {\n return path.resolve(process.env.WEBDRIVER_CACHE_DIR ?? process.env.OBSIDIAN_CACHE ?? \"./.obsidian-cache\")\n}\n\n/**\n * Minimum Obsidian version that wdio-obsidian-service supports.\n */\nexport const minSupportedObsidianVersion: string = \"1.0.3\"\n\n\n/**\n * wdio launcher service.\n * Use in wdio.conf.ts like so:\n * ```ts\n * services: ['obsidian'],\n * ```\n * @hidden\n */\nexport class ObsidianLauncherService implements Services.ServiceInstance {\n private obsidianLauncher: ObsidianLauncher\n private readonly helperPluginPath: string\n\n constructor (\n public options: ObsidianServiceOptions,\n public capabilities: WebdriverIO.Capabilities,\n public config: Options.Testrunner\n ) {\n this.obsidianLauncher = new ObsidianLauncher({\n cacheDir: config.cacheDir ?? getDefaultCacheDir(),\n versionsUrl: options.versionsUrl,\n communityPluginsUrl: options.communityPluginsUrl, communityThemesUrl: options.communityThemesUrl,\n });\n this.helperPluginPath = path.resolve(path.join(fileURLToPath(import.meta.url), '../../helper-plugin'));\n }\n\n /**\n * Validates wdio:obsidianOptions and downloads Obsidian, plugins, and themes.\n */\n async onPrepare(config: Options.Testrunner, capabilities: Capabilities.TestrunnerCapabilities) {\n if (!Array.isArray(capabilities)) {\n capabilities = Object.values(capabilities as Capabilities.RequestedMultiremoteCapabilities).map(\n (multiremoteOption) => (multiremoteOption as Capabilities.WithRequestedCapabilities).capabilities,\n );\n }\n\n const obsidianCapabilities = capabilities.flatMap((cap) => {\n if ((\"browserName\" in cap) && cap.browserName === \"obsidian\") {\n return [cap as WebdriverIO.Capabilities];\n } else {\n return [];\n }\n });\n\n try {\n for (const cap of obsidianCapabilities) {\n const obsidianOptions = cap[OBSIDIAN_CAPABILITY_KEY] ?? {};\n \n const vault = obsidianOptions.vault != undefined ? path.resolve(obsidianOptions.vault) : undefined;\n \n const [appVersion, installerVersion] = await this.obsidianLauncher.resolveVersions(\n cap.browserVersion ?? cap[OBSIDIAN_CAPABILITY_KEY]?.appVersion ?? \"latest\",\n obsidianOptions.installerVersion ?? \"earliest\",\n );\n const installerVersionInfo = await this.obsidianLauncher.getVersionInfo(installerVersion);\n if (semver.lt(appVersion, minSupportedObsidianVersion)) {\n throw Error(`Minimum supported Obsidian version is ${minSupportedObsidianVersion}`)\n }\n\n let installerPath = obsidianOptions.binaryPath;\n if (!installerPath) {\n installerPath = await this.obsidianLauncher.downloadInstaller(installerVersion);\n }\n let appPath = obsidianOptions.appPath;\n if (!appPath) {\n appPath = await this.obsidianLauncher.downloadApp(appVersion);\n }\n let chromedriverPath = cap['wdio:chromedriverOptions']?.binary\n // wdio can download chromedriver for versions greater than 115 automatically\n if (!chromedriverPath && Number(installerVersionInfo.chromeVersion!.split(\".\")[0]) <= 115) {\n chromedriverPath = await this.obsidianLauncher.downloadChromedriver(installerVersion);\n }\n\n let plugins = obsidianOptions.plugins ?? [\".\"];\n plugins.push(this.helperPluginPath); // Always install the helper plugin\n plugins = await this.obsidianLauncher.downloadPlugins(plugins);\n\n const themes = await this.obsidianLauncher.downloadThemes(obsidianOptions.themes ?? []);\n\n if (obsidianOptions.vault != undefined && !fs.existsSync(obsidianOptions.vault)) {\n throw Error(`Vault \"${obsidianOptions.vault}\" doesn't exist`)\n }\n\n const args = [\n // Workaround for SUID issue on AppImages. See https://github.com/electron/electron/issues/42510\n ...(process.platform == 'linux' ? [\"--no-sandbox\"] : []),\n ...(cap['goog:chromeOptions']?.args ?? [])\n ];\n\n cap.browserName = \"chrome\";\n cap.browserVersion = installerVersionInfo.chromeVersion;\n cap[OBSIDIAN_CAPABILITY_KEY] = {\n ...obsidianOptions,\n plugins: plugins,\n themes: themes,\n binaryPath: installerPath,\n appPath: appPath,\n vault: vault,\n appVersion: appVersion, // Resolve the versions\n installerVersion: installerVersion,\n };\n cap['goog:chromeOptions'] = {\n binary: installerPath,\n windowTypes: [\"app\", \"webview\"],\n ...cap['goog:chromeOptions'],\n args: args,\n }\n cap['wdio:chromedriverOptions'] = {\n // allowedIps is not included in the types, but gets passed as --allowed-ips to chromedriver.\n // It defaults to [\"0.0.0.0\"] which makes Windows Firewall complain, and we don't need remote\n // connections anyways.\n allowedIps: [],\n ...cap['wdio:chromedriverOptions'],\n binary: chromedriverPath,\n } as any\n cap[\"wdio:enforceWebDriverClassic\"] = true;\n }\n } catch (e: any) {\n // By default wdio just logs service errors, throwing this makes it bail if anything goes wrong.\n throw new SevereServiceError(`Failed to download and setup Obsidian. Caused by: ${e.stack}\\n`+\n ` ------The above causes:-----`);\n }\n }\n}\n\n\n/**\n * wdio worker service.\n * Use in wdio.conf.ts like so:\n * ```ts\n * services: ['obsidian'],\n * ```\n * @hidden\n */\nexport class ObsidianWorkerService implements Services.ServiceInstance {\n private obsidianLauncher: ObsidianLauncher\n /** Directories to clean up after the tests */\n private tmpDirs: string[]\n\n constructor (\n public options: ObsidianServiceOptions,\n public capabilities: WebdriverIO.Capabilities,\n public config: Options.Testrunner\n ) {\n this.obsidianLauncher = new ObsidianLauncher({\n cacheDir: config.cacheDir ?? getDefaultCacheDir(),\n versionsUrl: options.versionsUrl,\n communityPluginsUrl: options.communityPluginsUrl, communityThemesUrl: options.communityThemesUrl,\n });\n this.tmpDirs = [];\n }\n\n /**\n * Setup vault and config dir for a sandboxed Obsidian instance.\n */\n private async setupObsidian(obsidianOptions: ObsidianCapabilityOptions) {\n let vault = obsidianOptions.vault;\n if (vault != undefined) {\n log.info(`Opening vault ${obsidianOptions.vault}`);\n vault = await this.obsidianLauncher.setupVault({\n vault,\n copy: true,\n plugins: obsidianOptions.plugins,\n themes: obsidianOptions.themes,\n });\n this.tmpDirs.push(vault);\n } else {\n log.info(`Opening Obsidian without a vault`)\n }\n\n const configDir = await this.obsidianLauncher.setupConfigDir({\n appVersion: obsidianOptions.appVersion!, installerVersion: obsidianOptions.installerVersion!,\n appPath: obsidianOptions.appPath!,\n vault: vault,\n });\n this.tmpDirs.push(configDir);\n\n return configDir;\n }\n\n /**\n * Wait for Obsidian to fully boot.\n */\n private async waitForReady(browser: WebdriverIO.Browser) {\n if (browser.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].vault != undefined) {\n await browser.waitUntil( // wait until the helper plugin is loaded\n () => browser.execute(() => !!(window as any).wdioObsidianService),\n {timeout: 30 * 1000, interval: 100},\n );\n await browser.executeObsidian(async ({app}) => {\n await new Promise<void>((resolve) => app.workspace.onLayoutReady(resolve) );\n })\n } else {\n await browser.execute(async () => {\n if (document.readyState === \"loading\") {\n return new Promise<void>(resolve => document.addEventListener(\"DOMContentLoaded\", () => resolve()));\n }\n })\n }\n }\n\n /**\n * Handles vault and sandboxed config directory setup.\n */\n async beforeSession(config: Options.Testrunner, capabilities: WebdriverIO.Capabilities) {\n if (!capabilities[OBSIDIAN_CAPABILITY_KEY]) return;\n\n const configDir = await this.setupObsidian(capabilities[OBSIDIAN_CAPABILITY_KEY]);\n\n capabilities['goog:chromeOptions']!.args = [\n `--user-data-dir=${configDir}`,\n ...(capabilities['goog:chromeOptions']!.args ?? [])\n ];\n }\n\n /**\n * Returns a plugin list with only the selected plugin ids enabled.\n */\n private selectPlugins(currentPlugins: DownloadedPluginEntry[], selection?: string[]) {\n if (selection !== undefined) {\n const unknownPlugins = _.difference(selection, currentPlugins.map(p => p.id));\n if (unknownPlugins.length > 0) {\n throw Error(`Unknown plugin ids: ${unknownPlugins.join(', ')}`)\n }\n return currentPlugins.map(p => ({\n ...p,\n enabled: selection.includes(p.id) || p.id == \"wdio-obsidian-service-plugin\",\n }));\n } else {\n return currentPlugins;\n }\n }\n\n /**\n * Returns a theme list with only the selected theme enabled.\n */\n private selectThemes(currentThemes: DownloadedThemeEntry[], selection?: string) {\n if (selection !== undefined) {\n if (selection != \"default\" && currentThemes.every((t: any) => t.name != selection)) {\n throw Error(`Unknown theme: ${selection}`);\n }\n return currentThemes.map((t: any) => ({...t, enabled: selection != 'default' && t.name === selection}));\n } else {\n return currentThemes;\n }\n }\n\n /**\n * Setup custom browser commands.\n */\n async before(capabilities: WebdriverIO.Capabilities, specs: never, browser: WebdriverIO.Browser) {\n // There's a slow event listener link on the browser \"command\" event when you reloadSession that causes some\n // warnings. This will silence them. TODO: Make issue or PR to wdio to fix this.\n browser.setMaxListeners(1000);\n\n if (!capabilities[OBSIDIAN_CAPABILITY_KEY]) return;\n\n const service = this; // eslint-disable-line @typescript-eslint/no-this-alias\n const reloadObsidian: typeof browser['reloadObsidian'] = async function(\n this: WebdriverIO.Browser,\n {vault, plugins, theme} = {},\n ) {\n const oldObsidianOptions = this.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY];\n let newCapabilities: WebdriverIO.Capabilities\n\n if (vault) {\n const newObsidianOptions = {\n ...oldObsidianOptions,\n vault: path.resolve(vault),\n plugins: service.selectPlugins(oldObsidianOptions.plugins, plugins),\n themes: service.selectThemes(oldObsidianOptions.themes, theme),\n }\n \n const configDir = await service.setupObsidian(newObsidianOptions);\n \n const newArgs = [\n `--user-data-dir=${configDir}`,\n ...this.requestedCapabilities['goog:chromeOptions'].args.filter((arg: string) => {\n const match = arg.match(/^--user-data-dir=(.*)$/);\n return !match || !service.tmpDirs.includes(match[1]);\n }),\n ]\n \n newCapabilities = {\n [OBSIDIAN_CAPABILITY_KEY]: newObsidianOptions,\n 'goog:chromeOptions': {\n ...this.requestedCapabilities['goog:chromeOptions'],\n args: newArgs,\n },\n };\n } else {\n // preserve vault and config dir\n newCapabilities = {};\n // Since we aren't recreating the vault, we'll need to reset plugins and themes here if specified.\n if (plugins) {\n const enabledPlugins = await browser.executeObsidian(({app}) =>\n [...(app as any).plugins.enabledPlugins].sort()\n )\n for (const pluginId of _.difference(enabledPlugins, plugins, ['wdio-obsidian-service-plugin'])) {\n await obsidianPage.disablePlugin(pluginId);\n }\n for (const pluginId of _.difference(plugins, enabledPlugins)) {\n await obsidianPage.enablePlugin(pluginId);\n }\n }\n if (theme) {\n await obsidianPage.setTheme(theme);\n }\n // Obsidian debounces saves to the config dir, and so changes to configuration made in the tests may not\n // get saved to disk before the reboot. Here I manually trigger save for plugins and themes, but other\n // configurations might not get saved. I haven't found a better way to flush everything than just\n // waiting a bit. Wdio has ways to mock the clock, which might work, but it's only supported when using\n // BiDi, which I can't get working on Obsidian.\n await browser.executeObsidian(async ({app}) => await Promise.all([\n (app as any).plugins.saveConfig(),\n (app.vault as any).saveConfig(),\n ]))\n await sleep(2000);\n }\n\n const sessionId = await browser.reloadSession({\n // if browserName is set, reloadSession tries to restart the driver entirely, so unset those\n ..._.omit(this.requestedCapabilities, ['browserName', 'browserVersion']),\n ...newCapabilities,\n });\n await service.waitForReady(this);\n return sessionId;\n }\n\n await browser.addCommand(\"reloadObsidian\", reloadObsidian);\n\n for (const [name, cmd] of Object.entries(browserCommands)) {\n await browser.addCommand(name, cmd);\n }\n\n await service.waitForReady(browser);\n }\n\n /**\n * Cleanup\n */\n async afterSession() {\n for (const tmpDir of this.tmpDirs) {\n await fsAsync.rm(tmpDir, { recursive: true, force: true });\n }\n }\n}\n","import type { ObsidianBrowserCommands } from \"./browserCommands.js\";\nimport type { PluginEntry, ThemeEntry } from \"obsidian-launcher\";\n\nexport const OBSIDIAN_CAPABILITY_KEY = \"wdio:obsidianOptions\" as const;\n\nexport interface ObsidianCapabilityOptions {\n /**\n * Version of Obsidian to download and run.\n * \n * Can be set to a specific version or one of:\n * - \"latest\": Run the latest non-beta Obsidian version\n * - \"latest-beta\": Run the latest beta Obsidian version (or latest is there is no current beta)\n * - To download Obsidian beta versions you'll need to have an Obsidian account with Catalyst and set the \n * `OBSIDIAN_USERNAME` and `OBSIDIAN_PASSWORD` environment variables. 2FA needs to be disabled.\n * - \"earliest\": Run the `minAppVersion` set in your `manifest.json`\n * \n * Defaults to \"latest\".\n * \n * You can also use the wdio capability `browserVersion` field to set the Obsidian version.\n * \n * See also: [Obsidian App vs Installer Versions](../README.md#obsidian-app-vs-installer-versions)\n */\n appVersion?: string\n\n /**\n * Version of the Obsidian installer to download and run.\n * \n * Obsidian is distributed in two parts, the app which contains the JS, and the installer which is the binary with\n * electron. Obsidian's auto update only updates the app, so users on the same Obsidian version can be running\n * different Electron versions. You can use this to test your plugin against different installer/electron versions.\n * \n * See also: [Obsidian App vs Installer Versions](../README.md#obsidian-app-vs-installer-versions)\n *\n * Can be set to a specific version string or one of:\n * - \"latest\": Run the latest Obsidian installer compatible with `appVersion`.\n * - \"earliest\": Run the oldest Obsidian installer compatible with `appVersion`.\n * \n * Defaults to \"earliest\".\n */\n installerVersion?: string,\n\n /**\n * List of plugins to install.\n * \n * Each entry is a path to the local plugin to install, e.g. [\".\"] or [\"dist\"] depending on your build setup. You\n * can also pass objects. If you pass an object it can contain one of either `path` (to install a local plugin),\n * `repo` (to install a plugin from github), or `id` (to install a community plugin). You can set `enabled: false`\n * to install the plugin but start it disabled. You can enable the plugin later using `reloadObsidian` or the\n * `enablePlugin` command.\n */\n plugins?: PluginEntry[],\n\n /**\n * List of themes to install.\n * \n * Each entry is a path to the local theme to install. You can also pass an object. If you pass an object it can\n * contain one of either `path` (to install a local theme), `repo` (to install a theme from github), or `name` (to\n * install a community theme). You can set `enabled: false` to install the theme, but start it disabled. You can\n * only have one enabled theme, so if you pass multiple you'll have to disable all but one.\n */\n themes?: ThemeEntry[],\n\n /**\n * The path to the vault to open.\n * \n * The vault will be copied, so any changes made in your tests won't affect the original. If omitted, no vault will\n * be opened and you'll need to call `browser.reloadObsidian` to open a vault during your tests.\n */\n vault?: string,\n\n /**\n * Path to the Obsidian binary to use. If omitted it will be downloaded automatically.\n */\n binaryPath?: string,\n\n /**\n * Path to the app asar to load into obsidian. If omitted it will be downloaded automatically.\n */\n appPath?: string,\n}\n\n\nexport interface ObsidianServiceOptions {\n /**\n * Override the `obsidian-versions.json` used by the service. Can be a file URL.\n * This is only really useful for this package's own internal tests.\n */\n versionsUrl?: string,\n /**\n * Override the `community-plugins.json` used by the service. Can be a file URL.\n * This is only really useful for this package's own internal tests.\n */\n communityPluginsUrl?: string,\n /**\n * Override the `community-css-themes.json` used by the service. Can be a file URL.\n * This is only really useful for this package's own internal tests.\n */\n communityThemesUrl?: string,\n}\n\ndeclare global {\n namespace WebdriverIO {\n interface Capabilities {\n [OBSIDIAN_CAPABILITY_KEY]?: ObsidianCapabilityOptions,\n }\n\n interface Browser extends ObsidianBrowserCommands {}\n }\n}\n","import * as path from \"path\"\nimport * as fs from \"fs\"\nimport * as fsAsync from \"fs/promises\"\nimport { OBSIDIAN_CAPABILITY_KEY } from \"../types.js\";\nimport { TFile } from \"obsidian\";\nimport _ from \"lodash\";\n\n/**\n * Class with various helper methods for writing Obsidian tests using the\n * [page object pattern](https://webdriver.io/docs/pageobjects).\n * \n * You can get an instance of this class either by running\n * ```ts\n * const obsidianPage = await browser.getObsidianPage();\n * ```\n * or just importing it directly with\n * ```ts\n * import { obsidianPage } from \"wdio-obsidian-service\";\n * ```\n * \n * @hideconstructor\n */\nclass ObsidianPage {\n /**\n * Returns the path to the vault opened in Obsidian.\n * \n * wdio-obsidian-service copies your vault before running tests, so this is the path to the temporary copy.\n */\n async getVaultPath(): Promise<string|undefined> {\n if (browser.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].vault == undefined) {\n return undefined; // no vault open\n } else { // return the actual path to the vault\n return await browser.executeObsidian(({app, obsidian}) => {\n if (app.vault.adapter instanceof obsidian.FileSystemAdapter) {\n return app.vault.adapter.getBasePath()\n } else { // TODO handle CapacitorAdapater\n throw new Error(`Unrecognized DataAdapater type`)\n };\n })\n }\n }\n\n /**\n * Return the path to the Obsidian config dir (\".obsidian\" unless you've changed it explicitly)\n */\n async getConfigDir(): Promise<string> {\n return await browser.executeObsidian(({app}) => {\n return app.vault.configDir;\n })\n }\n\n /**\n * Enables a plugin by ID\n */\n async enablePlugin(pluginId: string): Promise<void> {\n await browser.executeObsidian(\n async ({app}, pluginId) => await (app as any).plugins.enablePluginAndSave(pluginId),\n pluginId,\n );\n }\n\n /**\n * Disables a plugin by ID\n */\n async disablePlugin(pluginId: string): Promise<void> {\n await browser.executeObsidian(\n async ({app}, pluginId) => await (app as any).plugins.disablePluginAndSave(pluginId),\n pluginId,\n );\n }\n\n /**\n * Sets the theme. Pass \"default\" to reset to the Obsidian theme.\n */\n async setTheme(themeName: string): Promise<void> {\n themeName = themeName == 'default' ? '' : themeName;\n await browser.executeObsidian(\n async ({app}, themeName) => await (app as any).customCss.setTheme(themeName),\n themeName,\n )\n }\n\n /**\n * Opens a file in a new tab.\n */\n async openFile(path: string) {\n await browser.executeObsidian(async ({app, obsidian}, path) => {\n const file = app.vault.getAbstractFileByPath(path);\n if (file instanceof obsidian.TFile) {\n await app.workspace.getLeaf('tab').openFile(file);\n } else {\n throw Error(`No file ${path} exists`);\n }\n }, path)\n }\n\n /**\n * Loads a saved workspace layout from `.obsidian/workspaces.json` by name. Use the core \"Workspaces\"\n * plugin to create the layouts. You can also pass the layout object directly.\n */\n async loadWorkspaceLayout(layout: any): Promise<void> {\n if (typeof layout == \"string\") {\n // read from .obsidian/workspaces.json like the built-in workspaces plugin does\n const vaultPath = (await this.getVaultPath())!;\n const configDir = await this.getConfigDir();\n const workspacesPath = path.join(vaultPath, configDir, 'workspaces.json');\n const layoutName = layout;\n try {\n const fileContent = await fsAsync.readFile(workspacesPath, 'utf-8');\n layout = JSON.parse(fileContent)?.workspaces?.[layoutName];\n } catch {\n throw new Error(`No workspace ${layoutName} found in ${configDir}/workspaces.json`);\n }\n }\n\n await browser.executeObsidian(async ({app}, layout) => {\n await app.workspace.changeLayout(layout)\n }, layout)\n }\n\n /**\n * Resets the vault files to the original state by deleting/creating/modifying vault files in place without\n * reloading Obsidian.\n * \n * This will only reset regular vault files, it won't touch anything under `.obsidian`, and it won't reset any\n * config and app state you've set in Obsidian. But if all you need is to reset the vault files, this can be used as\n * a faster alternative to reloadObsidian.\n * \n * If no vault is passed, it resets the vault back to the oringal vault opened by the tests. You can also pass a\n * path to a different vault, and it will sync the current vault to match that one (similar to \"rsync\"). Or,\n * instead of passing a vault path you can pass an object mapping vault file paths to file content. E.g.\n * ```ts\n * obsidianPage.resetVault({\n * 'path/in/vault.md': \"Hello World\",\n * })\n * ```\n * \n * You can also pass multiple vaults and the files will be merged. This can be useful if you want to add a few small\n * modifications to the base vault. e.g:\n * ```ts\n * obsidianPage.resetVault('./path/to/vault', {\n * \"books/leviathan-wakes.md\": \"...\",\n * })\n * ```\n */\n async resetVault(...vaults: (string|Record<string, string>)[]) {\n const origVaultPath: string = browser.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].vault;\n vaults = vaults.length == 0 ? [origVaultPath] : vaults;\n const configDir = await this.getConfigDir();\n\n async function readVaultFiles(vault: string): Promise<Map<string, fs.Stats>> {\n const files = await fsAsync.readdir(vault, { recursive: true, withFileTypes: true });\n const paths = files\n .filter(f => f.isFile())\n .map(f => path.relative(vault, path.join(f.parentPath, f.name)).split(path.sep).join(\"/\"))\n .filter(f => !f.startsWith(configDir + \"/\"));\n const promises = paths.map(async (p) => [p, await fsAsync.stat(path.join(vault, p))] as const);\n return new Map(await Promise.all(promises));\n }\n\n function getFolders(files: Iterable<string>): Set<string> {\n return new Set([...files].map(p => path.dirname(p)).filter(p => p != '.'));\n }\n\n // merge the vaults\n const newFiles: Map<string, {stat?: fs.Stats, sourcePath?: string, sourceContent?: string}> = new Map();\n for (let vault of vaults) {\n if (typeof vault == \"string\") {\n vault = path.resolve(vault);\n for (const [file, stat] of await readVaultFiles(vault)) {\n newFiles.set(file, {stat, sourcePath: path.join(vault, file)});\n }\n } else {\n for (const [file, sourceContent] of Object.entries(vault)) {\n newFiles.set(file, {sourceContent});\n }\n }\n }\n\n // calculate the changes needed to the current vault\n const newFolders = getFolders(newFiles.keys());\n const currFiles = await readVaultFiles((await obsidianPage.getVaultPath())!);\n const currFolders = getFolders(currFiles.keys());\n\n type FileUpdateInstruction = {\n action: string, path: string,\n sourcePath?: string, sourceContent?: string,\n };\n const instructions: FileUpdateInstruction[] = [];\n\n // delete files\n for (const currFile of currFiles.keys()) {\n if (!newFiles.has(currFile)) {\n instructions.push({action: \"delete-file\", path: currFile});\n }\n }\n // delete folders, sort so children are before parents\n for (const currFolder of [...currFolders].sort().reverse()) {\n if (!newFolders.has(currFolder)) {\n instructions.push({action: \"delete-folder\", path: currFolder});\n }\n }\n // create folders, sort so parents are before children\n for (const newFolder of [...newFolders].sort()) {\n if (!currFolders.has(newFolder)) {\n instructions.push({action: \"create-folder\", path: newFolder});\n }\n }\n // create/modify files\n for (let [newFile, newFileInfo] of newFiles.entries()) {\n const {stat: newStat, sourcePath, sourceContent} = newFileInfo;\n const args = {path: newFile, sourcePath, sourceContent};\n const currStat = currFiles.get(newFile);\n if (!currStat) {\n instructions.push({action: \"create-file\", ...args});\n } else if ( // check if file has changed (setupVault preserves mtimes)\n !newStat ||\n currStat.mtime.getTime() != newStat.mtime.getTime() ||\n currStat.size != newStat.size\n ) {\n instructions.push({action: \"modify-file\", ...args});\n }\n }\n\n await browser.executeObsidian(async ({app, require}, instructions) => {\n // the require is getting transpiled by tsup, so use it from args instead of globally\n const fs = require('fs');\n \n for (const {action, path, sourcePath, sourceContent} of instructions) {\n const isHidden = path.split(\"/\").some(p => p.startsWith(\".\"));\n if (action == \"delete-file\") {\n if (isHidden) {\n await app.vault.adapter.remove(path);\n } else {\n await app.vault.delete(app.vault.getAbstractFileByPath(path)!);\n }\n } else if (action == \"delete-folder\") {\n if (isHidden) {\n await app.vault.adapter.rmdir(path, true);\n } else {\n await app.vault.delete(app.vault.getAbstractFileByPath(path)!, true);\n }\n } else if (action == \"create-folder\") {\n if (isHidden) {\n await app.vault.adapter.mkdir(path);\n } else {\n await app.vault.createFolder(path);\n }\n } else if (action == \"create-file\" || action == \"modify-file\") {\n const content = sourceContent ?? await fs.readFileSync(sourcePath!, 'utf-8');\n if (isHidden) {\n await app.vault.adapter.write(path, content);\n } else if (action == \"modify-file\") {\n await app.vault.modify(app.vault.getAbstractFileByPath(path) as TFile, content);\n } else { // action == \"create-file\"\n await app.vault.create(path, content);\n }\n } else {\n throw Error(`Unknown action ${action}`)\n }\n }\n }, instructions);\n }\n}\n\n/**\n * Instance of {@link ObsidianPage} with helper methods for writing Obsidian tests.\n */\nconst obsidianPage = new ObsidianPage()\nexport default obsidianPage;\nexport { ObsidianPage };\n","import { OBSIDIAN_CAPABILITY_KEY } from \"./types.js\";\nimport type * as obsidian from \"obsidian\"\nimport obsidianPage, { ObsidianPage } from \"./pageobjects/obsidianPage.js\"\n\nconst browserCommands = {\n /**\n * Returns the Obsidian app version this test is running under.\n */\n async getObsidianVersion(this: WebdriverIO.Browser): Promise<string> {\n return this.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].appVersion;\n },\n\n /**\n * Returns the Obsidian installer version this test is running under.\n */\n async getObsidianInstallerVersion(this: WebdriverIO.Browser): Promise<string> {\n return this.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].installerVersion;\n },\n\n /**\n * Wrapper around browser.execute that passes the Obsidian API to the function. The first argument to the function\n * is an object containing keys:\n * - app: Obsidian app instance\n * - obsidian: Full Obsidian API\n * - plugins: Object of all installed plugins, mapped by plugin id converted to camelCase.\n * - require: The customized require function Obsidian makes available to plugins. This is also available globally,\n * so you can just use `require` directly instead of from {@link ExecuteObsidianArg} if you prefer.\n * \n * Like `brower.execute`, you can pass other extra arguments to the function.\n * \n * See also: https://webdriver.io/docs/api/browser/execute\n * \n * Example usage\n * ```ts\n * const file = browser.executeObsidian(({app, obsidian}, path) => {\n * return app.vault.getMarkdownFiles().find(f => f.path == path);\n * })\n * ```\n * \n * Note: The same caveats as `browser.execute` apply. The function will be stringified and then run inside Obsidian,\n * so you can't capture any local variables. E.g.\n * \n * This *won't* work:\n * ```ts\n * import { FileView } from Obsidian\n * browser.executeObsidian(({app, obsidian}) => {\n * if (leaf.view instanceof FileView) {\n * ...\n * }\n * })\n * ```\n * do this instead:\n * ```ts\n * browser.executeObsidian(({app, obsidian}) => {\n * if (leaf.view instanceof obsidian.FileView) {\n * ...\n * }\n * })\n * ```\n */\n async executeObsidian<Return, Params extends unknown[]>(\n this: WebdriverIO.Browser,\n func: (obs: ExecuteObsidianArg, ...params: Params) => Return,\n ...params: Params\n ): Promise<Return> {\n if (this.requestedCapabilities[OBSIDIAN_CAPABILITY_KEY].vault == undefined) {\n throw Error(\"No vault open\")\n }\n return await this.execute<Return, Params>(`\n const require = window.wdioObsidianService.require;\n return (${func.toString()}).call(null, {...window.wdioObsidianService}, ...arguments)\n `, ...params)\n },\n\n /**\n * Executes an Obsidian command by id.\n * @param id Id of the command to run.\n */\n async executeObsidianCommand(this: WebdriverIO.Browser, id: string) {\n const result = await this.executeObsidian(({app}, id) => (app as any).commands.executeCommandById(id), id);\n if (!result) {\n throw Error(`Obsidian command ${id} not found or failed.`);\n }\n },\n\n /**\n * Returns the Workspace page object with convenience helper functions.\n * You can also just import the page object directly with\n * ```ts\n * import { obsidianPage } from \"wdio-obsidian-service\"\n * ```\n */\n async getObsidianPage(this: WebdriverIO.Browser): Promise<ObsidianPage> {\n return obsidianPage;\n },\n} as const\n\n/** Define this type separately so we can @inline it in typedoc */\ntype PlainObsidianBrowserCommands = typeof browserCommands;\n\n/**\n * Extra commands added to the WDIO Browser instance.\n * \n * See also: https://webdriver.io/docs/api/browser\n * @interface\n */\nexport type ObsidianBrowserCommands = PlainObsidianBrowserCommands & {\n // This command is implemented in the service hooks.\n /**\n * Relaunch obsidian. Can be used to switch to a new vault, change the plugin list, or just to reboot\n * Obsidian.\n * \n * As this does a full reboot of Obsidian, avoid calling this too often so you don't slow your tests down.\n * You can also set the vault in the `wdio.conf.(m)ts` capabilities section which may be useful if all your\n * tests use the same vault.\n * \n * @param params.vault Path to the vault to open. The vault will be copied, so any changes made in your tests won't\n * be persited to the original. If omitted, it will reboot Obsidian with the current vault, without\n * creating a new copy of the vault.\n * @param params.plugins List of plugin ids to enable. If omitted it will keep current plugin list. Note, all the\n * plugins must be defined in your wdio.conf.ts capabilities. You can also use the enablePlugin and \n * disablePlugin commands to change plugins without relaunching Obsidian.\n * @param params.theme Name of the theme to enable. If omitted it will keep the current theme. Pass \"default\" to\n * switch back to the default theme. Like with plugins, the theme must be defined in wdio.conf.ts.\n * @returns Returns the new sessionId (same as browser.reloadSession()).\n */\n reloadObsidian(params?: {\n vault?: string,\n plugins?: string[], theme?: string,\n }): Promise<string>;\n};\n\n/**\n * Argument passed to the `executeObsidian` browser command.\n */\nexport interface ExecuteObsidianArg {\n /**\n * There is a global \"app\" instance, but that may be removed in the future so you can use this to access it from\n * tests. See https://docs.obsidian.md/Plugins/Releasing/Plugin+guidelines#Avoid+using+global+app+instance\n */\n app: obsidian.App,\n\n /**\n * The full obsidian API. See https://github.com/obsidianmd/obsidian-api/blob/master/obsidian.d.ts\n */\n obsidian: typeof obsidian,\n\n /**\n * Object containing all installed plugins mapped by their id. Plugin ids are converted to camelCase for ease of\n * destructuring.\n * \n * If you want to add typings for your plugin(s) you can use something like this in a `.d.ts`:\n * ```ts\n * import type MyPlugin from \"../src/main.js\"\n * declare module \"wdio-obsidian-service\" {\n * interface InstalledPlugins {\n * myPlugin: MyPlugin,\n * }\n * }\n * ```\n */\n plugins: InstalledPlugins,\n\n /**\n * The customized require function Obsidian makes available to plugins. This is also available globally, so you can\n * just use `require` directly instead of from `ExecuteObsidianArg` if you prefer.\n */\n require: NodeJS.Require,\n}\n\n/** Installed plugins, mapped by their id converted to camelCase */\nexport interface InstalledPlugins extends Record<string, obsidian.Plugin> {\n}\n\nexport default browserCommands;\n","export async function sleep(ms: number): Promise<void> {\n return new Promise(resolve => setTimeout(resolve, ms));\n}\n"],"mappings":"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","names":["ObsidianLauncher","fsAsync","path","pluginId","themeName","path","layout","stat","require","instructions","fs","id","resolve","path","browser","resolve","fsAsync","launcher","ObsidianLauncher"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "wdio-obsidian-service",
3
- "version": "1.1.0",
3
+ "version": "1.2.0",
4
4
  "description": "A WebdriverIO service for end-to-end testing of Obsidian plugins",
5
5
  "keywords": [
6
6
  "obsidian",
@@ -53,11 +53,11 @@
53
53
  "tsup": "^8.3.5",
54
54
  "tsx": "^4.19.2",
55
55
  "typescript": "^5.8.2",
56
- "wdio-obsidian-reporter": "^1.1.0"
56
+ "wdio-obsidian-reporter": "^1.2.0"
57
57
  },
58
58
  "dependencies": {
59
59
  "lodash": "^4.17.21",
60
- "obsidian-launcher": "^1.1.0",
60
+ "obsidian-launcher": "^1.2.0",
61
61
  "semver": "^7.7.1"
62
62
  },
63
63
  "peerDependencies": {