Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

AfO Register #398

Merged
merged 32 commits into from
Nov 28, 2023
Merged
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
6b090e8
Implement repository & tests (WiP)
khoidt Oct 25, 2023
bc7e1fe
Add AfO service, tests & bibliography routes and tabs
khoidt Oct 30, 2023
fd947c1
Add tests & format
khoidt Oct 31, 2023
2749a66
Impelment AfO Register search forms
khoidt Oct 31, 2023
ced19e9
Implement search & async options (WiP)
khoidt Nov 2, 2023
5d3aacd
Update & fix tests
khoidt Nov 7, 2023
526e661
Implement suggestions (WiP)
khoidt Nov 13, 2023
029b34d
Adjust search, sorting & form (WiP)
khoidt Nov 15, 2023
104f699
Merge remote-tracking branch 'origin/master' into afo-register
khoidt Nov 15, 2023
81b8fce
Correct Markdown string output
khoidt Nov 15, 2023
162e4fd
Update queries & interface, refactor
khoidt Nov 16, 2023
89b0fdf
Refactor search form
khoidt Nov 21, 2023
b556376
Add AfO Register & bibliography intro, style
khoidt Nov 21, 2023
2810b19
Merge remote-tracking branch 'origin/master' into afo-register
khoidt Nov 22, 2023
bf4a4b2
Implement AfO Reg. display in Fragmentarium
khoidt Nov 24, 2023
71a947a
Implement fragment injection, link, style, refactor & update tests
khoidt Nov 26, 2023
669b57a
Merge remote-tracking branch 'origin/master' into afo-register
khoidt Nov 26, 2023
5d30646
Fix CSS style
khoidt Nov 26, 2023
48f9a91
Refactor
khoidt Nov 26, 2023
3a26c4b
Refactor more
khoidt Nov 26, 2023
75d2217
Add missing space
khoidt Nov 26, 2023
b16cc16
Update repository & service tests
khoidt Nov 27, 2023
6b06553
Add factory & display tests
khoidt Nov 27, 2023
c5832c5
Add search results display test
khoidt Nov 27, 2023
8f01369
Add text select tests
khoidt Nov 27, 2023
fa10ed7
Add fragment records test & fix display
khoidt Nov 27, 2023
0d95895
Add & update tests
khoidt Nov 27, 2023
599ad57
Commit after git glitch
khoidt Nov 27, 2023
7a929a3
Extend repository tests
khoidt Nov 27, 2023
93adcf7
Update bibliography tests
khoidt Nov 27, 2023
dcee39c
Add repository test
khoidt Nov 28, 2023
43c19d8
Extend search form tests
khoidt Nov 28, 2023
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
41 changes: 41 additions & 0 deletions src/afo-register/application/AfoRegisterService.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
import { testDelegation, TestData } from 'test-support/utils'
import AfoRegisterRepository from 'afo-register/infrastructure/AfoRegisterRepository'
import AfoRegisterRecord from 'afo-register/domain/Record'
import { stringify } from 'query-string'
import ApiClient from 'http/ApiClient'

jest.mock('http/ApiClient')
const apiClient = new (ApiClient as jest.Mock<jest.Mocked<ApiClient>>)()

const afoRegisterRepository = new AfoRegisterRepository(apiClient)

const resultStub = {
afoNumber: 'AfO 1',
page: '2',
text: 'some text',
textNumber: '5',
discussedBy: '',
discussedByNotes: '',
linesDiscussed: '',
}

const query = { afoNumber: resultStub.afoNumber, page: resultStub.page }
const entry = new AfoRegisterRecord(resultStub)

const testData: TestData<AfoRegisterRepository>[] = [
new TestData(
'search',
[
stringify({
afoNumber: 'AfO 1',
page: '2',
}),
],
apiClient.fetchJson,
[entry],
[`/afo-register?${stringify(query)}`, false],
Promise.resolve([resultStub])
),
]
describe('afoRegisterService', () =>
testDelegation(afoRegisterRepository, testData))
27 changes: 27 additions & 0 deletions src/afo-register/application/AfoRegisterService.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
import Promise from 'bluebird'
import AfoRegisterRecord, {
AfoRegisterRecordSuggestion,
} from 'afo-register/domain/Record'
import AfoRegisterRepository from 'afo-register/infrastructure/AfoRegisterRepository'

export interface afoRegisterSearch {
search(query: string): Promise<readonly AfoRegisterRecord[]>
}

export default class AfoRegisterService implements afoRegisterSearch {
private readonly afoRegisterRepository: AfoRegisterRepository

constructor(afoRegisterRepository: AfoRegisterRepository) {
this.afoRegisterRepository = afoRegisterRepository
}

search(query: string): Promise<readonly AfoRegisterRecord[]> {
return this.afoRegisterRepository.search(query)
}

searchSuggestions(
query: string
): Promise<readonly AfoRegisterRecordSuggestion[]> {
return this.afoRegisterRepository.searchSuggestions(query)
}
}
72 changes: 72 additions & 0 deletions src/afo-register/domain/Record.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
import AfoRegisterRecord from 'afo-register/domain/Record'
import { fragmentFactory } from 'test-support/fragment-fixtures'

describe('AfoRegisterRecord', () => {
const mockRecord = {
afoNumber: '123',
page: '456',
text: 'Sample text',
textNumber: '789',
linesDiscussed: 'Some lines',
discussedBy: 'John Doe',
discussedByNotes: 'Notes by John',
}

const fragment1 = fragmentFactory.build({
traditionalReferences: ['Sample text 789'],
number: 'Fragment001',
})

const fragment2 = fragmentFactory.build({
traditionalReferences: [],
number: 'Fragment002',
})

describe('constructor', () => {
describe('constructor', () => {
it('should initialize properties correctly', () => {
const record = new AfoRegisterRecord(mockRecord)
expect(record.afoNumber).toEqual('123')
expect(record.page).toEqual('456')
expect(record.text).toEqual('Sample text')
expect(record.textNumber).toEqual('789')
expect(record.linesDiscussed).toEqual('Some lines')
expect(record.discussedBy).toEqual('John Doe')
expect(record.discussedByNotes).toEqual('Notes by John')
})
})
})

describe('Converts to Markdown string', () => {
it('Returns the correct markdown string with fragments', () => {
Copy link

Choose a reason for hiding this comment

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

Similar blocks of code found in 2 locations. Consider refactoring.

const record = new AfoRegisterRecord(mockRecord)
const result = record.toMarkdownString([fragment1, fragment2])
expect(result).toEqual(
'Sample text 789(Fragment001), Some lines: John Doe Notes by John<small class="text-black-50 ml-3">123456</small>'
)
})

it('Handles sup tags correctly', () => {
const recordWithSup = new AfoRegisterRecord({
...mockRecord,
text: 'Sample^text^',
})
const result = recordWithSup.toMarkdownString([fragment1, fragment2])
expect(result).toContain('<sup>text</sup>')
})
})

describe('Finds link to fragment', () => {
it('should return the correct link to fragment', () => {
Copy link

Choose a reason for hiding this comment

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

Similar blocks of code found in 2 locations. Consider refactoring.

const record = new AfoRegisterRecord(mockRecord)
const result = record.findLinkToFragment([fragment1, fragment2])
expect(result).toEqual('Fragment001')
})

it('Returns empty string if no matching fragments', () => {
const record = new AfoRegisterRecord(mockRecord)
const result = record.findLinkToFragment([fragment2])
expect(result).toEqual('')
})
})
})
85 changes: 85 additions & 0 deletions src/afo-register/domain/Record.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
import { immerable } from 'immer'
import { Fragment } from 'fragmentarium/domain/fragment'

interface RecordData {
readonly afoNumber: string
readonly page: string
readonly text: string
readonly textNumber: string
readonly linesDiscussed?: string
readonly discussedBy?: string
readonly discussedByNotes?: string
}

export class AfoRegisterRecordSuggestion {
[immerable] = true

readonly text: string
readonly textNumbers: string[]

constructor({
text,
textNumbers,
}: {
readonly text: string
readonly textNumbers: string[]
}) {
this.text = text
this.textNumbers = textNumbers
}
}

export default class AfoRegisterRecord {
[immerable] = true

readonly afoNumber: string
readonly page: string
readonly text: string
readonly textNumber: string
readonly linesDiscussed?: string
readonly discussedBy?: string
readonly discussedByNotes?: string

constructor({
Copy link

Choose a reason for hiding this comment

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

Similar blocks of code found in 3 locations. Consider refactoring.

afoNumber,
page,
text,
textNumber,
linesDiscussed,
discussedBy,
discussedByNotes,
}: RecordData) {
this.afoNumber = afoNumber
this.page = page
this.text = text
this.textNumber = textNumber
this.linesDiscussed = linesDiscussed
this.discussedBy = discussedBy
this.discussedByNotes = discussedByNotes
}

toMarkdownString(fragments: Fragment[]): string {
let result = this.text + (this.textNumber ? ' ' + this.textNumber : '')
Copy link
Contributor

Choose a reason for hiding this comment

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

I recently made a little helper function padLeft in src/fragmentarium/domain/archaeology.ts because I had to do the x ? ' ' + x : '' thing over and over, which is then just padLeft(this.textNumber). If you like you can move it to common/utils.ts or something and use it here, it might lower the complexity a bit.

const linkToFragment = this.findLinkToFragment(fragments)
if (linkToFragment) result += `(${linkToFragment})`
if (this.linesDiscussed) result += ', ' + this.linesDiscussed
if (this.discussedBy) result += ': ' + this.discussedBy
if (this.discussedByNotes) result += ' ' + this.discussedByNotes

result += `<small class="text-black-50 ml-3">${
this.afoNumber + this.page
}</small>`

result = result.replace(/\^([^^]+)\^/g, '<sup>$1</sup>')
return result
}

findLinkToFragment(fragments: Fragment[]): string {
const reference = this.text + ' ' + this.textNumber
const matchingFragments = fragments.filter((fragment) =>
fragment.traditionalReferences.includes(reference)
)
if (matchingFragments.length === 0) return ''
return matchingFragments.map((fragment) => fragment.number).join(', ')
}
}
35 changes: 35 additions & 0 deletions src/afo-register/infrastructure/AfoRegisterRepository.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
import AfoRegisterRecord from 'afo-register/domain/Record'
import ApiClient from 'http/ApiClient'
import AfoRegisterRepository from './AfoRegisterRepository'
import Promise from 'bluebird'
import { testDelegation, TestData } from 'test-support/utils'

jest.mock('http/ApiClient')

const apiClient = new (ApiClient as jest.Mock<jest.Mocked<ApiClient>>)()
const afoRegisterRepository = new AfoRegisterRepository(apiClient)
const query = '{"afoNumber": "AfO 12", "page": "321"}'
const resultStub = {
afoNumber: 'AfO 12',
page: '321',
text: 'text',
textNumber: 'text number',
linesDiscussed: '',
discussedBy: '',
discussedByNotes: '',
}
const entry = new AfoRegisterRecord(resultStub)

const testData: TestData<AfoRegisterRepository>[] = [
new TestData(
'search',
[query],
apiClient.fetchJson,
[entry],
[`/afo-register?${query}`, false],
Promise.resolve([resultStub])
),
]

describe('AfoRegisterRepository', () =>
testDelegation(afoRegisterRepository, testData))
33 changes: 33 additions & 0 deletions src/afo-register/infrastructure/AfoRegisterRepository.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
import AfoRegisterRecord, {
AfoRegisterRecordSuggestion,
} from 'afo-register/domain/Record'
import Promise from 'bluebird'
import ApiClient from 'http/ApiClient'

function createAfoRegisterRecord(data) {
return new AfoRegisterRecord(data)
}

function createAfoRegisterRecordSuggestion(data) {
return new AfoRegisterRecordSuggestion(data)
}

export default class AfoRegisterRepository {
private readonly apiClient: ApiClient

constructor(apiClient: ApiClient) {
this.apiClient = apiClient
}

search(query: string): Promise<AfoRegisterRecord[]> {
Copy link

Choose a reason for hiding this comment

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

Similar blocks of code found in 2 locations. Consider refactoring.

return this.apiClient
.fetchJson(`/afo-register?${query}`, false)
.then((result) => result.map(createAfoRegisterRecord))
}

searchSuggestions(query: string): Promise<AfoRegisterRecordSuggestion[]> {
Copy link

Choose a reason for hiding this comment

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

Similar blocks of code found in 2 locations. Consider refactoring.

return this.apiClient
.fetchJson(`/afo-register/suggestions?text_query=${query}`, false)
.then((result) => result.map(createAfoRegisterRecordSuggestion))
}
}
60 changes: 60 additions & 0 deletions src/afo-register/ui/AfoRegisterSearch.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
import React from 'react'
import _ from 'lodash'

import withData from 'http/withData'

import AfoRegisterRecord from 'afo-register/domain/Record'
import AfoRegisterService from 'afo-register/application/AfoRegisterService'
import { LiteratureRedirectBox } from 'common/LiteratureRedirectBox'
import { AfoRegisterQuery } from './AfoRegisterSearchForm'
import { stringify } from 'query-string'
import MarkdownAndHtmlToHtml from 'common/MarkdownAndHtmlToHtml'

export const AfoRegisterRedirectBox = (
Copy link

Choose a reason for hiding this comment

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

Similar blocks of code found in 2 locations. Consider refactoring.

<LiteratureRedirectBox
authors="Hirsch, H.; Hunger, H.; Jursa, M.; Weszeli, M.; et al."
book="Archiv für Orientforschung (Register Assyriologie)"
notelink=""
subtitle="25 (1974/1977) – 54 (2021)"
note="By permission from the AfO Redaktion"
link="https://orientalistik.univie.ac.at/publikationen/afo/register/"
icon="pointer__hover my-2 fas fa-external-link-square-alt"
/>
)

function afoRegisterSearch({ data }: { data: readonly AfoRegisterRecord[] }) {
return (
<>
<ol className="afoRegisterSearch">
{data.map((record) => (
<li
key={record.afoNumber + record.page}
className="afoRegisterSearch__record"
>
<MarkdownAndHtmlToHtml
markdownAndHtml={record.toMarkdownString([])}
/>
</li>
))}
</ol>
{data.length > 0 && AfoRegisterRedirectBox}
</>
)
}

export default withData<
unknown,
{
afoRegisterService: AfoRegisterService
query: AfoRegisterQuery
},
readonly AfoRegisterRecord[]
>(
afoRegisterSearch,
(props) => props.afoRegisterService.search(stringify(props.query)),
{
watch: (props) => [props.query],
filter: (props) => !_.isEmpty(props.query),
defaultData: () => [],
}
)
Loading
Loading