From abe85fa3aed63db150dc942468648b07b4b379ae Mon Sep 17 00:00:00 2001
From: RahulGautamSingh <rahultesnik@gmail.com>
Date: Mon, 25 Nov 2024 13:53:12 +0530
Subject: [PATCH] feat(datasource/cpan): populate latest tag (#32677)

---
 .../datasource/cpan/__fixtures__/Plack.json   | 33 +++++++++
 lib/modules/datasource/cpan/index.spec.ts     |  1 +
 lib/modules/datasource/cpan/index.ts          | 73 +++++++++----------
 lib/modules/datasource/cpan/schema.ts         | 64 ++++++++++++++++
 lib/modules/datasource/cpan/types.ts          | 25 +------
 lib/modules/datasource/types.ts               |  1 +
 6 files changed, 135 insertions(+), 62 deletions(-)
 create mode 100644 lib/modules/datasource/cpan/schema.ts

diff --git a/lib/modules/datasource/cpan/__fixtures__/Plack.json b/lib/modules/datasource/cpan/__fixtures__/Plack.json
index 9fb33a17fd..7d402d1db6 100644
--- a/lib/modules/datasource/cpan/__fixtures__/Plack.json
+++ b/lib/modules/datasource/cpan/__fixtures__/Plack.json
@@ -24,6 +24,7 @@
                      "version" : "1.0048"
                   }
                ],
+               "status": "latest",
                "date" : "2020-11-30T00:21:36",
                "maturity" : "released",
                "distribution" : "Plack",
@@ -47,6 +48,7 @@
                      "name" : "Plack"
                   }
                ],
+               "status": "cpan",
                "date" : "2018-02-10T09:25:30",
                "maturity" : "released"
             },
@@ -63,6 +65,7 @@
                      "name" : "Plack"
                   }
                ],
+               "status": "cpan",
                "date" : "2018-02-10T07:52:31",
                "deprecated" : false
             },
@@ -84,6 +87,7 @@
                      "name" : "Plack"
                   }
                ],
+               "status": "cpan",
                "date" : "2017-12-31T20:42:50",
                "distribution" : "Plack",
                "download_url" : "https://cpan.metacpan.org/authors/id/M/MI/MIYAGAWA/Plack-1.0045.tar.gz"
@@ -107,6 +111,7 @@
                      "version" : "1.0044"
                   }
                ],
+               "status": "cpan",
                "date" : "2017-04-27T17:48:20",
                "maturity" : "released",
                "deprecated" : false
@@ -134,6 +139,7 @@
                      "version" : "1.0043"
                   }
                ],
+               "status": "cpan",
                "date" : "2017-02-22T03:02:05",
                "maturity" : "released",
                "distribution" : "Plack",
@@ -158,6 +164,7 @@
                      "version" : "1.0042"
                   }
                ],
+               "status": "cpan",
                "date" : "2016-09-29T05:38:42",
                "deprecated" : false
             }
@@ -172,6 +179,7 @@
                      "version" : "1.0041"
                   }
                ],
+               "status": "cpan",
                "date" : "2016-09-25T21:25:47",
                "download_url" : "https://cpan.metacpan.org/authors/id/M/MI/MIYAGAWA/Plack-1.0041.tar.gz",
                "distribution" : "Plack"
@@ -195,6 +203,7 @@
                      "name" : "Plack"
                   }
                ],
+               "status": "cpan",
                "date" : "2016-04-01T16:58:21",
                "maturity" : "developer"
             },
@@ -217,6 +226,7 @@
                      "name" : "Plack"
                   }
                ],
+               "status": "cpan",
                "date" : "2015-12-06T11:29:40",
                "distribution" : "Plack",
                "download_url" : "https://cpan.metacpan.org/authors/id/M/MI/MIYAGAWA/Plack-1.0039.tar.gz"
@@ -227,6 +237,29 @@
             "sort" : [
                1449401380000
             ]
