partial implementation
This commit is contained in:
parent
99b8fbfdfb
commit
740a4f63a2
|
@ -20,6 +20,7 @@ import eu.dnetlib.app.directindex.errors.DirectIndexApiException;
|
|||
import eu.dnetlib.app.directindex.input.ResultEntry;
|
||||
import eu.dnetlib.app.directindex.mapping.SolrRecordMapper;
|
||||
import eu.dnetlib.app.directindex.service.DirectIndexService;
|
||||
import jakarta.servlet.http.HttpServletRequest;
|
||||
|
||||
@RestController("/api")
|
||||
@ConditionalOnProperty(value = "dnet.directindex.legacy.enabled", havingValue = "true", matchIfMissing = false)
|
||||
|
@ -34,18 +35,21 @@ public class LegacyApiController {
|
|||
private SolrRecordMapper solrRecordMapper;
|
||||
|
||||
@PostMapping("/results/feedObject")
|
||||
public String feedResult(@RequestBody final ResultEntry pub, @RequestParam(value = "commit", required = false, defaultValue = "true") final boolean commit)
|
||||
public String feedResult(@RequestBody final ResultEntry pub,
|
||||
@RequestParam(value = "commit", required = false, defaultValue = "true") final boolean commit,
|
||||
final HttpServletRequest req)
|
||||
throws DirectIndexApiException {
|
||||
|
||||
return service.prepareMetadataInsertion(pub);
|
||||
return service.prepareMetadataInsertion(pub, req.getRemoteAddr());
|
||||
|
||||
}
|
||||
|
||||
@DeleteMapping("/result/{openaireId}")
|
||||
public boolean deleteResultWithOpenaireId(@PathVariable(value = "openaireId") final String openaireId,
|
||||
@RequestParam(value = "commit", required = false, defaultValue = "true") final boolean commit) throws DirectIndexApiException {
|
||||
@RequestParam(value = "commit", required = false, defaultValue = "true") final boolean commit,
|
||||
final HttpServletRequest req) throws DirectIndexApiException {
|
||||
|
||||
service.prepareMetadataDeletion(openaireId);
|
||||
service.prepareMetadataDeletion(openaireId, req.getRemoteAddr());
|
||||
|
||||
return true;
|
||||
}
|
||||
|
@ -54,11 +58,12 @@ public class LegacyApiController {
|
|||
public boolean deleteResultWithOriginalId(
|
||||
@RequestParam(value = "originalId", required = true) final String originalId,
|
||||
@RequestParam(value = "collectedFromId", required = true) final String collectedFromId,
|
||||
@RequestParam(value = "commit", required = false, defaultValue = "true") final boolean commit) throws DirectIndexApiException {
|
||||
@RequestParam(value = "commit", required = false, defaultValue = "true") final boolean commit,
|
||||
final HttpServletRequest req) throws DirectIndexApiException {
|
||||
|
||||
final String openaireId = solrRecordMapper.calculateOpenaireId(originalId, collectedFromId);
|
||||
|
||||
service.prepareMetadataDeletion(openaireId);
|
||||
service.prepareMetadataDeletion(openaireId, req.getRemoteAddr());
|
||||
|
||||
return true;
|
||||
}
|
||||
|
|
|
@ -2,6 +2,7 @@ package eu.dnetlib.app.directindex.controllers;
|
|||
|
||||
import java.io.IOException;
|
||||
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.beans.factory.annotation.Value;
|
||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
|
||||
|
@ -9,6 +10,7 @@ import org.springframework.http.HttpHeaders;
|
|||
import org.springframework.http.HttpStatus;
|
||||
import org.springframework.http.MediaType;
|
||||
import org.springframework.http.ResponseEntity;
|
||||
import org.springframework.security.core.Authentication;
|
||||
import org.springframework.web.bind.annotation.DeleteMapping;
|
||||
import org.springframework.web.bind.annotation.ExceptionHandler;
|
||||
import org.springframework.web.bind.annotation.GetMapping;
|
||||
|
@ -30,7 +32,6 @@ import eu.dnetlib.app.directindex.errors.SwordException;
|
|||
import eu.dnetlib.app.directindex.input.ResultEntry;
|
||||
import eu.dnetlib.app.directindex.mapping.SolrRecordMapper;
|
||||
import eu.dnetlib.app.directindex.service.DirectIndexService;
|
||||
import eu.dnetlib.app.directindex.solr.SolrIndexClient;
|
||||
import eu.dnetlib.app.directindex.solr.SolrIndexClientFactory;
|
||||
import eu.dnetlib.app.directindex.sword.model.SwordServiceDocument;
|
||||
import eu.dnetlib.app.directindex.sword.model.SwordStatusDocument;
|
||||
|
@ -42,6 +43,8 @@ import jakarta.servlet.http.HttpServletResponse;
|
|||
@ConditionalOnProperty(value = "dnet.directindex.sword.enabled", havingValue = "true", matchIfMissing = false)
|
||||
public class SwordApiController {
|
||||
|
||||
private static final String UNKNOWN_USER = "UNKNOWN";
|
||||
|
||||
@Autowired
|
||||
private DirectIndexService service;
|
||||
|
||||
|
@ -75,7 +78,8 @@ public class SwordApiController {
|
|||
@RequestHeader(value = "Metadata-Format", defaultValue = "http://purl.org/net/sword/3.0/types/Metadata") final String mdFormat,
|
||||
@RequestHeader(value = "Packaging", defaultValue = "http://purl.org/net/sword/3.0/package/Binary") final String packaging,
|
||||
@RequestHeader("Slug") final String slug,
|
||||
@RequestBody final String json) throws SwordException {
|
||||
@RequestBody final String json,
|
||||
final Authentication authentication) throws SwordException {
|
||||
|
||||
// TODO: DIGEST Evaluate if the digest (md5) of the json string
|
||||
|
||||
|
@ -89,7 +93,7 @@ public class SwordApiController {
|
|||
responseHeaders.setLocation(null);
|
||||
|
||||
try {
|
||||
final String openaireId = service.prepareMetadataInsertion(parseMetadata(json));
|
||||
final String openaireId = service.prepareMetadataInsertion(parseMetadata(json), StringUtils.firstNonBlank(authentication.getName(), UNKNOWN_USER));
|
||||
|
||||
final SwordStatusDocument status = SwordStatusDocument.newInstance(baseUrl + "/api/sword/3.0", openaireId);
|
||||
|
||||
|
@ -101,15 +105,17 @@ public class SwordApiController {
|
|||
}
|
||||
|
||||
@GetMapping("/objects/{id}/metadata")
|
||||
public ResponseEntity<ResultEntry> getMetadata(@PathVariable final String id) throws SwordException {
|
||||
public ResponseEntity<ResultEntry> getMetadata(@PathVariable final String id, final Authentication authentication) throws SwordException {
|
||||
|
||||
final SolrIndexClient solr = solrIndexClientFactory.getClient();
|
||||
|
||||
final SolrRecord metadata = solr.findRecord(id);
|
||||
try {
|
||||
final SolrRecord metadata = solrIndexClientFactory.getClient().findRecord(id);
|
||||
|
||||
if (metadata == null) { throw new SwordException(SwordErrorType.NotFound); }
|
||||
|
||||
return new ResponseEntity<>(solRecordMapper.toResultEntry(metadata), HttpStatus.OK);
|
||||
} catch (final DirectIndexApiException e) {
|
||||
throw new SwordException(SwordErrorType.ContentMalformed);
|
||||
}
|
||||
}
|
||||
|
||||
@PutMapping("/objects/{id}/metadata")
|
||||
|
@ -120,21 +126,24 @@ public class SwordApiController {
|
|||
@RequestHeader("Digest") final String digest,
|
||||
@RequestHeader(value = "Metadata-Format", defaultValue = "http://purl.org/net/sword/3.0/types/Metadata") final String mdFormat,
|
||||
@PathVariable final String id,
|
||||
@RequestBody final ResultEntry result) throws SwordException {
|
||||
@RequestBody final ResultEntry result,
|
||||
final Authentication authentication) throws SwordException {
|
||||
|
||||
final SolrIndexClient solr = solrIndexClientFactory.getClient();
|
||||
try {
|
||||
if (!solrIndexClientFactory.getClient().existsRecord(id)) { throw new SwordException(SwordErrorType.NotFound); }
|
||||
|
||||
if (!solr.existsRecord(id)) { throw new SwordException(SwordErrorType.NotFound); }
|
||||
|
||||
service.prepareMetadataReplacement(id, result);
|
||||
service.prepareMetadataReplacement(id, result, StringUtils.firstNonBlank(authentication.getName(), UNKNOWN_USER));
|
||||
|
||||
return new ResponseEntity<>(HttpStatus.ACCEPTED);
|
||||
} catch (final DirectIndexApiException e) {
|
||||
throw new SwordException(SwordErrorType.ContentMalformed);
|
||||
}
|
||||
}
|
||||
|
||||
@DeleteMapping("/objects/{id}")
|
||||
public ResponseEntity<Void> deleteObject(@PathVariable final String id) {
|
||||
public ResponseEntity<Void> deleteObject(@PathVariable final String id, final Authentication authentication) {
|
||||
|
||||
service.prepareMetadataDeletion(id);
|
||||
service.prepareMetadataDeletion(id, StringUtils.firstNonBlank(authentication.getName(), UNKNOWN_USER));
|
||||
|
||||
return new ResponseEntity<>(HttpStatus.ACCEPTED);
|
||||
|
||||
|
|
|
@ -26,24 +26,24 @@ public class DirectIndexService {
|
|||
@Autowired
|
||||
private DatasourceManagerClient dsmClient;
|
||||
|
||||
public void prepareMetadataDeletion(final String openaireId) {
|
||||
public void prepareMetadataDeletion(final String openaireId, final String createdBy) {
|
||||
final PendingAction action = new PendingAction();
|
||||
|
||||
action.setId(openaireId);
|
||||
action.setOperation("DELETE");
|
||||
action.setCreatedBy("TODO"); // TODO
|
||||
action.setCreatedBy(createdBy);
|
||||
action.setCreationDate(OffsetDateTime.now());
|
||||
action.setExecutionDate(null);
|
||||
|
||||
pendingActionRepository.save(action);
|
||||
}
|
||||
|
||||
public void prepareMetadataReplacement(final String id, final ResultEntry document) {
|
||||
public void prepareMetadataReplacement(final String id, final ResultEntry document, final String createdBy) {
|
||||
// TODO Auto-generated method stub
|
||||
|
||||
}
|
||||
|
||||
public String prepareMetadataInsertion(final ResultEntry r) throws DirectIndexApiException {
|
||||
public String prepareMetadataInsertion(final ResultEntry r, final String createdBy) throws DirectIndexApiException {
|
||||
final PendingAction info = new PendingAction();
|
||||
|
||||
if (StringUtils.isNotBlank(r.getOpenaireId())) {
|
||||
|
@ -71,7 +71,7 @@ public class DirectIndexService {
|
|||
info.setId(r.getOpenaireId());
|
||||
info.setBody(r.toJson());
|
||||
info.setType(r.getType());
|
||||
info.setCreatedBy("TODO"); // TODO
|
||||
info.setCreatedBy(createdBy);
|
||||
info.setCreationDate(OffsetDateTime.now());
|
||||
info.setExecutionDate(null);
|
||||
|
||||
|
|
|
@ -2,14 +2,23 @@ package eu.dnetlib.app.directindex.solr;
|
|||
|
||||
import java.io.IOException;
|
||||
import java.util.Arrays;
|
||||
import java.util.HashMap;
|
||||
import java.util.Iterator;
|
||||
import java.util.Map;
|
||||
import java.util.stream.Stream;
|
||||
|
||||
import org.apache.commons.logging.Log;
|
||||
import org.apache.commons.logging.LogFactory;
|
||||
import org.apache.solr.client.solrj.SolrServerException;
|
||||
import org.apache.solr.client.solrj.impl.CloudSolrClient;
|
||||
import org.apache.solr.client.solrj.response.QueryResponse;
|
||||
import org.apache.solr.client.solrj.util.ClientUtils;
|
||||
import org.apache.solr.common.SolrDocumentList;
|
||||
import org.apache.solr.common.SolrInputDocument;
|
||||
import org.apache.solr.common.params.MapSolrParams;
|
||||
import org.apache.solr.common.params.SolrParams;
|
||||
|
||||
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||
|
||||
import eu.dnetlib.app.directindex.errors.DirectIndexApiException;
|
||||
import eu.dnetlib.dhp.schema.solr.SolrRecord;
|
||||
|
@ -58,19 +67,48 @@ public class SolrIndexClient {
|
|||
}
|
||||
|
||||
protected SolrInputDocument prepareSolrDocument(final SolrRecord record) {
|
||||
// TODO (usare classe ccondivisa preparata da Claudio)
|
||||
// TODO (usare classe condivisa preparata da Claudio)
|
||||
|
||||
return null;
|
||||
|
||||
}
|
||||
|
||||
public SolrRecord findRecord(final String id) {
|
||||
// TODO Auto-generated method stub
|
||||
return null;
|
||||
public SolrRecord findRecord(final String id) throws DirectIndexApiException {
|
||||
try {
|
||||
final QueryResponse response = cloudSolrClient.query(queryParamsForId(id));
|
||||
final SolrDocumentList documents = response.getResults();
|
||||
|
||||
if (documents.isEmpty()) { return null; }
|
||||
|
||||
final String json = (String) documents.get(0).getFirstValue("__json");
|
||||
|
||||
return new ObjectMapper().readValue(json, SolrRecord.class);
|
||||
} catch (SolrServerException | IOException e) {
|
||||
throw new DirectIndexApiException("Error executing solr query", e);
|
||||
}
|
||||
}
|
||||
|
||||
public boolean existsRecord(final String id) {
|
||||
// TODO Auto-generated method stub
|
||||
return false;
|
||||
public boolean existsRecord(final String id) throws DirectIndexApiException {
|
||||
try {
|
||||
final QueryResponse response = cloudSolrClient.query(queryParamsForId(id));
|
||||
final SolrDocumentList documents = response.getResults();
|
||||
return !documents.isEmpty();
|
||||
} catch (SolrServerException | IOException e) {
|
||||
throw new DirectIndexApiException("Error executing solr query", e);
|
||||
}
|
||||
}
|
||||
|
||||
private SolrParams queryParamsForId(final String id) {
|
||||
|
||||
// TODO: verificare se esiste un campo __date
|
||||
|
||||
final Map<String, String> params = new HashMap<String, String>();
|
||||
|
||||
params.put("q", "objidentifier:" + ClientUtils.escapeQueryChars(id));
|
||||
params.put("fl", "objidentifier,__date,__json");
|
||||
params.put("sort", "__date desc");
|
||||
|
||||
return new MapSolrParams(params);
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -1,250 +0,0 @@
|
|||
package eu.dnetlib.app.directindex.solr;
|
||||
|
||||
import java.io.StringReader;
|
||||
import java.io.StringWriter;
|
||||
import java.util.HashMap;
|
||||
import java.util.Iterator;
|
||||
import java.util.LinkedList;
|
||||
import java.util.List;
|
||||
|
||||
import javax.xml.stream.XMLEventFactory;
|
||||
import javax.xml.stream.XMLEventReader;
|
||||
import javax.xml.stream.XMLEventWriter;
|
||||
import javax.xml.stream.XMLInputFactory;
|
||||
import javax.xml.stream.XMLOutputFactory;
|
||||
import javax.xml.stream.XMLStreamException;
|
||||
import javax.xml.stream.events.Namespace;
|
||||
import javax.xml.stream.events.StartElement;
|
||||
import javax.xml.stream.events.XMLEvent;
|
||||
|
||||
import org.apache.solr.common.SolrInputDocument;
|
||||
|
||||
/**
|
||||
* Optimized version of the document parser, drop in replacement of InputDocumentFactory.
|
||||
* <p>
|
||||
* Faster because:
|
||||
* <ul>
|
||||
* <li>Doesn't create a DOM for the full document
|
||||
* <li>Doesn't execute xpaths agains the DOM
|
||||
* <li>Quickly serialize the 'result' element directly in a string.
|
||||
* <li>Uses less memory: less pressure on GC and allows more threads to process this in parallel
|
||||
* </ul>
|
||||
* <p>
|
||||
* This class is fully reentrant and can be invoked in parallel.
|
||||
*
|
||||
* @author claudio
|
||||
*/
|
||||
public class StreamingInputDocumentFactory {
|
||||
|
||||
private static final String INDEX_FIELD_PREFIX = "__";
|
||||
|
||||
private static final String DS_VERSION = INDEX_FIELD_PREFIX + "dsversion";
|
||||
|
||||
private static final String DS_ID = INDEX_FIELD_PREFIX + "dsid";
|
||||
|
||||
private static final String RESULT = "result";
|
||||
|
||||
private static final String INDEX_RESULT = INDEX_FIELD_PREFIX + RESULT;
|
||||
|
||||
private static final String INDEX_RECORD_ID = INDEX_FIELD_PREFIX + "indexrecordidentifier";
|
||||
|
||||
private static final String DEFAULTDNETRESULT = "dnetResult";
|
||||
|
||||
private static final String TARGETFIELDS = "targetFields";
|
||||
|
||||
private static final String INDEX_RECORD_ID_ELEMENT = "indexRecordIdentifier";
|
||||
|
||||
private static final String ROOT_ELEMENT = "indexRecord";
|
||||
|
||||
private static final int MAX_FIELD_LENGTH = 25000;
|
||||
|
||||
private final ThreadLocal<XMLInputFactory> inputFactory = ThreadLocal
|
||||
.withInitial(XMLInputFactory::newInstance);
|
||||
|
||||
private final ThreadLocal<XMLOutputFactory> outputFactory = ThreadLocal
|
||||
.withInitial(XMLOutputFactory::newInstance);
|
||||
|
||||
private final ThreadLocal<XMLEventFactory> eventFactory = ThreadLocal
|
||||
.withInitial(XMLEventFactory::newInstance);
|
||||
|
||||
private final String version;
|
||||
|
||||
private String resultName = DEFAULTDNETRESULT;
|
||||
|
||||
public StreamingInputDocumentFactory(final String version) {
|
||||
this(version, DEFAULTDNETRESULT);
|
||||
}
|
||||
|
||||
public StreamingInputDocumentFactory(final String version, final String resultName) {
|
||||
this.version = version;
|
||||
this.resultName = resultName;
|
||||
}
|
||||
|
||||
public SolrInputDocument parseDocument(final String inputDocument) {
|
||||
|
||||
final StringWriter results = new StringWriter();
|
||||
final List<Namespace> nsList = new LinkedList<>();
|
||||
try {
|
||||
|
||||
final XMLEventReader parser = inputFactory.get().createXMLEventReader(new StringReader(inputDocument));
|
||||
|
||||
final SolrInputDocument indexDocument = new SolrInputDocument(new HashMap<>());
|
||||
|
||||
while (parser.hasNext()) {
|
||||
final XMLEvent event = parser.nextEvent();
|
||||
if (event != null && event.isStartElement()) {
|
||||
final String localName = event.asStartElement().getName().getLocalPart();
|
||||
|
||||
if (ROOT_ELEMENT.equals(localName)) {
|
||||
nsList.addAll(getNamespaces(event));
|
||||
} else if (INDEX_RECORD_ID_ELEMENT.equals(localName)) {
|
||||
final XMLEvent text = parser.nextEvent();
|
||||
final String recordId = getText(text);
|
||||
indexDocument.addField(INDEX_RECORD_ID, recordId);
|
||||
} else if (TARGETFIELDS.equals(localName)) {
|
||||
parseTargetFields(indexDocument, parser);
|
||||
} else if (resultName.equals(localName)) {
|
||||
copyResult(indexDocument, results, parser, nsList, resultName);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (version != null) {
|
||||
indexDocument.addField(DS_VERSION, version);
|
||||
}
|
||||
|
||||
if (!indexDocument.containsKey(INDEX_RECORD_ID)) { throw new IllegalStateException("cannot extract record ID from: " + inputDocument); }
|
||||
|
||||
return indexDocument;
|
||||
} catch (final XMLStreamException e) {
|
||||
throw new IllegalStateException(e);
|
||||
}
|
||||
}
|
||||
|
||||
private List<Namespace> getNamespaces(final XMLEvent event) {
|
||||
final List<Namespace> res = new LinkedList<>();
|
||||
final Iterator<Namespace> nsIter = event.asStartElement().getNamespaces();
|
||||
while (nsIter.hasNext()) {
|
||||
final Namespace ns = nsIter.next();
|
||||
res.add(ns);
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse the targetFields block and add fields to the solr document.
|
||||
*
|
||||
* @param indexDocument
|
||||
* @param parser
|
||||
* @throws XMLStreamException
|
||||
*/
|
||||
protected void parseTargetFields(
|
||||
final SolrInputDocument indexDocument,
|
||||
final XMLEventReader parser)
|
||||
throws XMLStreamException {
|
||||
|
||||
boolean hasFields = false;
|
||||
|
||||
while (parser.hasNext()) {
|
||||
final XMLEvent targetEvent = parser.nextEvent();
|
||||
if (targetEvent.isEndElement()
|
||||
&& TARGETFIELDS.equals(targetEvent.asEndElement().getName().getLocalPart())) {
|
||||
break;
|
||||
}
|
||||
|
||||
if (targetEvent.isStartElement()) {
|
||||
final String fieldName = targetEvent.asStartElement().getName().getLocalPart();
|
||||
final XMLEvent text = parser.nextEvent();
|
||||
|
||||
final String data = getText(text);
|
||||
|
||||
addField(indexDocument, fieldName, data);
|
||||
hasFields = true;
|
||||
}
|
||||
}
|
||||
|
||||
if (!hasFields) {
|
||||
indexDocument.clear();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Copy the /indexRecord/result element and children, preserving namespace declarations etc.
|
||||
*
|
||||
* @param indexDocument
|
||||
* @param results
|
||||
* @param parser
|
||||
* @param nsList
|
||||
* @throws XMLStreamException
|
||||
*/
|
||||
protected void copyResult(
|
||||
final SolrInputDocument indexDocument,
|
||||
final StringWriter results,
|
||||
final XMLEventReader parser,
|
||||
final List<Namespace> nsList,
|
||||
final String dnetResult)
|
||||
throws XMLStreamException {
|
||||
final XMLEventWriter writer = outputFactory.get().createXMLEventWriter(results);
|
||||
|
||||
for (final Namespace ns : nsList) {
|
||||
eventFactory.get().createNamespace(ns.getPrefix(), ns.getNamespaceURI());
|
||||
}
|
||||
|
||||
final StartElement newRecord = eventFactory.get().createStartElement("", null, RESULT, null, nsList.iterator());
|
||||
|
||||
// new root record
|
||||
writer.add(newRecord);
|
||||
|
||||
// copy the rest as it is
|
||||
while (parser.hasNext()) {
|
||||
final XMLEvent resultEvent = parser.nextEvent();
|
||||
|
||||
// TODO: replace with depth tracking instead of close tag tracking.
|
||||
if (resultEvent.isEndElement()
|
||||
&& resultEvent.asEndElement().getName().getLocalPart().equals(dnetResult)) {
|
||||
writer.add(eventFactory.get().createEndElement("", null, RESULT));
|
||||
break;
|
||||
}
|
||||
|
||||
writer.add(resultEvent);
|
||||
}
|
||||
writer.close();
|
||||
indexDocument.addField(INDEX_RESULT, results.toString());
|
||||
}
|
||||
|
||||
/**
|
||||
* Helper used to add a field to a solr doc. It avoids to add empy fields
|
||||
*
|
||||
* @param indexDocument
|
||||
* @param field
|
||||
* @param value
|
||||
*/
|
||||
private final void addField(
|
||||
final SolrInputDocument indexDocument,
|
||||
final String field,
|
||||
final String value) {
|
||||
final String cleaned = value.trim();
|
||||
if (!cleaned.isEmpty()) {
|
||||
// log.info("\n\n adding field " + field.toLowerCase() + " value: " + cleaned + "\n");
|
||||
indexDocument.addField(field.toLowerCase(), cleaned);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Helper used to get the string from a text element.
|
||||
*
|
||||
* @param text
|
||||
* @return the
|
||||
*/
|
||||
protected final String getText(final XMLEvent text) {
|
||||
if (text.isEndElement()) {
|
||||
// text.asEndElement().getName().getLocalPart());
|
||||
return "";
|
||||
}
|
||||
|
||||
final String data = text.asCharacters().getData();
|
||||
if (data != null && data.length() > MAX_FIELD_LENGTH) { return data.substring(0, MAX_FIELD_LENGTH); }
|
||||
|
||||
return data;
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue