always upsert library items
This commit is contained in:
@ -17,6 +17,10 @@ export const getColumns = <T>(repository: Repository<T>): (keyof T)[] => {
|
||||
) as (keyof T)[]
|
||||
}
|
||||
|
||||
export const getColumnsDbName = <T>(repository: Repository<T>): string[] => {
|
||||
return repository.metadata.columns.map((col) => col.databaseName)
|
||||
}
|
||||
|
||||
export const setClaims = async (
|
||||
manager: EntityManager,
|
||||
uid = '00000000-0000-0000-0000-000000000000',
|
||||
|
||||
@ -1,5 +1,16 @@
|
||||
import { DeepPartial } from 'typeorm'
|
||||
import { getColumnsDbName } from '.'
|
||||
import { appDataSource } from '../data_source'
|
||||
import { LibraryItem } from '../entity/library_item'
|
||||
import { keysToCamelCase, wordsCount } from '../utils/helpers'
|
||||
import { logger } from '../utils/logger'
|
||||
|
||||
const convertToLibraryItem = (item: DeepPartial<LibraryItem>) => {
|
||||
return {
|
||||
...item,
|
||||
wordCount: item.wordCount ?? wordsCount(item.readableContent || ''),
|
||||
}
|
||||
}
|
||||
|
||||
export const libraryItemRepository = appDataSource
|
||||
.getRepository(LibraryItem)
|
||||
@ -20,6 +31,33 @@ export const libraryItemRepository = appDataSource
|
||||
return this.countBy({ createdAt })
|
||||
},
|
||||
|
||||
async upsertLibraryItem(item: DeepPartial<LibraryItem>) {
|
||||
// overwrites columns except id and slug
|
||||
const overwrites = getColumnsDbName(this).filter(
|
||||
(column) => !['id', 'slug'].includes(column)
|
||||
)
|
||||
const hashedUrl = 'md5(original_url)'
|
||||
|
||||
const [query, params] = this.createQueryBuilder()
|
||||
.insert()
|
||||
.into(LibraryItem)
|
||||
.values(convertToLibraryItem(item))
|
||||
.orUpdate(overwrites, ['user_id', hashedUrl], {
|
||||
skipUpdateIfNoValuesChanged: true,
|
||||
})
|
||||
.returning('*')
|
||||
.getQueryAndParameters()
|
||||
|
||||
// this is a workaround for the typeorm bug which quotes the md5 function
|
||||
const newQuery = query.replace(`"${hashedUrl}"`, hashedUrl)
|
||||
const results = (await this.query(newQuery, params)) as never[]
|
||||
|
||||
// convert to camel case
|
||||
const newItem = keysToCamelCase(results[0]) as LibraryItem
|
||||
|
||||
return newItem
|
||||
},
|
||||
|
||||
createByPopularRead(name: string, userId: string) {
|
||||
return this.query(
|
||||
`
|
||||
|
||||
@ -74,7 +74,7 @@ import {
|
||||
import {
|
||||
batchDelete,
|
||||
batchUpdateLibraryItems,
|
||||
createLibraryItem,
|
||||
createOrUpdateLibraryItem,
|
||||
findLibraryItemById,
|
||||
findLibraryItemByUrl,
|
||||
findLibraryItemsByPrefix,
|
||||
@ -357,7 +357,7 @@ export const createArticleResolver = authorized<
|
||||
)
|
||||
} else {
|
||||
// create new item in database
|
||||
libraryItemToReturn = await createLibraryItem(
|
||||
libraryItemToReturn = await createOrUpdateLibraryItem(
|
||||
libraryItemToSave,
|
||||
uid,
|
||||
pubsub
|
||||
|
||||
@ -13,7 +13,7 @@ import { PageType, UploadFileStatus } from '../generated/graphql'
|
||||
import { authTrx } from '../repository'
|
||||
import { Claims } from '../resolvers/types'
|
||||
import {
|
||||
createLibraryItem,
|
||||
createOrUpdateLibraryItem,
|
||||
findLibraryItemById,
|
||||
findLibraryItemByUrl,
|
||||
restoreLibraryItem,
|
||||
@ -101,7 +101,7 @@ export function pageRouter() {
|
||||
if (item) {
|
||||
await restoreLibraryItem(item.id, claims.uid)
|
||||
} else {
|
||||
await createLibraryItem(
|
||||
await createOrUpdateLibraryItem(
|
||||
{
|
||||
originalUrl: signedUrl,
|
||||
id: clientRequestId,
|
||||
|
||||
@ -9,7 +9,7 @@ import { UploadFile } from '../../entity/upload_file'
|
||||
import { env } from '../../env'
|
||||
import { PageType, UploadFileStatus } from '../../generated/graphql'
|
||||
import { authTrx } from '../../repository'
|
||||
import { createLibraryItem } from '../../services/library_item'
|
||||
import { createOrUpdateLibraryItem } from '../../services/library_item'
|
||||
import { findNewsletterEmailByAddress } from '../../services/newsletters'
|
||||
import { updateReceivedEmail } from '../../services/received_emails'
|
||||
import {
|
||||
@ -170,7 +170,7 @@ export function emailAttachmentRouter() {
|
||||
: ContentReaderType.EPUB,
|
||||
}
|
||||
|
||||
const item = await createLibraryItem(itemToCreate, user.id)
|
||||
const item = await createOrUpdateLibraryItem(itemToCreate, user.id)
|
||||
|
||||
// update received email type
|
||||
await updateReceivedEmail(receivedEmailId, 'article', user.id)
|
||||
|
||||
@ -6,7 +6,7 @@ import {
|
||||
PreparedDocumentInput,
|
||||
} from '../../generated/graphql'
|
||||
import { createAndSaveLabelsInLibraryItem } from '../../services/labels'
|
||||
import { createLibraryItem } from '../../services/library_item'
|
||||
import { createOrUpdateLibraryItem } from '../../services/library_item'
|
||||
import { parsedContentToLibraryItem } from '../../services/save_page'
|
||||
import { cleanUrl, generateSlug } from '../../utils/helpers'
|
||||
import { createThumbnailUrl } from '../../utils/imageproxy'
|
||||
@ -123,7 +123,7 @@ export function followingServiceRouter() {
|
||||
state: ArticleSavingRequestStatus.ContentNotFetched,
|
||||
})
|
||||
|
||||
const newItem = await createLibraryItem(itemToSave, userId)
|
||||
const newItem = await createOrUpdateLibraryItem(itemToSave, userId)
|
||||
logger.info('feed item saved in following')
|
||||
|
||||
// save RSS label in the item
|
||||
|
||||
@ -18,7 +18,7 @@ import {
|
||||
import { logger } from '../utils/logger'
|
||||
import {
|
||||
countByCreatedAt,
|
||||
createLibraryItem,
|
||||
createOrUpdateLibraryItem,
|
||||
findLibraryItemByUrl,
|
||||
updateLibraryItem,
|
||||
} from './library_item'
|
||||
@ -118,7 +118,7 @@ export const createPageSaveRequest = async ({
|
||||
logger.info('libraryItem does not exist', { url })
|
||||
|
||||
// create processing item
|
||||
libraryItem = await createLibraryItem(
|
||||
libraryItem = await createOrUpdateLibraryItem(
|
||||
{
|
||||
id: articleSavingRequestId,
|
||||
user: { id: userId },
|
||||
|
||||
@ -14,15 +14,9 @@ import { LibraryItem, LibraryItemState } from '../entity/library_item'
|
||||
import { BulkActionType, InputMaybe, SortParams } from '../generated/graphql'
|
||||
import { createPubSubClient, EntityType } from '../pubsub'
|
||||
import { redisDataSource } from '../redis_data_source'
|
||||
import {
|
||||
authTrx,
|
||||
getColumns,
|
||||
isUniqueViolation,
|
||||
queryBuilderToRawSql,
|
||||
} from '../repository'
|
||||
import { authTrx, getColumns, queryBuilderToRawSql } from '../repository'
|
||||
import { libraryItemRepository } from '../repository/library_item'
|
||||
import { setRecentlySavedItemInRedis, wordsCount } from '../utils/helpers'
|
||||
import { logger } from '../utils/logger'
|
||||
import { setRecentlySavedItemInRedis } from '../utils/helpers'
|
||||
import { parseSearchQuery } from '../utils/search'
|
||||
import { addLabelsToLibraryItem } from './labels'
|
||||
|
||||
@ -831,74 +825,44 @@ export const createLibraryItems = async (
|
||||
)
|
||||
}
|
||||
|
||||
export const createLibraryItem = async (
|
||||
export const createOrUpdateLibraryItem = async (
|
||||
libraryItem: DeepPartial<LibraryItem>,
|
||||
userId: string,
|
||||
pubsub = createPubSubClient(),
|
||||
skipPubSub = false
|
||||
): Promise<LibraryItem> => {
|
||||
if (!libraryItem.originalUrl) {
|
||||
throw new Error('Original url is required')
|
||||
const newLibraryItem = await authTrx(
|
||||
async (tx) =>
|
||||
tx.withRepository(libraryItemRepository).upsertLibraryItem(libraryItem),
|
||||
undefined,
|
||||
userId
|
||||
)
|
||||
|
||||
// set recently saved item in redis if redis is enabled
|
||||
if (redisDataSource.redisClient) {
|
||||
await setRecentlySavedItemInRedis(
|
||||
redisDataSource.redisClient,
|
||||
userId,
|
||||
newLibraryItem.originalUrl
|
||||
)
|
||||
}
|
||||
|
||||
try {
|
||||
const newLibraryItem = await authTrx(
|
||||
async (tx) =>
|
||||
tx.withRepository(libraryItemRepository).save({
|
||||
...libraryItem,
|
||||
wordCount:
|
||||
libraryItem.wordCount ??
|
||||
wordsCount(libraryItem.readableContent || ''),
|
||||
}),
|
||||
undefined,
|
||||
userId
|
||||
)
|
||||
logger.info('item created', { url: libraryItem.originalUrl })
|
||||
|
||||
// set recently saved item in redis if redis is enabled
|
||||
if (redisDataSource.redisClient) {
|
||||
await setRecentlySavedItemInRedis(
|
||||
redisDataSource.redisClient,
|
||||
userId,
|
||||
newLibraryItem.originalUrl
|
||||
)
|
||||
}
|
||||
|
||||
if (skipPubSub) {
|
||||
return newLibraryItem
|
||||
}
|
||||
|
||||
await pubsub.entityCreated<DeepPartial<LibraryItem>>(
|
||||
EntityType.PAGE,
|
||||
{
|
||||
...newLibraryItem,
|
||||
// don't send original content and readable content
|
||||
originalContent: undefined,
|
||||
readableContent: undefined,
|
||||
},
|
||||
userId
|
||||
)
|
||||
|
||||
if (skipPubSub) {
|
||||
return newLibraryItem
|
||||
} catch (error) {
|
||||
if (isUniqueViolation(error)) {
|
||||
logger.info('item already created', { url: libraryItem.originalUrl })
|
||||
|
||||
const existingItem = await findLibraryItemByUrl(
|
||||
libraryItem.originalUrl,
|
||||
userId
|
||||
)
|
||||
|
||||
if (!existingItem) {
|
||||
throw new Error(`Item not found for url: ${libraryItem.originalUrl}`)
|
||||
}
|
||||
|
||||
return existingItem
|
||||
}
|
||||
|
||||
logger.error('error creating item', error)
|
||||
throw error
|
||||
}
|
||||
|
||||
await pubsub.entityCreated<DeepPartial<LibraryItem>>(
|
||||
EntityType.PAGE,
|
||||
{
|
||||
...newLibraryItem,
|
||||
// don't send original content and readable content
|
||||
originalContent: undefined,
|
||||
readableContent: undefined,
|
||||
},
|
||||
userId
|
||||
)
|
||||
|
||||
return newLibraryItem
|
||||
}
|
||||
|
||||
export const findLibraryItemsByPrefix = async (
|
||||
|
||||
@ -5,7 +5,7 @@ import { Recommendation } from '../entity/recommendation'
|
||||
import { authTrx } from '../repository'
|
||||
import { logger } from '../utils/logger'
|
||||
import { createHighlights } from './highlights'
|
||||
import { createLibraryItem, findLibraryItemByUrl } from './library_item'
|
||||
import { createOrUpdateLibraryItem, findLibraryItemByUrl } from './library_item'
|
||||
|
||||
export const addRecommendation = async (
|
||||
item: LibraryItem,
|
||||
@ -39,7 +39,7 @@ export const addRecommendation = async (
|
||||
publishedAt: item.publishedAt,
|
||||
}
|
||||
|
||||
recommendedItem = await createLibraryItem(newItem, userId)
|
||||
recommendedItem = await createOrUpdateLibraryItem(newItem, userId)
|
||||
|
||||
const highlights = item.highlights
|
||||
?.filter((highlight) => highlightIds?.includes(highlight.id))
|
||||
|
||||
@ -17,7 +17,7 @@ import {
|
||||
} from '../utils/parser'
|
||||
import { createAndSaveLabelsInLibraryItem } from './labels'
|
||||
import {
|
||||
createLibraryItem,
|
||||
createOrUpdateLibraryItem,
|
||||
findLibraryItemByUrl,
|
||||
restoreLibraryItem,
|
||||
} from './library_item'
|
||||
@ -80,7 +80,7 @@ export const saveEmail = async (
|
||||
}
|
||||
|
||||
// start a transaction to create the library item and update the received email
|
||||
const newLibraryItem = await createLibraryItem(
|
||||
const newLibraryItem = await createOrUpdateLibraryItem(
|
||||
{
|
||||
user: { id: input.userId },
|
||||
slug,
|
||||
|
||||
@ -1,6 +1,5 @@
|
||||
import { Readability } from '@omnivore/readability'
|
||||
import { DeepPartial } from 'typeorm'
|
||||
import { QueryDeepPartialEntity } from 'typeorm/query-builder/QueryPartialEntity'
|
||||
import { Highlight } from '../entity/highlight'
|
||||
import { LibraryItem, LibraryItemState } from '../entity/library_item'
|
||||
import { User } from '../entity/user'
|
||||
@ -12,8 +11,6 @@ import {
|
||||
SavePageInput,
|
||||
SaveResult,
|
||||
} from '../generated/graphql'
|
||||
import { authTrx } from '../repository'
|
||||
import { libraryItemRepository } from '../repository/library_item'
|
||||
import { enqueueThumbnailJob } from '../utils/createTask'
|
||||
import {
|
||||
cleanUrl,
|
||||
@ -28,7 +25,7 @@ import { contentReaderForLibraryItem } from '../utils/uploads'
|
||||
import { createPageSaveRequest } from './create_page_save_request'
|
||||
import { createHighlight } from './highlights'
|
||||
import { createAndSaveLabelsInLibraryItem } from './labels'
|
||||
import { createLibraryItem, updateLibraryItem } from './library_item'
|
||||
import { createOrUpdateLibraryItem } from './library_item'
|
||||
|
||||
// where we can use APIs to fetch their underlying content.
|
||||
const FORCE_PUPPETEER_URLS = [
|
||||
@ -118,44 +115,15 @@ export const savePage = async (
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// check if the item already exists
|
||||
const existingLibraryItem = await authTrx((t) =>
|
||||
t
|
||||
.withRepository(libraryItemRepository)
|
||||
.findByUserIdAndUrl(user.id, input.url)
|
||||
// do not publish a pubsub event if the item is imported
|
||||
const newItem = await createOrUpdateLibraryItem(
|
||||
itemToSave,
|
||||
user.id,
|
||||
undefined,
|
||||
isImported
|
||||
)
|
||||
if (existingLibraryItem) {
|
||||
clientRequestId = existingLibraryItem.id
|
||||
slug = existingLibraryItem.slug
|
||||
|
||||
// we don't want to update an rss feed item if rss-feeder is tring to re-save it
|
||||
if (existingLibraryItem.subscription === input.rssFeedUrl) {
|
||||
return {
|
||||
clientRequestId,
|
||||
url: `${homePageURL()}/${user.profile.username}/${slug}`,
|
||||
}
|
||||
}
|
||||
|
||||
// update the item except for id and slug
|
||||
await updateLibraryItem(
|
||||
clientRequestId,
|
||||
{
|
||||
...itemToSave,
|
||||
id: undefined,
|
||||
slug: undefined,
|
||||
} as QueryDeepPartialEntity<LibraryItem>,
|
||||
user.id
|
||||
)
|
||||
} else {
|
||||
// do not publish a pubsub event if the item is imported
|
||||
const newItem = await createLibraryItem(
|
||||
itemToSave,
|
||||
user.id,
|
||||
undefined,
|
||||
isImported
|
||||
)
|
||||
clientRequestId = newItem.id
|
||||
}
|
||||
clientRequestId = newItem.id
|
||||
slug = newItem.slug
|
||||
|
||||
await createAndSaveLabelsInLibraryItem(
|
||||
clientRequestId,
|
||||
|
||||
@ -17,7 +17,7 @@ import {
|
||||
generateUploadSignedUrl,
|
||||
} from '../utils/uploads'
|
||||
import { validateUrl } from './create_page_save_request'
|
||||
import { createLibraryItem } from './library_item'
|
||||
import { createOrUpdateLibraryItem } from './library_item'
|
||||
|
||||
const isFileUrl = (url: string): boolean => {
|
||||
const parsedUrl = new URL(url)
|
||||
@ -122,7 +122,7 @@ export const uploadFile = async (
|
||||
// If we have a file:// URL, don't try to match it
|
||||
// and create a copy of the item, just create a
|
||||
// new item.
|
||||
const item = await createLibraryItem(
|
||||
const item = await createOrUpdateLibraryItem(
|
||||
{
|
||||
id: input.clientRequestId || undefined,
|
||||
originalUrl: isFileUrl(input.url) ? attachmentUrl : input.url,
|
||||
|
||||
@ -12,7 +12,7 @@ import { authTrx, getRepository, setClaims } from '../src/repository'
|
||||
import { highlightRepository } from '../src/repository/highlight'
|
||||
import { userRepository } from '../src/repository/user'
|
||||
import { createUser } from '../src/services/create_user'
|
||||
import { createLibraryItem } from '../src/services/library_item'
|
||||
import { createOrUpdateLibraryItem } from '../src/services/library_item'
|
||||
import { createDeviceToken } from '../src/services/user_device_tokens'
|
||||
import {
|
||||
bulkEnqueueUpdateLabels,
|
||||
@ -120,7 +120,7 @@ export const createTestLibraryItem = async (
|
||||
slug: 'test-with-omnivore',
|
||||
}
|
||||
|
||||
const createdItem = await createLibraryItem(item, userId)
|
||||
const createdItem = await createOrUpdateLibraryItem(item, userId)
|
||||
if (labels) {
|
||||
await saveLabelsInLibraryItem(labels, createdItem.id, userId)
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user