Skip to content
Snippets Groups Projects
Commit 9890e235 authored by Tarje Lavik's avatar Tarje Lavik
Browse files

Add KulturNav import of Concepts

parent d524ebf9
No related branches found
No related tags found
No related merge requests found
/* eslint-disable no-undef */ /* eslint-disable no-undef */
import React, {useReducer, useEffect} from 'react' import React, {useReducer, useEffect} from 'react'
import ReactPaginate from 'react-paginate' //import ReactPaginate from 'react-paginate'
// import fetch from 'unfetch' // import fetch from 'unfetch'
import Preview from './components/Preview' import Card from './components/Card'
import Search from './components/Search' import Search from './components/Search'
import styles from '../ImportTool.css' // import styles from '../ImportTool.css'
import {searchReducer} from './reducers/searchReducer' import {searchReducer} from './reducers/searchReducer'
import {chooseItem} from './apis' import {chooseItem} from './apis'
import {Box, Container, Grid, Flex, Text} from '@sanity/ui' import {Box, Container, Grid, Flex, Text} from '@sanity/ui'
const IMPORT_API_URL = 'https://kulturnav.org/api/search/' const IMPORT_API_URL = 'https://kulturnav.org/api/search/'
const GET_TYPES = 'Concept'
export const initialState = { export const initialState = {
sourceAPI: 'nb', sourceAPI: 'kn',
apiURL: 'https://kulturnav.org/api/search/', apiURL: IMPORT_API_URL,
loading: true, loading: true,
searchParameter: '', searchParameter: '*',
items: [], items: [],
page: 0, page: 0,
totalElements: 0, totalElements: 0,
limit: 30, max: 64,
errorMessage: null, errorMessage: null,
} }
const SearchNB = () => { const SearchNB = () => {
const [state, dispatch] = useReducer(searchReducer, initialState) const [state, dispatch] = useReducer(searchReducer, initialState)
/* useEffect(() => { useEffect(() => {
fetch(state.apiURL + new URLSearchParams({})) fetch(
`${state.apiURL}actualEntityType:${GET_TYPES},compoundName:${state.searchParameter}/${state.page}/${state.max}`
)
.then((response) => response.json()) .then((response) => response.json())
.then((jsonResponse) => { .then((jsonResponse) => {
dispatch({ dispatch({
...@@ -36,9 +39,9 @@ const SearchNB = () => { ...@@ -36,9 +39,9 @@ const SearchNB = () => {
totalElements: jsonResponse.length, totalElements: jsonResponse.length,
}) })
}) })
}, []) */ }, [])
const handlePageClick = (data) => { /* const handlePageClick = (data) => {
let selected = data.selected let selected = data.selected
let page = selected let page = selected
...@@ -48,7 +51,7 @@ const SearchNB = () => { ...@@ -48,7 +51,7 @@ const SearchNB = () => {
}) })
fetch( fetch(
state.apiURL + 'actualEntityType:Person%20OR%20Concept,compoundName:' + state.searchParameter state.apiURL + 'actualEntityType:${GET_TYPES},compoundName:' + state.searchParameter
? state.searchParameter ? state.searchParameter
: '' + new URLSearchParams({}), : '' + new URLSearchParams({}),
) )
...@@ -68,7 +71,7 @@ const SearchNB = () => { ...@@ -68,7 +71,7 @@ const SearchNB = () => {
}) })
} }
}) })
} } */
const search = (searchValue) => { const search = (searchValue) => {
// setSearchParameter(searchValue) // setSearchParameter(searchValue)
...@@ -79,10 +82,7 @@ const SearchNB = () => { ...@@ -79,10 +82,7 @@ const SearchNB = () => {
}) })
fetch( fetch(
IMPORT_API_URL + `${state.apiURL}actualEntityType:${GET_TYPES},compoundName:${searchValue}/0/${state.max}`
'actualEntityType:Person%20OR%20Concept,compoundName:' +
searchValue +
new URLSearchParams({}),
) )
.then((response) => response.json()) .then((response) => response.json())
.then((jsonResponse) => { .then((jsonResponse) => {
...@@ -102,8 +102,8 @@ const SearchNB = () => { ...@@ -102,8 +102,8 @@ const SearchNB = () => {
}) })
} }
const {searchParameter, items, totalElements, page, limit, errorMessage, loading} = state const {searchParameter, items, totalElements, page, max, errorMessage, loading} = state
console.log(items)
return ( return (
<Container width={5} paddingY={5}> <Container width={5} paddingY={5}>
<form> <form>
...@@ -115,13 +115,13 @@ const SearchNB = () => { ...@@ -115,13 +115,13 @@ const SearchNB = () => {
<Text flex={1} size={1}>{totalElements} result found</Text> <Text flex={1} size={1}>{totalElements} result found</Text>
</Box> </Box>
<Box marginBottom={3}> {/* <Box marginBottom={3}>
<ReactPaginate <ReactPaginate
previousLabel={'previous'} previousLabel={'previous'}
nextLabel={'next'} nextLabel={'next'}
breakLabel={'...'} breakLabel={'...'}
forcePage={page} forcePage={page}
pageCount={totalElements / limit} pageCount={totalElements / max}
marginPagesDisplayed={2} marginPagesDisplayed={2}
pageRangeDisplayed={3} pageRangeDisplayed={3}
containerClassName={styles.pagination} containerClassName={styles.pagination}
...@@ -132,7 +132,7 @@ const SearchNB = () => { ...@@ -132,7 +132,7 @@ const SearchNB = () => {
activeClassName={styles.active} activeClassName={styles.active}
onPageChange={handlePageClick} onPageChange={handlePageClick}
/> />
</Box> </Box> */}
<Grid columns={[3, 4, 4, 4]} gap={[1, 1, 2, 3]}> <Grid columns={[3, 4, 4, 4]} gap={[1, 1, 2, 3]}>
{loading && !errorMessage ? ( {loading && !errorMessage ? (
<span>loading... </span> <span>loading... </span>
...@@ -140,7 +140,7 @@ const SearchNB = () => { ...@@ -140,7 +140,7 @@ const SearchNB = () => {
<div className="errorMessage">{errorMessage}</div> <div className="errorMessage">{errorMessage}</div>
) : ( ) : (
items.map((item) => ( items.map((item) => (
<Preview key={item.id} item={item} searchValue={searchParameter} onClick={chooseItem} /> <Card key={item.uuid} item={item} searchValue={searchParameter} onClick={chooseItem} />
)) ))
)} )}
</Grid> </Grid>
......
import {nanoid} from 'nanoid' import {nanoid} from 'nanoid'
import {mapMediatypes} from './mapMediatypes'
import sanityClient from 'part:@sanity/base/client' import sanityClient from 'part:@sanity/base/client'
const client = sanityClient.withConfig({apiVersion: '2021-03-25'}) const client = sanityClient.withConfig({apiVersion: '2021-03-25'})
export const chooseItem = async (item) => { export const chooseItem = async (item) => {
// Get a 200x200px thumbnail. Maybe change to a bigger size based on thumbnail_custom.
const imageUrl = item._links.thumbnail_custom.href
function customImageSize(image, h, w) {
if (!image) {
console.error('No image input')
throw Error
}
const height = '600' || h
const width = '600' || w
const template = image.replace('{height}', height).replace('{width}', width)
return template
}
const types = mapMediatypes(item.metadata.mediaTypes)
const doc = { const doc = {
_type: 'HumanMadeObject', _type: 'Concept',
_id: `${item.id}`, _id: `${item.uuid}`,
accessState: 'open', accessState: 'open',
editorialState: 'published', editorialState: 'published',
license: label: {
item.accessInfo && item.accessInfo.isPublicDomain ...(item.caption.no ? {
? 'https://creativecommons.org/publicdomain/mark/1.0/' nor: item.caption.no
: 'https://rightsstatements.org/vocab/CNE/1.0/', } : null),
label: item.metadata.title, ...(item.caption.sv ? {
preferredIdentifier: item.id, swe: item.caption.sv
} : null)
},
/* preferredIdentifier: item.uuid,
identifiedBy: [ identifiedBy: [
{ {
_type: 'Identifier', _type: 'Identifier',
_key: nanoid(), _key: nanoid(),
content: item.id, content: item.uuid,
hasType: { hasType: {
_type: 'reference', _type: 'reference',
_key: nanoid(), _key: nanoid(),
_ref: 'de22df48-e3e7-47f2-9d29-cae1b5e4d728', _ref: 'de22df48-e3e7-47f2-9d29-cae1b5e4d728',
}, },
}, },
], ], */
hasCurrentOwner: [ /* hasType: types, */
{
_type: 'reference',
_key: nanoid(),
_ref: '37f7376a-c635-420b-8ec6-ec0fd4c4a55c',
},
],
subjectOfManifest: item._links.presentation.href,
hasType: types,
wasOutputOf: { wasOutputOf: {
_type: 'DataTransferEvent', _type: 'DataTransferEvent',
_key: nanoid(), _key: nanoid(),
...@@ -68,14 +46,14 @@ export const chooseItem = async (item) => { ...@@ -68,14 +46,14 @@ export const chooseItem = async (item) => {
_type: 'DigitalDevice', _type: 'DigitalDevice',
_key: nanoid(), _key: nanoid(),
/* _ref: nanoid(36), */ /* _ref: nanoid(36), */
label: 'api.nb.no', label: 'kulturnav.no/api',
}, },
}, },
} }
/* TODO /* TODO
Important to include iiif manifest in asset metadata as the asset could be reused else where in the dataset */ Important to include iiif manifest in asset metadata as the asset could be reused else where in the dataset */
const assetMeta = { /* const assetMeta = {
source: { source: {
// The source this image is from // The source this image is from
name: 'nb.no', name: 'nb.no',
...@@ -86,7 +64,7 @@ export const chooseItem = async (item) => { ...@@ -86,7 +64,7 @@ export const chooseItem = async (item) => {
}, },
description: item.metadata.title, description: item.metadata.title,
creditLine: 'From nb.no', creditLine: 'From nb.no',
} } */
const getImageBlob = async (url) => { const getImageBlob = async (url) => {
// eslint-disable-next-line no-undef // eslint-disable-next-line no-undef
...@@ -151,11 +129,20 @@ export const chooseItem = async (item) => { ...@@ -151,11 +129,20 @@ export const chooseItem = async (item) => {
} }
const createDoc = async (doc) => { const createDoc = async (doc) => {
const res = client.createIfNotExists(doc).then((result) => { const transaction = client.transaction()
console.log(`${result._id} was imported!`)
return result transaction.createOrReplace(doc)
})
return res transaction
.commit()
.then((res) => {
console.log(JSON.stringify(res, null, 2))
return res
})
.catch((err) => {
console.log('Transaction failed', err)
return err
})
} }
const setAssetRef = async (docID, assetID) => { const setAssetRef = async (docID, assetID) => {
...@@ -180,23 +167,21 @@ export const chooseItem = async (item) => { ...@@ -180,23 +167,21 @@ export const chooseItem = async (item) => {
} }
try { try {
const imageResonse = await getImageBlob(customImageSize(imageUrl)) /* const imageResonse = await getImageBlob(customImageSize(imageUrl))
const asset = await uploadImageBlob(imageResonse) const asset = await uploadImageBlob(imageResonse)
await patchAssetMeta(asset._id, assetMeta) await patchAssetMeta(asset._id, assetMeta) */
const document = await createDoc(doc) await createDoc(doc)
if (asset && document) { /* if (asset && document) {
await setAssetRef(document._id, asset._id) await setAssetRef(document._id, asset._id)
} } */
return { return {
success: true, success: true,
body: JSON.stringify(document, asset),
} }
} catch (err) { } catch (err) {
return { return {
success: false, success: false,
body: JSON.stringify(response.status, response.statusText),
} }
} }
} }
import React, {useState} from 'react' import React, {useState} from 'react'
import Button from 'part:@sanity/components/buttons/default' import {
import Card from 'part:@sanity/components/previews/card' Box,
import DefaultBadge from 'part:@sanity/components/badges/default' Card as SanityCard,
import DateBadge from '../../components/DateBadge' Heading,
import styled, {keyframes} from 'styled-components' Text,
Badge,
Button,
Inline,
Stack
} from '@sanity/ui'
import {RiDownloadLine} from 'react-icons/ri'
import {chooseItem} from '../apis'
const Preview = ({item, searchValue, onClick}) => { const Card = ({item}) => {
const [isFetching, setIsFetching] = useState(false) const [isFetching, setIsFetching] = useState(false)
const [isImported, setIsImported] = useState(false) const [isImported, setIsImported] = useState(false)
const [buttonLabel, setButtonLabel] = useState('Import') const [buttonLabel, setButtonLabel] = useState('Import')
...@@ -13,12 +20,11 @@ const Preview = ({item, searchValue, onClick}) => { ...@@ -13,12 +20,11 @@ const Preview = ({item, searchValue, onClick}) => {
const onChooseItem = async (item) => { const onChooseItem = async (item) => {
setIsFetching(true) setIsFetching(true)
setButtonLabel('...importing') setButtonLabel('...importing')
const importStatus = await onClick(item) const importStatus = await chooseItem(item)
if (!importStatus.success) { if (!importStatus.success) {
setIsFetching(false) setIsFetching(false)
setButtonLabel('Import failed!') setButtonLabel('Import failed!')
console.log(importStatus.body)
return return
} }
...@@ -28,10 +34,52 @@ const Preview = ({item, searchValue, onClick}) => { ...@@ -28,10 +34,52 @@ const Preview = ({item, searchValue, onClick}) => {
} }
return ( return (
<SanityCard style={{display: "flex", flexDirection: "column"}} key={item._uuid} padding={[2, 2, 3]} radius={2} shadow={1}>
{/* NOT EASILY AVAILABLE
<Box>
<img style={{width: "100%"}} src={item.hasThumbnail} />
</Box> */}
<Box style={{flexGrow: "1"}} marginY={3}>
<Heading size="1">
{item.caption.no || item.caption.sv || 'Manglende caption?'}
</Heading>
<Stack paddingY={2} space={3}>
<Inline space={2}>
<Badge tone="primary">{item.entityType}</Badge>
</Inline>
{item.description && (
<Text muted size={[1, 1, 2]}>
{item.description}
</Text>
)}
</Stack>
</Box>
<Stack style={{marginTop: "auto"}} style={{borderTop: "1px dotted gray"}} paddingTop={2} space={3}>
<Inline space={2}>
<Button
fontSize={[1, 1, 2]}
padding={[1, 1, 2]}
icon={RiDownloadLine}
text={buttonLabel}
mode={isImported ? 'ghost' : 'default'}
disabled={isFetching}
onClick={() => onChooseItem(item)}
/>
<a href={`https://kulturnav.org/${item.uuid}`} target="_blank" rel="noopener noreferrer">Åpne i Kulturnav</a>
</Inline>
</Stack>
</SanityCard>
)
}
export default Card
/* return (
<Container key={item.uuid}> <Container key={item.uuid}>
<Card <Card
title={item.caption.no} title={item.caption.no}
/* media={() => ( media={() => (
<img <img
src={ src={
item && item._links && item._links.thumbnail_large item && item._links && item._links.thumbnail_large
...@@ -39,7 +87,7 @@ const Preview = ({item, searchValue, onClick}) => { ...@@ -39,7 +87,7 @@ const Preview = ({item, searchValue, onClick}) => {
: '' : ''
} }
/> />
)} */ )}
> >
<p> <p>
{item.mediaTypes && {item.mediaTypes &&
...@@ -54,26 +102,5 @@ const Preview = ({item, searchValue, onClick}) => { ...@@ -54,26 +102,5 @@ const Preview = ({item, searchValue, onClick}) => {
View at Kulturnav View at Kulturnav
</a> </a>
</Card> </Card>
</Container> </Container> */
)
} \ No newline at end of file
const FadeIn = keyframes`
0% {
opacity: 0;
}
100% {
opacity: 1;
}
`
const Container = styled.div`
display: flex;
flex-flow: column;
width: 300px;
padding: 6px 0;
margin: 6px;
animation: ${FadeIn} 1.2s cubic-bezier(0.39, 0.575, 0.565, 1) both;
`
export default Preview
...@@ -54,8 +54,6 @@ export const chooseItem = async (uri) => { ...@@ -54,8 +54,6 @@ export const chooseItem = async (uri) => {
// Map type to Sanity types // Map type to Sanity types
const types = mapMediatypes([cleanJSON.type]) const types = mapMediatypes([cleanJSON.type])
/* TODO
Include iiif manifest in asset metadata as the asset could be reused elsewhere in the dataset */
const assetMeta = { const assetMeta = {
source: { source: {
// The source this image is from // The source this image is from
......
...@@ -157,7 +157,7 @@ const SearchNB = () => { ...@@ -157,7 +157,7 @@ const SearchNB = () => {
<div className="errorMessage">{errorMessage}</div> <div className="errorMessage">{errorMessage}</div>
) : ( ) : (
items.map((item) => ( items.map((item) => (
<Card key={item._id} item={item} searchValue={searchParameter} onClick={chooseItem} /> <Card key={item.id} item={item} searchValue={searchParameter} onClick={chooseItem} />
)) ))
)} )}
</Grid> </Grid>
......
...@@ -29,7 +29,7 @@ const Card = ({item, searchValue, onClick}) => { ...@@ -29,7 +29,7 @@ const Card = ({item, searchValue, onClick}) => {
return ( return (
<SanityCard <SanityCard
style={{display: "flex", flexDirection: "column"}} style={{display: "flex", flexDirection: "column"}}
key={item._id} key={item.id}
padding={[2, 2, 3]} padding={[2, 2, 3]}
radius={2} radius={2}
shadow={1} shadow={1}
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment