src/strategy/unnamed-version-string.ts
import { Effect, Option, pipe } from 'effect';
import { isNonEmptyObject } from 'tightrope/guard/is-non-empty-object.js';
import type { PackageJsonFile } from '../get-package-json-files/package-json-file.js';
import { get } from '../lib/get.js';
import type { Delete } from '../version-group/lib/delete.js';
import { DELETE } from '../version-group/lib/delete.js';
import { getNonEmptyStringProp } from './lib/get-non-empty-string-prop.js';
const getOptionOfNonEmptyObject = Option.liftPredicate(isNonEmptyObject<any>);
export class UnnamedVersionStringStrategy {
_tag = 'version';
name: string;
path: string;
constructor(name: string, path: string) {
this.name = name;
this.path = path;
}
read(file: PackageJsonFile): Effect.Effect<[string, string][]> {
return pipe(
// get version prop
getNonEmptyStringProp(this.path, file),
// if it is a non empty string, we can read it
Effect.map((version): [string, string][] => {
const name = this.path.split('.').slice(-1).join('');
return [[name, version]];
}),
Effect.tapError(() =>
Effect.logDebug(
`UnnamedVersionStringStrategy#${this.name} found nothing at <${file.jsonFile.shortPath}>.${this.path}`,
),
),
// if value is invalid, default to empty
Effect.catchAll(() => Effect.succeed([])),
);
}
write(
file: PackageJsonFile,
[, version]: [string, string | Delete],
): Effect.Effect<PackageJsonFile> {
const { contents } = file.jsonFile;
const isNestedPath = this.path.includes('.');
const nextValue = version === DELETE ? undefined : version;
if (isNestedPath) {
const fullPath = this.path.split('.');
const pathToParent = fullPath.slice(0, fullPath.length - 1).join('.');
const key = fullPath.slice(-1).join('');
return pipe(
get(contents, ...pathToParent.split('.')),
Effect.flatMap(getOptionOfNonEmptyObject),
Effect.flatMap((parent) =>
Effect.try(() => {
parent[key] = nextValue;
}),
),
Effect.tapError(() =>
Effect.logDebug(
`strategy ${this._tag} with name ${this.name} failed to write to <${file.jsonFile.shortPath}>.${this.path}`,
),
),
Effect.catchAll(() => Effect.succeed(file)),
Effect.map(() => file),
);
} else {
return pipe(
getOptionOfNonEmptyObject(contents),
Effect.flatMap((parent) =>
Effect.try(() => {
parent[this.path] = nextValue;
}),
),
Effect.tapError(() =>
Effect.logDebug(
`strategy ${this._tag} with name ${this.name} failed to write to <${file.jsonFile.shortPath}>.${this.path}`,
),
),
Effect.catchAll(() => Effect.succeed(file)),
Effect.map(() => file),
);
}
}
}