cspace-ui
Version:
CollectionSpace user interface for browsers
345 lines (284 loc) • 8.47 kB
JavaScript
import get from 'lodash/get';
import { getSearchResult, isSearchDirty, isSearchPending } from '../reducers';
import getSession from './cspace';
import { advancedSearchConditionToNXQL } from '../helpers/searchHelpers';
import {
ERR_UNKNOWN_RECORD_TYPE,
ERR_UNKNOWN_VOCABULARY,
ERR_INVALID_SORT,
ERR_API,
} from '../constants/errorCodes';
import {
CLEAR_SELECTED,
CLEAR_SEARCH_RESULTS,
SET_MOST_RECENT_SEARCH,
CREATE_EMPTY_SEARCH_RESULT,
SEARCH_STARTED,
SEARCH_FULFILLED,
SEARCH_REJECTED,
SET_ALL_RESULT_ITEMS_SELECTED,
SET_RESULT_ITEM_SELECTED,
DESELECT_RESULT_ITEM,
} from '../constants/actionCodes';
const toJS = obj => ((obj && typeof obj === 'object' && obj.toJS) ? obj.toJS() : obj);
const getSortParam = (config, searchDescriptor, columnSetName) => {
const sortSpec = searchDescriptor.getIn(['searchQuery', 'sort']);
const [sortColumnName, sortDir] = sortSpec.split(' ');
if (sortDir && sortDir !== 'desc') {
return null;
}
const column = get(config,
['recordTypes', searchDescriptor.get('recordType'), 'columns', columnSetName, sortColumnName]);
if (column && column.sortBy) {
return (column.sortBy + (sortDir ? ' DESC' : ''));
}
return null;
};
export const clearSelected = searchName => ({
type: CLEAR_SELECTED,
meta: {
searchName,
},
});
export const clearSearchResults = searchName => ({
type: CLEAR_SEARCH_RESULTS,
meta: {
searchName,
},
});
export const search = (config, searchName, searchDescriptor, listType = 'common', columnSetName = 'default') =>
(dispatch, getState) => {
const recordType = searchDescriptor.get('recordType');
const vocabulary = searchDescriptor.get('vocabulary');
const csid = searchDescriptor.get('csid');
const subresource = searchDescriptor.get('subresource');
const searchQuery = searchDescriptor.get('searchQuery');
if (
isSearchPending(getState(), searchName, searchDescriptor) ||
(
getSearchResult(getState(), searchName, searchDescriptor) &&
!isSearchDirty(getState(), searchName)
)
) {
// There's already a result for this search. Just set this search to be the most recent.
dispatch({
type: SET_MOST_RECENT_SEARCH,
meta: {
searchName,
searchDescriptor,
},
});
return Promise.resolve();
}
const listTypeConfig = config.listTypes[listType];
dispatch({
type: SEARCH_STARTED,
meta: {
listTypeConfig,
searchName,
searchDescriptor,
},
});
const recordTypeConfig = config.recordTypes[recordType];
if (!recordTypeConfig) {
dispatch({
type: SEARCH_REJECTED,
payload: {
code: ERR_UNKNOWN_RECORD_TYPE,
},
meta: {
searchName,
searchDescriptor,
},
});
return Promise.reject();
}
const recordTypeServicePath = get(recordTypeConfig, ['serviceConfig', 'servicePath']);
if (!recordTypeServicePath) {
dispatch({
type: SEARCH_REJECTED,
payload: {
code: ERR_UNKNOWN_RECORD_TYPE,
},
meta: {
searchName,
searchDescriptor,
},
});
return Promise.reject();
}
const vocabularyServicePath = vocabulary
? get(recordTypeConfig, ['vocabularies', vocabulary, 'serviceConfig', 'servicePath'])
: null;
if (vocabulary && !vocabularyServicePath) {
dispatch({
type: SEARCH_REJECTED,
payload: {
code: ERR_UNKNOWN_VOCABULARY,
},
meta: {
searchName,
searchDescriptor,
},
});
return Promise.reject();
}
// Check for conditions where we just want to create an empty result.
// These happen when a new record is being created.
if (
// A related record search for an empty csid.
(searchQuery.get('rel') === '') ||
// A subresource query without a csid.
(typeof subresource !== 'undefined' && !searchDescriptor.get('csid'))
) {
dispatch({
type: CREATE_EMPTY_SEARCH_RESULT,
meta: {
listTypeConfig,
searchName,
searchDescriptor,
},
});
return Promise.resolve();
}
const nxql = advancedSearchConditionToNXQL(recordTypeConfig.fields, searchQuery.get('as'));
const requestConfig = {
params: {
as: nxql ? `(${nxql})` : undefined,
csid: searchQuery.get('csid'),
doctype: searchQuery.get('doctype'),
kw: searchQuery.get('kw'),
mkRtSbj: searchQuery.get('mkRtSbj'),
mode: toJS(searchQuery.get('mode')),
pgNum: searchQuery.get('p'),
pgSz: searchQuery.get('size'),
rtSbj: searchQuery.get('rel'),
rtPredicate: searchQuery.get('relType'),
sn: searchQuery.get('sn'), // accounts screen name
wf_deleted: false,
},
};
if (searchQuery.get('sort')) {
const sortParam = getSortParam(config, searchDescriptor, columnSetName);
if (!sortParam) {
dispatch({
type: SEARCH_REJECTED,
payload: {
code: ERR_INVALID_SORT,
},
meta: {
searchName,
searchDescriptor,
},
});
return Promise.reject();
}
requestConfig.params.sortBy = sortParam;
}
const pathParts = [recordTypeServicePath];
if (vocabularyServicePath) {
pathParts.push(vocabularyServicePath);
pathParts.push('items');
}
if (csid) {
pathParts.push(csid);
}
if (subresource) {
const subresourceConfig = config.subresources[subresource];
const subresourceServicePath = subresourceConfig.serviceConfig.servicePath;
pathParts.push(subresourceServicePath);
}
const path = pathParts.join('/');
return getSession().read(path, requestConfig)
// Insert an artificial delay for testing.
// .then(response => new Promise((resolve) => {
// window.setTimeout(() => {
// resolve(response);
// }, 1000);
// }))
.then(response => dispatch({
type: SEARCH_FULFILLED,
payload: response,
meta: {
listTypeConfig,
searchName,
searchDescriptor,
},
}))
.catch(error => dispatch({
type: SEARCH_REJECTED,
payload: {
code: ERR_API,
error,
},
meta: {
searchName,
searchDescriptor,
},
}));
};
export const searchCsid = (config, recordType, csid) => () => {
const requestConfig = {
params: {
as: `(ecm:name = "${csid}")`,
pgSz: 1,
wf_deleted: false,
},
};
const recordTypeConfig = config.recordTypes[recordType];
if (!recordTypeConfig) {
return Promise.reject();
}
const recordTypeServicePath = get(recordTypeConfig, ['serviceConfig', 'servicePath']);
if (!recordTypeServicePath) {
return Promise.reject();
}
const pathParts = [recordTypeServicePath];
const serviceType = get(recordTypeConfig, ['serviceConfig', 'serviceType']);
if (serviceType === 'authority') {
const vocabularyServicePath =
get(recordTypeConfig, ['vocabularies', 'all', 'serviceConfig', 'servicePath']);
if (!vocabularyServicePath) {
return Promise.reject();
}
pathParts.push(vocabularyServicePath);
pathParts.push('items');
}
const path = pathParts.join('/');
return getSession().read(path, requestConfig);
};
export const setResultItemSelected =
(config, searchName, searchDescriptor, listType = 'common', index, isSelected) => {
const listTypeConfig = config.listTypes[listType];
return {
type: SET_RESULT_ITEM_SELECTED,
payload: isSelected,
meta: {
listTypeConfig,
searchName,
searchDescriptor,
index,
},
};
};
export const setAllResultItemsSelected =
(config, searchName, searchDescriptor, listType = 'common', isSelected, filter) => {
const listTypeConfig = config.listTypes[listType];
return {
type: SET_ALL_RESULT_ITEMS_SELECTED,
payload: isSelected,
meta: {
filter,
listTypeConfig,
searchName,
searchDescriptor,
},
};
};
export const deselectResultItem = (searchName, csid) => ({
type: DESELECT_RESULT_ITEM,
meta: {
searchName,
csid,
},
});