Skip to content
This repository has been archived by the owner on Sep 18, 2024. It is now read-only.

[Issue #96]: Opportunity listing page (first pass) #97

Merged
merged 7 commits into from
Jun 20, 2024
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
64 changes: 64 additions & 0 deletions frontend/src/app/[locale]/opportunity/[id]/page.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
import { Metadata } from "next";
import NotFound from "../../../not-found";
import OpportunityListingAPI from "../../../api/OpportunityListingAPI";
import { getTranslations } from "next-intl/server";

export async function generateMetadata() {
const t = await getTranslations({ locale: "en" });
const meta: Metadata = {
title: t("OpportunityListing.page_title"),
description: t("OpportunityListing.meta_description"),
};
return meta;
}

export default async function OpportunityListing({
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm still catching up on whether or not making a page async affects ability to statically render the page. The opportunity pages are the same for everyone and rarely update so should not be rebuilt for every single request. Statically rendering seems like the easiest way to achieve that.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can punt this consideration for #85 .

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

params,
}: {
params: { id: string };
}) {
const id = Number(params.id);

// Opportunity id needs to be a number greater than 1
if (isNaN(id) || id < 0) {
return <NotFound />;
}

const api = new OpportunityListingAPI();
let opportunity;
try {
opportunity = await api.getOpportunityById(id);
} catch (error) {
console.error("Failed to fetch opportunity:", error);
return <NotFound />;
}

if (!opportunity.data) {
return <NotFound />;
}

return (
<div className="grid-container">
<div className="grid-row margin-y-4">
<div className="usa-table-container">
<table className="usa-table usa-table--borderless margin-x-auto width-full maxw-desktop-lg">
<thead>
<tr>
<th>Field Name</th>
<th>Data</th>
</tr>
</thead>
<tbody>
{Object.entries(opportunity.data).map(([key, value]) => (
<tr key={key}>
<td className="word-wrap">{key}</td>
<td className="word-wrap">{JSON.stringify(value)}</td>
</tr>
))}
</tbody>
</table>
Copy link
Collaborator

@chouinar chouinar Jun 20, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you add a small bit of special logic for the summary field so that it doesn't print out as the full JSON, but instead pulls out each of the fields? Just to make it look a little bit nicer / make it look like we've done a lot more and are truly just waiting on shoving it into whatever designs.

So, rather than summary with a value of { ... } (the whole object), do something like (in hacky python pseudo-code):

for key, value in obj.items():
     if key == "summary":
          for summary_key, summary_value in value.items():
                 print(summary_key)
                 print(JSON.stringify(summary_value)

    else:
         print(key)
         print(value)
 

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks @chouinar I added this:

image
image

</div>
</div>
</div>
);
}
27 changes: 13 additions & 14 deletions frontend/src/app/api/BaseApi.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,3 @@
// This server-only package is recommended by Next.js to ensure code is only run on the server.
// It provides a build-time error if client-side code attempts to invoke the code here.
// Since we're pulling in an API Auth Token here, this should be server only
// https://nextjs.org/docs/app/building-your-application/rendering/composition-patterns#keeping-server-only-code-out-of-the-client-environment
import "server-only";

import {
Expand Down Expand Up @@ -69,7 +65,7 @@ export default abstract class BaseApi {
basePath: string,
namespace: string,
subPath: string,
queryParamData: QueryParamData,
queryParamData?: QueryParamData,
body?: JSONRequestBody,
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

all queryParamData / searchInput references need to be an optional param because they're only used for search (SearchOpportunityAPI class) and not when the derived OpportunityListingAPI class is used.

options: {
additionalHeaders?: HeadersDict;
Expand Down Expand Up @@ -109,11 +105,12 @@ export default abstract class BaseApi {
private async sendRequest(
url: string,
fetchOptions: RequestInit,
queryParamData: QueryParamData,
queryParamData?: QueryParamData,
) {
let response: Response;
let responseBody: SearchAPIResponse;
try {
console.log("url => ", url, " fethcOptions=>", fetchOptions);
response = await fetch(url, fetchOptions);
responseBody = (await response.json()) as SearchAPIResponse;
} catch (error) {
Expand Down Expand Up @@ -153,14 +150,14 @@ export function createRequestUrl(
let url = [...cleanedPaths].join("/");
if (method === "GET" && body && !(body instanceof FormData)) {
// Append query string to URL
const body: { [key: string]: string } = {};
const bodyParams: { [key: string]: string } = {};
Object.entries(body).forEach(([key, value]) => {
const stringValue =
typeof value === "string" ? value : JSON.stringify(value);
body[key] = stringValue;
bodyParams[key] = stringValue;
});

const params = new URLSearchParams(body).toString();
const params = new URLSearchParams(bodyParams).toString();
url = `${url}?${params}`;
}
return url;
Expand Down Expand Up @@ -189,19 +186,21 @@ function createRequestBody(payload?: JSONRequestBody): XMLHttpRequestBodyInit {
*/
export function fetchErrorToNetworkError(
error: unknown,
searchInputs: QueryParamData,
searchInputs?: QueryParamData,
) {
// Request failed to send or something failed while parsing the response
// Log the JS error to support troubleshooting
console.error(error);
return new NetworkError(error, searchInputs);
return searchInputs
? new NetworkError(error, searchInputs)
: new NetworkError(error);
}

function handleNotOkResponse(
response: SearchAPIResponse,
message: string,
status_code: number,
searchInputs: QueryParamData,
searchInputs?: QueryParamData,
) {
const { errors } = response;
if (isEmpty(errors)) {
Expand All @@ -218,7 +217,7 @@ function handleNotOkResponse(
const throwError = (
message: string,
status_code: number,
searchInputs: QueryParamData,
searchInputs?: QueryParamData,
firstError?: APIResponseError,
) => {
console.log("Throwing error: ", message, status_code, searchInputs);
Expand Down Expand Up @@ -246,9 +245,9 @@ const throwError = (
default:
throw new ApiRequestError(
error,
searchInputs,
"APIRequestError",
status_code,
searchInputs,
);
}
};
24 changes: 24 additions & 0 deletions frontend/src/app/api/OpportunityListingAPI.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import "server-only";
Copy link
Collaborator Author

@rylew1 rylew1 Jun 19, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

only ever run this code on the next server


import BaseApi from "./BaseApi";

export default class OpportunityListingAPI extends BaseApi {
get basePath(): string {
return process.env.API_URL || "";
}

get namespace(): string {
return "opportunities";
}

async getOpportunityById(opportunityId: number) {
const subPath = `${opportunityId}`;
const response = await this.request(
"GET",
this.basePath,
this.namespace,
subPath,
);
return response;
}
}
56 changes: 30 additions & 26 deletions frontend/src/errors.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,10 @@ import { QueryParamData } from "src/services/search/searchfetcher/SearchFetcher"
*/

export class NetworkError extends Error {
constructor(error: unknown, searchInputs: QueryParamData) {
const serializedSearchInputs = convertSearchInputSetsToArrays(searchInputs);
constructor(error: unknown, searchInputs?: QueryParamData) {
const serializedSearchInputs = searchInputs
? convertSearchInputSetsToArrays(searchInputs)
: {};

const serializedData = JSON.stringify({
type: "NetworkError",
Expand All @@ -29,15 +31,17 @@ export class NetworkError extends Error {
export class BaseFrontendError extends Error {
constructor(
error: unknown,
searchInputs: QueryParamData,
type: string,
type = "BaseFrontendError",
status?: number,
searchInputs?: QueryParamData,
) {
// Sets cannot be properly serialized so convert to arrays first
const serializedSearchInputs = convertSearchInputSetsToArrays(searchInputs);
const serializedSearchInputs = searchInputs
? convertSearchInputSetsToArrays(searchInputs)
: {};

const serializedData = JSON.stringify({
type: type || "BaseFrontendError",
type,
searchInputs: serializedSearchInputs,
message: error instanceof Error ? error.message : "Unknown Error",
status,
Expand All @@ -61,29 +65,29 @@ export class BaseFrontendError extends Error {
export class ApiRequestError extends BaseFrontendError {
constructor(
error: unknown,
searchInputs: QueryParamData,
type: string,
status: number,
type = "APIRequestError",
status = 400,
searchInputs?: QueryParamData,
) {
super(error, searchInputs, type || "APIRequestError", status || 400);
super(error, type, status, searchInputs);
}
}

/**
* An API response returned a 400 status code and its JSON body didn't include any `errors`
*/
export class BadRequestError extends ApiRequestError {
constructor(error: unknown, searchInputs: QueryParamData) {
super(error, searchInputs, "BadRequestError", 400);
constructor(error: unknown, searchInputs?: QueryParamData) {
super(error, "BadRequestError", 400, searchInputs);
}
}

/**
* An API response returned a 401 status code
*/
export class UnauthorizedError extends ApiRequestError {
constructor(error: unknown, searchInputs: QueryParamData) {
super(error, searchInputs, "UnauthorizedError", 401);
constructor(error: unknown, searchInputs?: QueryParamData) {
super(error, "UnauthorizedError", 401, searchInputs);
}
}

Expand All @@ -93,35 +97,35 @@ export class UnauthorizedError extends ApiRequestError {
* being created, or the user hasn't consented to the data sharing agreement.
*/
export class ForbiddenError extends ApiRequestError {
constructor(error: unknown, searchInputs: QueryParamData) {
super(error, searchInputs, "ForbiddenError", 403);
constructor(error: unknown, searchInputs?: QueryParamData) {
super(error, "ForbiddenError", 403, searchInputs);
}
}

/**
* A fetch request failed due to a 404 error
*/
export class NotFoundError extends ApiRequestError {
constructor(error: unknown, searchInputs: QueryParamData) {
super(error, searchInputs, "NotFoundError", 404);
constructor(error: unknown, searchInputs?: QueryParamData) {
super(error, "NotFoundError", 404, searchInputs);
}
}

/**
* An API response returned a 408 status code
*/
export class RequestTimeoutError extends ApiRequestError {
constructor(error: unknown, searchInputs: QueryParamData) {
super(error, searchInputs, "RequestTimeoutError", 408);
constructor(error: unknown, searchInputs?: QueryParamData) {
super(error, "RequestTimeoutError", 408, searchInputs);
}
}

/**
* An API response returned a 422 status code
*/
export class ValidationError extends ApiRequestError {
constructor(error: unknown, searchInputs: QueryParamData) {
super(error, searchInputs, "ValidationError", 422);
constructor(error: unknown, searchInputs?: QueryParamData) {
super(error, "ValidationError", 422, searchInputs);
}
}

Expand All @@ -133,17 +137,17 @@ export class ValidationError extends ApiRequestError {
* An API response returned a 500 status code
*/
export class InternalServerError extends ApiRequestError {
constructor(error: unknown, searchInputs: QueryParamData) {
super(error, searchInputs, "InternalServerError", 500);
constructor(error: unknown, searchInputs?: QueryParamData) {
super(error, "InternalServerError", 500, searchInputs);
}
}

/**
* An API response returned a 503 status code
*/
export class ServiceUnavailableError extends ApiRequestError {
constructor(error: unknown, searchInputs: QueryParamData) {
super(error, searchInputs, "ServiceUnavailableError", 503);
constructor(error: unknown, searchInputs?: QueryParamData) {
super(error, "ServiceUnavailableError", 503, searchInputs);
}
}

Expand Down
4 changes: 4 additions & 0 deletions frontend/src/i18n/messages/en/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,10 @@ export const messages = {
alert:
"Simpler.Grants.gov is a work in progress. Thank you for your patience as we build this new website.",
},
OpportunityListing: {
page_title: "Opportunity Listing",
meta_description: "Summary details for the specific opportunity listing.",
},
Index: {
page_title: "Simpler.Grants.gov",
meta_description:
Expand Down
1 change: 1 addition & 0 deletions frontend/src/utils/getRoutes.ts
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ export function getNextRoutes(src: string): string[] {
.replace("/page.tsx", "")
.replace(/\[locale\]/g, "")
.replace(/\\/g, "/")
.replace(/\[id\]/g, "1") // for id-based routes like /opportunity/[id]
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

allow pa11y to run on the opportunity listing page by just setting any [id] found to 1 - resulting in /opportunity/1

: "/";
return route.replace(/\/\//g, "/");
});
Expand Down
2 changes: 2 additions & 0 deletions frontend/tests/utils/getRoutes.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ jest.mock("../../src/utils/getRoutes", () => {

const mockedListPaths = listPaths as jest.MockedFunction<typeof listPaths>;

// TODO: https://github.com/navapbc/simpler-grants-gov/issues/98
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this test is a little wonky rn - it's not actually mocking properly so it pulls what the sitemap actually pulls (so we have to update the test every time there's a new route - which is not really testing the functions in getRoutes in isolation) - so there's a ticket to come back to this

describe("getNextRoutes", () => {
beforeEach(() => {
jest.resetAllMocks();
Expand All @@ -30,6 +31,7 @@ describe("getNextRoutes", () => {
"/newsletter/confirmation",
"/newsletter",
"/newsletter/unsubscribe",
"/opportunity/1",
"/",
"/process",
"/research",
Expand Down
Loading