Skip to content

Commit

Permalink
Fix race condition with lighting update on staging
Browse files Browse the repository at this point in the history
  • Loading branch information
rakyi committed Jan 14, 2025
1 parent 7e95bbe commit 4c5d982
Show file tree
Hide file tree
Showing 2 changed files with 85 additions and 81 deletions.
2 changes: 1 addition & 1 deletion adminSiteServer/apiRouter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -240,7 +240,7 @@ getRouteNonIdempotentWithRWTransaction(
"/gdocs/:id",
getIndividualGdoc
)
putRouteWithRWTransaction(apiRouter, "/gdocs/:id", createOrUpdateGdoc)
apiRouter.put("/gdocs/:id", createOrUpdateGdoc)
deleteRouteWithRWTransaction(apiRouter, "/gdocs/:id", deleteGdoc)
postRouteWithRWTransaction(apiRouter, "/gdocs/:gdocId/setTags", setGdocTags)

Expand Down
164 changes: 84 additions & 80 deletions adminSiteServer/apiRoutes/gdocs.ts
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
import { getCanonicalUrl } from "@ourworldindata/components"
import {
GdocsContentSource,
DbInsertUser,
JsonError,
GDOCS_BASE_URL,
gdocUrlRegex,
OwidGdocPostInterface,
PostsGdocsLinksTableName,
PostsGdocsXImagesTableName,
PostsGdocsTableName,
Expand Down Expand Up @@ -84,103 +84,107 @@ export async function getIndividualGdoc(
}
}

/**
* Handles all four `GdocPublishingAction` cases
* - SavingDraft (no action)
* - Publishing (index and bake)
* - Updating (index and bake (potentially via lightning deploy))
* - Unpublishing (remove from index and bake)
*/
async function indexAndBakeGdocIfNeccesary(
trx: db.KnexReadWriteTransaction,
user: Required<DbInsertUser>,
prevGdoc:
| GdocPost
| GdocDataInsight
| GdocHomepage
| GdocAbout
| GdocAuthor,
nextGdoc: GdocPost | GdocDataInsight | GdocHomepage | GdocAbout | GdocAuthor
) {
const prevJson = prevGdoc.toJSON()
const nextJson = nextGdoc.toJSON()
const hasChanges = checkHasChanges(prevGdoc, nextGdoc)
const action = getPublishingAction(prevJson, nextJson)
const isGdocPost = checkIsGdocPostExcludingFragments(nextJson)

await match(action)
.with(GdocPublishingAction.SavingDraft, lodash.noop)
.with(GdocPublishingAction.Publishing, async () => {
if (isGdocPost) {
await indexIndividualGdocPost(
nextJson,
trx,
// If the gdoc is being published for the first time, prevGdoc.slug will be undefined
// In that case, we pass nextJson.slug to see if it has any page views (i.e. from WP)
prevGdoc.slug || nextJson.slug
)
}
await triggerStaticBuild(user, `${action} ${nextJson.slug}`)
})
.with(GdocPublishingAction.Updating, async () => {
if (isGdocPost) {
await indexIndividualGdocPost(nextJson, trx, prevGdoc.slug)
}
if (checkIsLightningUpdate(prevJson, nextJson, hasChanges)) {
await enqueueLightningChange(
user,
`Lightning update ${nextJson.slug}`,
nextJson.slug
)
} else {
await triggerStaticBuild(user, `${action} ${nextJson.slug}`)
}
})
.with(GdocPublishingAction.Unpublishing, async () => {
if (isGdocPost) {
await removeIndividualGdocPostFromIndex(nextJson)
}
await triggerStaticBuild(user, `${action} ${nextJson.slug}`)
})
.exhaustive()
}

