Skip to content

Commit

Permalink
WIP: add football live page
Browse files Browse the repository at this point in the history
  • Loading branch information
marjisound committed Dec 6, 2024
1 parent a61774d commit f699361
Show file tree
Hide file tree
Showing 12 changed files with 345 additions and 18 deletions.
1 change: 1 addition & 0 deletions dotcom-rendering/configs/webpack/server.dev.mjs
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,7 @@ export const devServer = {
},
},
setupMiddlewares: (middlewares, { app, compiler }) => {
console.log('marji 2: ');
if (!app) {
throw new Error('webpack-dev-server is not defined');
}
Expand Down
1 change: 1 addition & 0 deletions dotcom-rendering/scripts/json-schema/gen-schema.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ const tagPageSchema = getTagPageSchema();
const newsletterPageSchema = getNewsletterPageSchema();
const blockSchema = getBlockSchema();
const editionsCrosswordSchema = getEditionsCrosswordSchema();
const sportsSchema = getArticleSchema();

Check failure on line 21 in dotcom-rendering/scripts/json-schema/gen-schema.js

View workflow job for this annotation

GitHub Actions / lint / check

'sportsSchema' is assigned a value but never used

fs.writeFile(
`${root}/src/model/article-schema.json`,
Expand Down
8 changes: 8 additions & 0 deletions dotcom-rendering/scripts/json-schema/get-schema.js
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,14 @@ const getArticleSchema = () => {
);
};

const getSportsSchema = () => {

Check failure on line 29 in dotcom-rendering/scripts/json-schema/get-schema.js

View workflow job for this annotation

GitHub Actions / lint / check

'getSportsSchema' is assigned a value but never used
return JSON.stringify(
TJS.generateSchema(program, 'FELiveScoresType', settings),
null,
4,
);
};

const getFrontSchema = () => {
return JSON.stringify(
TJS.generateSchema(program, 'FEFrontType', settings),
Expand Down
48 changes: 48 additions & 0 deletions dotcom-rendering/src/components/Football/LiveScoresPage.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import { FELiveScoresType } from 'src/types/sports';

Check failure on line 1 in dotcom-rendering/src/components/Football/LiveScoresPage.tsx

View workflow job for this annotation

GitHub Actions / lint / check

'src/types/sports' import is restricted from being used by a pattern. Paths starting with “src/” are forbidden. Please use a relative path instead

Check failure on line 1 in dotcom-rendering/src/components/Football/LiveScoresPage.tsx

View workflow job for this annotation

GitHub Actions / lint / check

All imports in the declaration are only used as types. Use `import type`
import { css } from '@emotion/react';

Check failure on line 2 in dotcom-rendering/src/components/Football/LiveScoresPage.tsx

View workflow job for this annotation

GitHub Actions / lint / check

`@emotion/react` import should occur before import of `src/types/sports`
import { MatchList } from './MatchList';

interface Props {
liveScores: FELiveScoresType;
}

const sportsPageStyles = css`
padding-top: 0;
padding-bottom: 2.25rem;
`;

const titleStyles = css`
font-size: 1.25rem;
line-height: 1.4375rem;
font-family: 'Guardian Egyptian Web', Georgia, serif;
font-weight: 900;
box-sizing: border-box;
padding: 0.375rem 0 0.75rem;
border-top: 0.0625rem dotted;
`;

const matchContainerStyles = css`
clear: both;
`;

export const LiveScoresPage = ({ liveScores }: Props) => {
return (
<article id="article" css={[sportsPageStyles]} role="main">
<div>
<h2 css={[titleStyles]}>{liveScores.pageTitle}</h2>
<div
css={[matchContainerStyles]}
data-show-more-contains="football-matches"
>
{liveScores.matchesGroupedByDateAndCompetition.map(
(item) => {
return (
<MatchList dateCompetition={item}></MatchList>

Check failure on line 40 in dotcom-rendering/src/components/Football/LiveScoresPage.tsx

View workflow job for this annotation

GitHub Actions / lint / check

Missing "key" prop for element in iterator
);
},
)}
</div>
</div>
</article>
);
};
48 changes: 48 additions & 0 deletions dotcom-rendering/src/components/Football/MatchList.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import { DateCompetitionMatch } from 'src/types/sports';

Check failure on line 1 in dotcom-rendering/src/components/Football/MatchList.tsx

View workflow job for this annotation

GitHub Actions / lint / check

'src/types/sports' import is restricted from being used by a pattern. Paths starting with “src/” are forbidden. Please use a relative path instead

Check failure on line 1 in dotcom-rendering/src/components/Football/MatchList.tsx

View workflow job for this annotation

GitHub Actions / lint / check

All imports in the declaration are only used as types. Use `import type`

interface Props {
dateCompetition: DateCompetitionMatch;
}

export const MatchList: React.FC<Props> = ({ dateCompetition }) => {

Check failure on line 7 in dotcom-rendering/src/components/Football/MatchList.tsx

View workflow job for this annotation

GitHub Actions / lint / check

Don't use `React.FC` as a type. Please use const MyThing = ({foo, bar}: Props) instead
return (
<>
<div>{dateCompetition.date}</div>
{dateCompetition.competitions.map((comp) => (
<div key={comp.competition.id}>
<h3>{comp.competition.fullName}</h3>
<table>
<thead hidden>

Check failure on line 15 in dotcom-rendering/src/components/Football/MatchList.tsx

View workflow job for this annotation

GitHub Actions / lint / check

Value must be set for boolean attributes
<tr>
<th>Match status / kick off time</th>
<th>Match details</th>
</tr>
</thead>
<tbody>
{comp.matches.map((match) => (
<tr key={match.id} id={match.id}>
<td>{match.date.toString()}</td>
<td>
<strong>{match.homeTeam.name}</strong>{' '}
vs{' '}
<strong>{match.awayTeam.name}</strong>
{match.type === 'MatchDay' &&
match.liveMatch && (
<span> (Live!)</span>
)}
{match.type === 'Fixture' && (
<span> (Fixture)</span>
)}
{match.type === 'Result' && (
<span> (Result)</span>
)}
</td>
</tr>
))}
</tbody>
</table>
</div>
))}
</>
);
};
12 changes: 12 additions & 0 deletions dotcom-rendering/src/model/sports-schema.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
{
"type": "object",
"properties": {
"pageTitle": {
"type": "string"
}
},
"required": [
"pageTitle"
],
"$schema": "http://json-schema.org/draft-07/schema#"
}
14 changes: 14 additions & 0 deletions dotcom-rendering/src/model/validate.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,11 @@ import type { FETagPageType } from '../types/tagPage';
import articleSchema from './article-schema.json';
import blockSchema from './block-schema.json';
import editionsCrosswordSchema from './editions-crossword-schema.json';
import sportSchema from './sports-schema.json';
import frontSchema from './front-schema.json';
import newslettersPageSchema from './newsletter-page-schema.json';
import tagPageSchema from './tag-page-schema.json';
import { FELiveScoresType } from 'src/types/sports';

const options: Options = {
verbose: false,
Expand All @@ -36,6 +38,8 @@ const validateEditionsCrossword = ajv.compile<FEEditionsCrosswords>(
editionsCrosswordSchema,
);

const validateSports = ajv.compile<FELiveScoresType>(sportSchema);

export const validateAsArticleType = (data: unknown): FEArticleType => {
if (validateArticle(data)) return data;

Expand All @@ -60,6 +64,16 @@ export const validateAsEditionsCrosswordType = (
);
};

export const validateAsSports = (data: unknown): FELiveScoresType => {
if (validateSports(data)) {
return data;
}
throw new TypeError(
`Unable to validate request body for editions crosswords.\n
${JSON.stringify(validateSports.errors, null, 2)}`,
);
};

export const validateAsFrontType = (data: unknown): FEFrontType => {
if (validateFront(data)) return data;

Expand Down
16 changes: 16 additions & 0 deletions dotcom-rendering/src/server/handler.sports.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
import type { RequestHandler } from 'express';
import { validateAsSports } from '../model/validate';
import { renderSportsHtml } from './render.sports.web';
import { makePrefetchHeader } from './lib/header';

export const handleSports: RequestHandler = ({ body }, res) => {
console.log(`marji: `);
const matchList = validateAsSports(body);
console.log(matchList);

const { html, prefetchScripts } = renderSportsHtml({
sports: matchList,
});

res.status(200).set('Link', makePrefetchHeader(prefetchScripts)).send(html);
};
58 changes: 40 additions & 18 deletions dotcom-rendering/src/server/lib/get-content-from-url.js
Original file line number Diff line number Diff line change
Expand Up @@ -27,11 +27,18 @@ async function getContentFromURL(url, _headers) {
.filter(isStringTuple),
);

console.log(`fetch url: ${jsonUrl}`);

// pick all the keys from the JSON except `html`
// eslint-disable-next-line @typescript-eslint/no-unused-vars -- we don't want `html` in the config
const { html, ...config } = await fetch(jsonUrl, { headers })

Check failure

Code scanning / CodeQL

Server-side request forgery Critical

The
URL
of this request depends on a
user-provided value
.
.then((response) => response.json())
.then((response) => {
console.log(response);

return response.json();
})
.catch((error) => {
console.log(error);
if (error?.type === 'invalid-json') {
throw new Error(
'Did not receive JSON response - are you sure this URL supports .json?dcr requests?',
Expand Down Expand Up @@ -63,24 +70,39 @@ exports.parseURL = parseURL;

/** @type {import('webpack-dev-server').ExpressRequestHandler} */
exports.getContentFromURLMiddleware = async (req, res, next) => {
const sourceURL = parseURL(req.originalUrl);

if (sourceURL) {
if (
req.path.startsWith('/AMP') &&
sourceURL.hostname === 'www.theguardian.com'
) {
res.redirect(
req.path.replace('www.theguardian.com', 'amp.theguardian.com'),
);
}
console.log(`req.path: ${req.path}`);
if (req.path === '/EditionsCrossword') {
const sourceURL = parseURL(req.originalUrl);
const sourceURLNew = new URL(
`http://localhost:9000${sourceURL.pathname}`,
);
console.log(`sourceUrl: ${sourceURL}`);
console.log(sourceURL);
req.body = await getContentFromURL(sourceURL, req.headers);
console.log(`req.body: ${req.body}`);
console.log(req.body);
} else {
const sourceURL = parseURL(req.originalUrl);
if (sourceURL) {
if (
req.path.startsWith('/AMP') &&
sourceURL.hostname === 'www.theguardian.com'
) {
res.redirect(
req.path.replace(
'www.theguardian.com',
'amp.theguardian.com',
),
);
}

try {
req.body = await getContentFromURL(sourceURL, req.headers);
} catch (error) {
console.error(error);
next(error);
try {
req.body = await getContentFromURL(sourceURL, req.headers);
} catch (error) {
console.error(error);
next(error);
}
}
next();
}
next();
};
33 changes: 33 additions & 0 deletions dotcom-rendering/src/server/render.sports.web.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
import { FELiveScoresType } from '../types/sports';
import { LiveScoresPage } from '../components/Football/LiveScoresPage';
import { renderToStringWithEmotion } from '../lib/emotion';
import { getPathFromManifest } from '../lib/assets';
import { polyfillIO } from '../lib/polyfill.io';
import { isString } from '@guardian/libs';

interface Props {
sports: FELiveScoresType;
}

export const renderSportsHtml = ({
sports,
}: Props): { html: string; prefetchScripts: string[] } => {
const { html } = renderToStringWithEmotion(
<LiveScoresPage liveScores={sports} />,
);

/**
* The highest priority scripts.
* These scripts have a considerable impact on site performance.
* Only scripts critical to application execution may go in here.
* Please talk to the dotcom platform team before adding more.
* Scripts will be executed in the order they appear in this array
*/
const prefetchScripts = [
polyfillIO,
getPathFromManifest('client.web', 'frameworks.js'),
getPathFromManifest('client.web', 'index.js'),
].filter(isString);

return { html: html, prefetchScripts };
};
6 changes: 6 additions & 0 deletions dotcom-rendering/src/server/server.dev.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import {
handleTagPage,
handleTagPageJson,
} from './handler.front.web';
import { handleSports } from './handler.sports';

/** article URLs contain a part that looks like “2022/nov/25” */
const ARTICLE_URL = /\/\d{4}\/[a-z]{3}\/\d{2}\//;
Expand Down Expand Up @@ -52,8 +53,11 @@ const editionalisefront = (url: string): string => {
// for more info
export const devServer = (): Handler => {
return (req, res, next) => {
console.log(`req.path: ${req.path}`);
const path = req.path.split('/')[1];

console.log(`path: ${path}`);

// handle urls with the ?url=… query param
const sourceUrl = req.url.split('?url=')[1];
if (path && sourceUrl) {
Expand Down Expand Up @@ -91,6 +95,8 @@ export const devServer = (): Handler => {
return handleAppsBlocks(req, res, next);
case 'EditionsCrossword':
return handleEditionsCrossword(req, res, next);
case 'Sports':
return handleSports(req, res, next);
default: {
// Do not redirect assets urls
if (req.url.match(ASSETS_URL)) return next();
Expand Down
Loading

0 comments on commit f699361

Please sign in to comment.