+         },
+         {
+            "_type" : "file",
+            "_source" : {
+               "deprecated" : false,
+               "maturity" : "released",
+               "module" : [
+                  {
+                     "version" : "1.0038",
+                     "name" : "Plack"
+                  }
+               ],
+               "date" : "2015-12-06T11:29:40",
+               "distribution" : "Plack",
+               "download_url" : "https://cpan.metacpan.org/authors/id/M/MI/MIYAGAWA/Plack-1.0039.tar.gz"
+            },
+            "status": "invalid",
+            "_index" : "cpan_v1_01",
+            "_id" : "Y7WlIYOZjk3rh9O05F3UZE6WwGo",
+            "_score" : null,
+            "sort" : [
+               1449401380000
+            ]
          }
       ],
       "max_score" : null
diff --git a/lib/modules/datasource/cpan/index.spec.ts b/lib/modules/datasource/cpan/index.spec.ts
index c4727a1281..231e5bd316 100644
--- a/lib/modules/datasource/cpan/index.spec.ts
+++ b/lib/modules/datasource/cpan/index.spec.ts
@@ -86,6 +86,7 @@ describe('modules/datasource/cpan/index', () => {
         releaseTimestamp: '2020-11-30T00:21:36.000Z',
         version: '1.0048',
       });
+      expect(res?.tags?.latest).toBe('1.0048');
     });
   });
 });
diff --git a/lib/modules/datasource/cpan/index.ts b/lib/modules/datasource/cpan/index.ts
index 85436a1d92..435c48764f 100644
--- a/lib/modules/datasource/cpan/index.ts
+++ b/lib/modules/datasource/cpan/index.ts
@@ -2,8 +2,9 @@ import { cache } from '../../../util/cache/package/decorator';
 import { joinUrlParts } from '../../../util/url';
 import * as perlVersioning from '../../versioning/perl';
 import { Datasource } from '../datasource';
-import type { GetReleasesConfig, Release, ReleaseResult } from '../types';
-import type { MetaCpanApiFile, MetaCpanApiFileSearchResult } from './types';
+import type { GetReleasesConfig, ReleaseResult } from '../types';
+import { MetaCpanApiFileSearchResponse } from './schema';
+import type { CpanRelease } from './types';
 
 export class CpanDatasource extends Datasource {
   static readonly id = 'cpan';
@@ -38,7 +39,7 @@ export class CpanDatasource extends Datasource {
     let result: ReleaseResult | null = null;
     const searchUrl = joinUrlParts(registryUrl, 'v1/file/_search');
 
-    let hits: MetaCpanApiFile[] | null = null;
+    let releases: CpanRelease[] | null = null;
     try {
       const body = {
         query: {
@@ -60,55 +61,47 @@ export class CpanDatasource extends Datasource {
           'date',
           'deprecated',
           'maturity',
+          'status',
         ],
         sort: [{ date: 'desc' }],
       };
-      const res = await this.http.postJson<MetaCpanApiFileSearchResult>(
-        searchUrl,
-        { body },
-      );
-      hits = res.body?.hits?.hits?.map(({ _source }) => _source);
+
+      releases = (
+        await this.http.postJson(
+          searchUrl,
+          { body },
+          MetaCpanApiFileSearchResponse,
+        )
+      ).body;
     } catch (err) {
       this.handleGenericErrors(err);
     }
 
     let latestDistribution: string | null = null;
-    if (hits) {
-      const releases: Release[] = [];
-      for (const hit of hits) {
-        const {
-          module,
-          distribution,
-          date: releaseTimestamp,
-          deprecated: isDeprecated,
-          maturity,
-        } = hit;
-        const version = module.find(
-          ({ name }) => name === packageName,
-        )?.version;
-        if (version) {
-          // https://metacpan.org/pod/CPAN::DistnameInfo#maturity
-          const isStable = maturity === 'released';
-          releases.push({
-            isDeprecated,
-            isStable,
-            releaseTimestamp,
-            version,
-          });
-
-          if (!latestDistribution) {
-            latestDistribution = distribution;
-          }
+    let latestVersion: string | null = null;
+    if (releases) {
+      for (const release of releases) {
+        if (!latestDistribution) {
+          latestDistribution = release.distribution;
+        }
+        if (!latestVersion && release.isLatest) {
+          latestVersion = release.version;
         }
       }
-      if (releases.length > 0 && latestDistribution) {
-        result = {
-          releases,
-          changelogUrl: `https://metacpan.org/dist/${latestDistribution}/changes`,
-          homepage: `https://metacpan.org/pod/${packageName}`,
-        };
+    }
+    if (releases.length > 0 && latestDistribution) {
+      result = {
+        releases,
+        changelogUrl: `https://metacpan.org/dist/${latestDistribution}/changes`,
+        homepage: `https://metacpan.org/pod/${packageName}`,
+      };
+
+      if (latestVersion) {
+        result.tags ??= {};
+        result.tags.latest = latestVersion;
       }
     }
+
     return result;
   }
 }
diff --git a/lib/modules/datasource/cpan/schema.ts b/lib/modules/datasource/cpan/schema.ts
new file mode 100644
index 0000000000..8e997143ba
--- /dev/null
+++ b/lib/modules/datasource/cpan/schema.ts
@@ -0,0 +1,64 @@
+import { z } from 'zod';
+import { LooseArray } from '../../../util/schema-utils';
+import type { CpanRelease } from './types';
+
+/**
+ * https://fastapi.metacpan.org/v1/file/_mapping
+ */
+const MetaCpanApiFileSchema = z
+  .object({
+    module: LooseArray(
+      z.object({
+        name: z.string(),
+        version: z.string(),
+      }),
+    ),
+    distribution: z.string(),
+    date: z.string(),
+    deprecated: z.boolean(),
+    maturity: z.string(),
+    status: z.union([
+      z.literal('backpan'),
+      z.literal('cpan'),
+      z.literal('latest'),
+    ]),
+  })
+  .transform(
+    ({
+      module,
+      distribution,
+      date,
+      deprecated,
+      maturity,
+      status,
+    }): CpanRelease | undefined => {
+      return {
+        version: module[0].version,
+        distribution,
+        isDeprecated: deprecated,
+        isStable: maturity === 'released',
+        releaseTimestamp: date,
+        isLatest: status === 'latest',
+      };
+    },
+  )
+  .catch(undefined);
+/**
+ * https://github.com/metacpan/metacpan-api/blob/master/docs/API-docs.md#available-fields
+ */
+export const MetaCpanApiFileSearchResponse = z
+  .object({
+    hits: z.object({
+      hits: LooseArray(
+        z.object({
+          _source: MetaCpanApiFileSchema,
+        }),
+      ),
+    }),
+  })
+  .transform((data): CpanRelease[] => {
+    // Extract all hits and filter out ones where _source transformed to undefined
+    return data.hits.hits
+      .map((hit) => hit._source)
+      .filter((source) => source !== undefined);
+  });
diff --git a/lib/modules/datasource/cpan/types.ts b/lib/modules/datasource/cpan/types.ts
index e6279b5b86..0d9cef2408 100644
--- a/lib/modules/datasource/cpan/types.ts
+++ b/lib/modules/datasource/cpan/types.ts
@@ -1,24 +1,5 @@
-/**
- * https://fastapi.metacpan.org/v1/file/_mapping
- */
-export interface MetaCpanApiFile {
-  module: {
-    name: string;
-    version?: string;
-  }[];
-  distribution: string;
-  date: string;
-  deprecated: boolean;
-  maturity: string;
-}
+import type { Release } from '../types';
 
-/**
- * https://github.com/metacpan/metacpan-api/blob/master/docs/API-docs.md#available-fields
- */
-export interface MetaCpanApiFileSearchResult {
-  hits: {
-    hits: {
-      _source: MetaCpanApiFile;
-    }[];
-  };
+export interface CpanRelease extends Release {
+  distribution: string;
 }
diff --git a/lib/modules/datasource/types.ts b/lib/modules/datasource/types.ts
index 94336ff106..aea32290d9 100644
--- a/lib/modules/datasource/types.ts
+++ b/lib/modules/datasource/types.ts
@@ -71,6 +71,7 @@ export interface Release {
   sourceUrl?: string | undefined;
   sourceDirectory?: string;
   currentAge?: string;
+  isLatest?: boolean;
 }
 
 export interface ReleaseResult {
-- 
GitLab