Skip to content

Commit 419bc9b

Browse files
committed
Add rhsRoot defaulting logic for the "sunny day" scenario.
1 parent 2c25bd6 commit 419bc9b

File tree

5 files changed

+68
-71
lines changed

5 files changed

+68
-71
lines changed

CHANGELOG.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
# @azure-tools/rest-api-diff
22

3-
## 0.2.0 (TBD)
3+
## 0.2.0 (2025-03-11)
44

55
- Fixed issue where relative references would sometimes be resolved incorrectly. `--lhs-root`
66
and `--rhs-root` are still needed when compiling TypeSpec. See README.md.

README.md

+4-2
Original file line numberDiff line numberDiff line change
@@ -116,13 +116,15 @@ debugging purposes.
116116

117117
## Running against the REST API Specs Repo
118118

119-
These steps assumed that `--lhs` points to a Swagger folder and `--rhs` points to a TypeSpec folder, both within the REST API specs repo. If this is not the case, your steps will differ.
119+
These steps assume that `--lhs` points to a Swagger folder and `--rhs` points to a TypeSpec folder, both within the REST API specs repo. If this is not the case, your steps will differ.
120120

121121
1. Ensure you have updated the dependencies in your fork by running `npm install` in the REST API specs repo root. You may need to delete `package-lock.json` first. Copy the path to the `node_modules/@typespec/compiler` package.
122122
2. Set the `TYPESPEC_COMPILER_PATH` environment variable (ideally in .env) to the path you copied in step 1.
123123
3. Ensure that LHS and RHS point to the appropriate paths in the REST API specs repo.
124124
4. By convention, if you are comparing hand-written Swagger to TypeSpec, the Swagger should be LHS and the TypeSpec should be RHS. When compiling TypeSpec, you will
125-
need to set RHS_ROOT. It will generally be the same path as LHS, but with the "stable" or "preview" segment replaced with "generated".
125+
need to set RHS_ROOT. **If you are following the convention, the tool will automatically use the same folder as LHS for RHS_ROOT except changing the "stable" or "preview"
126+
folder to "temp", which will subsequently be deleted after the tool completes.** This is so the relative paths get generated and resolve correctly without bulldozing
127+
and existing files.
126128
5. If you are comparing to a multi-versioned TypeSpec, you should probably include the `TYPESPEC_VERSION_SELECTOR` environment variable to ensure you are generating the right version for comparison.
127129

128130
## Rules

pnpm-lock.yaml

+2-65
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/definitions.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -181,7 +181,7 @@ export class DefinitionRegistry {
181181
const expVal = this.#expand(val);
182182
expanded.push(expVal);
183183
}
184-
return expanded.toSorted();
184+
return toSorted(expanded);
185185
}
186186

187187
#expandDerivedClasses(base: any): any {

src/diff-client.ts

+60-2
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ export interface DiffClientConfig {
2121
}
2222

2323
export class DiffClient {
24-
private args: any;
24+
public args: any;
2525
private rules: RuleSignature[];
2626
private lhsParser?: SwaggerParser;
2727
private rhsParser?: SwaggerParser;
@@ -38,6 +38,8 @@ export class DiffClient {
3838
public diffResults?: DiffResult;
3939
/** The output files generated by the diff operation. Available once buildOutput() called. */
4040
public resultFiles?: ResultFiles;
41+
/** The temporary root folder for TypeSpec-generated Swagger. */
42+
private tempRhsRoot?: string;
4143

4244
/** Creates an instance of the DiffClient class asynchronously. */
4345
static async create(config: DiffClientConfig): Promise<DiffClient> {
@@ -47,9 +49,18 @@ export class DiffClient {
4749
const lhsRoot = client.args["lhs-root"]
4850
? path.resolve(client.args["lhs-root"])
4951
: undefined;
50-
const rhsRoot = client.args["rhs-root"]
52+
53+
let rhsRoot = client.args["rhs-root"]
5154
? path.resolve(client.args["rhs-root"])
5255
: undefined;
56+
if (!rhsRoot) {
57+
const tempRhsRoot = client.checkDefaultRhsRoot(lhs, rhs);
58+
if (tempRhsRoot) {
59+
// since we are using the default rhsRoot, we should delete it when we are done.
60+
client.tempRhsRoot = tempRhsRoot;
61+
rhsRoot = tempRhsRoot;
62+
}
63+
}
5364
const lhsParser = await SwaggerParser.create(lhs, lhsRoot, client);
5465
const rhsParser = await SwaggerParser.create(rhs, rhsRoot, client);
5566
client.lhsParser = lhsParser;
@@ -66,6 +77,47 @@ export class DiffClient {
6677
this.args["rhs"] = rhs;
6778
}
6879

80+
/**
81+
* If the rhsRoot is not provided, rhs points to TypeSpec and the lhs is a folder, use the lhs as the basis
82+
* for the rhsRoot.
83+
* @param lhs lhs paths
84+
* @param rhs rhs paths
85+
* @returns the inferred rhsRoot or undefined if it cannot be inferred
86+
*/
87+
private checkDefaultRhsRoot(
88+
lhs: string[],
89+
rhs: string[]
90+
): string | undefined {
91+
if (lhs.length === 1 && rhs.length === 1) {
92+
const lhsStat = fs.statSync(lhs[0]);
93+
const rhsStat = fs.statSync(rhs[0]);
94+
if (lhsStat.isDirectory() && rhsStat.isDirectory()) {
95+
// check if the lhs folder is swagger and rhs folder is typespec
96+
const lhsFiles = fs.readdirSync(lhs[0]);
97+
const lhsSwagger = !lhsFiles.some((x) => x.endsWith(".tsp"));
98+
const rhsFiles = fs.readdirSync(rhs[0]);
99+
const rhsTypespec = rhsFiles.some((x) => x.endsWith(".tsp"));
100+
if (lhsSwagger && rhsTypespec) {
101+
const lhsSegments = lhs[0].split(path.sep);
102+
const rhsSegments = [];
103+
for (const segment of lhsSegments) {
104+
if (segment === "preview" || segment === "stable") {
105+
rhsSegments.push("temp");
106+
} else {
107+
rhsSegments.push(segment);
108+
}
109+
}
110+
const rhsRoot = rhsSegments.join(path.sep);
111+
console.warn(
112+
`WARN: No rhs-root provided. Using '${rhsRoot}' as the rhs-root.`
113+
);
114+
return rhsRoot;
115+
}
116+
}
117+
}
118+
return undefined;
119+
}
120+
69121
/**
70122
* Parses the documents, expanding them into canonical transformations.
71123
* Upon completion, the lhs and rhs documents will be available for diffing.
@@ -271,6 +323,12 @@ export class DiffClient {
271323
}
272324
}
273325

326+
// delete the tempRhsRoot if used
327+
if (this.tempRhsRoot) {
328+
console.warn(`WARN: Cleaning up temporary folder '${this.tempRhsRoot}'`);
329+
fs.rmdirSync(this.tempRhsRoot, { recursive: true });
330+
}
331+
274332
// create inverse files that show only the stuff that has been pruned
275333
// for diagnostic purposes.
276334
fs.writeFileSync(

0 commit comments

Comments
 (0)