All files / src/entities/releases schema-readonly.ts

100% Statements 8/8
100% Branches 0/0
100% Functions 0/0
100% Lines 8/8

Press n or j to go to the next uncovered block, b, p or k for the previous block.

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 7818x 18x                   18x     18x         18x                                           18x                   18x                             18x                      
import { z } from "zod";
import { flexibleBoolean, requiredId } from "../utils";
 
// ============================================================================
// browse_releases - CQRS Query Tool (discriminated union schema)
// Actions: list, get, assets
// Uses z.discriminatedUnion() for type-safe action handling.
// Schema pipeline flattens to flat JSON Schema for AI clients that don't support oneOf.
// ============================================================================
 
// --- Shared fields ---
const projectIdField = requiredId.describe(
  "Project ID or URL-encoded path (e.g., 'my-group/my-project')"
);
const tagNameField = z
  .string()
  .describe("The Git tag associated with the release (e.g., 'v1.0.0')");
 
// --- Action: list ---
const ListReleasesSchema = z.object({
  action: z.literal("list").describe("List all releases for a project, sorted by release date"),
  project_id: projectIdField,
  order_by: z
    .enum(["released_at", "created_at"])
    .optional()
    .describe("Sort releases by field (default: released_at)"),
  sort: z.enum(["desc", "asc"]).optional().describe("Sort direction (default: desc)"),
  include_html_description: flexibleBoolean
    .optional()
    .describe("Include HTML-rendered description in response"),
  per_page: z
    .number()
    .int()
    .min(1)
    .max(100)
    .optional()
    .describe("Number of items per page (max 100)"),
  page: z.number().int().min(1).optional().describe("Page number"),
});
 
// --- Action: get ---
const GetReleaseSchema = z.object({
  action: z.literal("get").describe("Get a specific release by its tag name"),
  project_id: projectIdField,
  tag_name: tagNameField,
  include_html_description: flexibleBoolean
    .optional()
    .describe("Include HTML-rendered description in response"),
});
 
// --- Action: assets ---
const ListReleaseAssetsSchema = z.object({
  action: z.literal("assets").describe("List all asset links for a specific release"),
  project_id: projectIdField,
  tag_name: tagNameField,
  per_page: z
    .number()
    .int()
    .min(1)
    .max(100)
    .optional()
    .describe("Number of items per page (max 100)"),
  page: z.number().int().min(1).optional().describe("Page number"),
});
 
// --- Discriminated union combining all actions ---
export const BrowseReleasesSchema = z.discriminatedUnion("action", [
  ListReleasesSchema,
  GetReleaseSchema,
  ListReleaseAssetsSchema,
]);
 
// ============================================================================
// Type exports
// ============================================================================
 
export type BrowseReleasesInput = z.infer<typeof BrowseReleasesSchema>;