/**
* Only supports creating a new empty Gdoc or updating an existing one. Does not
* support creating a new Gdoc from an existing one. Relevant updates will
* trigger a deploy.
*/
export async function createOrUpdateGdoc(
req: Request,
res: e.Response<any, Record<string, any>>,
trx: db.KnexReadWriteTransaction
res: e.Response<any, Record<string, any>>
) {
const { id } = req.params

if (isEmpty(req.body)) {
return createOrLoadGdocById(trx, id)
return await db.knexReadWriteTransaction(async (trx) => {
return await createOrLoadGdocById(trx, id)
})
}

const prevGdoc = await getAndLoadGdocById(trx, id)
if (!prevGdoc) throw new JsonError(`No Google Doc with id ${id} found`)
const user = res.locals.user
let nextGdoc:
| GdocPost
| GdocDataInsight
| GdocHomepage
| GdocAbout
| GdocAuthor
| undefined
let build: "full" | "lightning" | undefined
let buildMessage = ""

await db.knexReadWriteTransaction(async (trx) => {
const prevGdoc = await getAndLoadGdocById(trx, id)
if (!prevGdoc) throw new JsonError(`No Google Doc with id ${id} found`)

const nextGdoc = gdocFromJSON(req.body)
await nextGdoc.loadState(trx)
nextGdoc = gdocFromJSON(req.body)
await nextGdoc.loadState(trx)

await addImagesToContentGraph(trx, nextGdoc)
await addImagesToContentGraph(trx, nextGdoc)

await setLinksForGdoc(
trx,
nextGdoc.id,
nextGdoc.links,
nextGdoc.published
? GdocLinkUpdateMode.DeleteAndInsert
: GdocLinkUpdateMode.DeleteOnly
)
await setLinksForGdoc(
trx,
nextGdoc.id,
nextGdoc.links,
nextGdoc.published
? GdocLinkUpdateMode.DeleteAndInsert
: GdocLinkUpdateMode.DeleteOnly
)

await upsertGdoc(trx, nextGdoc)
await upsertGdoc(trx, nextGdoc)

await indexAndBakeGdocIfNeccesary(trx, res.locals.user, prevGdoc, nextGdoc)
const prevJson = prevGdoc.toJSON()
const nextJson = nextGdoc.toJSON()
const hasChanges = checkHasChanges(prevGdoc, nextGdoc)
const action = getPublishingAction(prevJson, nextJson)
const isGdocPost = checkIsGdocPostExcludingFragments(nextJson)

await match(action)
.with(GdocPublishingAction.SavingDraft, lodash.noop)
.with(GdocPublishingAction.Publishing, async () => {
if (isGdocPost) {
await indexIndividualGdocPost(
nextJson as OwidGdocPostInterface,
trx,
// If the gdoc is being published for the first time, prevGdoc.slug will be undefined
// In that case, we pass nextJson.slug to see if it has any page views (i.e. from WP)
prevGdoc.slug || nextJson.slug
)
}
build = "full"
buildMessage = `${action} ${nextJson.slug}`
})
.with(GdocPublishingAction.Updating, async () => {
if (isGdocPost) {
await indexIndividualGdocPost(nextJson, trx, prevGdoc.slug)
}
if (checkIsLightningUpdate(prevJson, nextJson, hasChanges)) {
build = "lightning"
buildMessage = `Lightning update ${nextJson.slug}`
} else {
build = "full"
buildMessage = `${action} ${nextJson.slug}`
}
})
.with(GdocPublishingAction.Unpublishing, async () => {
if (isGdocPost) {
await removeIndividualGdocPostFromIndex(nextJson)
}
build = "full"
buildMessage = `${action} ${nextJson.slug}`
})
.exhaustive()
})

if (!nextGdoc) return

// The build must be triggered after the transaction has been committed
// otherwise the deploy-queue process can get stale data from the DB.
// https://github.com/owid/owid-grapher/issues/3908
if (build === "full") {
await triggerStaticBuild(user, buildMessage)
} else if (build === "lightning") {
await enqueueLightningChange(user, buildMessage, nextGdoc.slug)
}

return nextGdoc
}
Expand Down

0 comments on commit 4c5d982

Please sign in to comment.