diff --git a/src/main/java/org/olat/core/util/pdf/PdfDocument.java b/src/main/java/org/olat/core/util/pdf/PdfDocument.java index 5f38d9177d6fca6e4ae2a9ea5e55aee4a3579ca3..ddd1e803caa2431f0677f9fac3d22c5c1b20f6b7 100644 --- a/src/main/java/org/olat/core/util/pdf/PdfDocument.java +++ b/src/main/java/org/olat/core/util/pdf/PdfDocument.java @@ -142,7 +142,7 @@ public class PdfDocument { PDFont textFont = bold ? fontBold : font; - text = text.replace('\n', ' ').replace('\r', ' '); + text = cleanString(text); List<String> lines = new ArrayList<>(); int lastSpace = -1; diff --git a/src/main/java/org/olat/course/nodes/cl/ui/CheckboxPDFExport.java b/src/main/java/org/olat/course/nodes/cl/ui/CheckboxPDFExport.java index 7c2956a8e493d8a773c2e01f4a06197b18157ffc..3c469aed89068fb212781b2449f55999a609ef8c 100644 --- a/src/main/java/org/olat/course/nodes/cl/ui/CheckboxPDFExport.java +++ b/src/main/java/org/olat/course/nodes/cl/ui/CheckboxPDFExport.java @@ -354,7 +354,7 @@ public class CheckboxPDFExport extends PdfDocument implements MediaResource { currentContentStream.setTextMatrix(Matrix.getRotateInstance(3 * (Math.PI / 2), textx + cellMargin, texty - cellMargin)); textx += colWidth; } - currentContentStream.showText(text); + currentContentStream.showText(cleanString(text)); currentContentStream.endText(); } diff --git a/src/main/java/org/olat/course/nodes/cl/ui/CheckedPDFExport.java b/src/main/java/org/olat/course/nodes/cl/ui/CheckedPDFExport.java index 1624f6349cfcd15120a8fb6bcd319d6326e57a51..2a0c5418fad966a982a54369f9128e97fbe401b3 100644 --- a/src/main/java/org/olat/course/nodes/cl/ui/CheckedPDFExport.java +++ b/src/main/java/org/olat/course/nodes/cl/ui/CheckedPDFExport.java @@ -315,7 +315,7 @@ public class CheckedPDFExport extends PdfDocument implements MediaResource { currentContentStream.beginText(); currentContentStream.setFont(font, fontSize); currentContentStream.newLineAtOffset(textx, texty - headerHeight + cellMargin); - currentContentStream.showText(text); + currentContentStream.showText(cleanString(text)); currentContentStream.endText(); textx += colWidth; } diff --git a/src/main/java/org/olat/modules/lecture/ui/export/LecturesBlockPDFExport.java b/src/main/java/org/olat/modules/lecture/ui/export/LecturesBlockPDFExport.java index 7b37f059b422ea1ccc76ed86bb3799b2c0aeb5ea..6a94c88efeec33ed5aba3065a54fb1f803edfe63 100644 --- a/src/main/java/org/olat/modules/lecture/ui/export/LecturesBlockPDFExport.java +++ b/src/main/java/org/olat/modules/lecture/ui/export/LecturesBlockPDFExport.java @@ -415,7 +415,7 @@ public class LecturesBlockPDFExport extends PdfDocument implements MediaResource //draw the content texty = currentY - 15; for (int i=offset; i<end; i++) { - String text = content[i].getName(); + String text = cleanString(content[i].getName()); if(text == null) continue; if(rowHeights[i] > rowHeight + 1) { diff --git a/src/main/java/org/olat/modules/lecture/ui/export/LecturesBlockSignaturePDFExport.java b/src/main/java/org/olat/modules/lecture/ui/export/LecturesBlockSignaturePDFExport.java index a65df821b1e842440f31423ca9b95c02575140c6..227a49997650879ecf4d5495788e911313f0fc21 100644 --- a/src/main/java/org/olat/modules/lecture/ui/export/LecturesBlockSignaturePDFExport.java +++ b/src/main/java/org/olat/modules/lecture/ui/export/LecturesBlockSignaturePDFExport.java @@ -276,7 +276,7 @@ public class LecturesBlockSignaturePDFExport extends PdfDocument implements Medi //draw the content texty = currentY - 15; for (int i=offset; i<end; i++) { - String text = content[i]; + String text = cleanString(content[i]); if(text == null) continue; if(rowHeights[i] > rowHeight + 1) { diff --git a/src/main/java/org/olat/search/service/SearchServiceImpl.java b/src/main/java/org/olat/search/service/SearchServiceImpl.java index f366ae6bcafe0937ea0e225f4d064998b9f12b0b..a6e3e07392eebe98a940c2ad1d079247b35ed31f 100644 --- a/src/main/java/org/olat/search/service/SearchServiceImpl.java +++ b/src/main/java/org/olat/search/service/SearchServiceImpl.java @@ -82,6 +82,7 @@ import org.olat.search.service.indexer.Index; import org.olat.search.service.indexer.IndexerEvent; import org.olat.search.service.indexer.LifeFullIndexer; import org.olat.search.service.indexer.MainIndexer; +import org.olat.search.service.searcher.ConditionalQueryAnalyzer; import org.olat.search.service.searcher.JmsSearchProvider; import org.olat.search.service.spell.SearchSpellChecker; import org.quartz.JobDetail; @@ -104,7 +105,8 @@ public class SearchServiceImpl implements SearchService, GenericEventListener { private Scheduler scheduler; private CoordinatorManager coordinatorManager; - private Analyzer analyzer; + private final Analyzer analyzer; + private final ConditionalQueryAnalyzer conditionalQueryAnalyzer; private LifeFullIndexer lifeIndexer; private SearchSpellChecker searchSpellChecker; @@ -148,6 +150,7 @@ public class SearchServiceImpl implements SearchService, GenericEventListener { this.mainIndexer = mainIndexer; this.coordinatorManager = coordinatorManager; analyzer = new StandardAnalyzer(); + conditionalQueryAnalyzer = new ConditionalQueryAnalyzer(); searchProvider.setSearchService(this); coordinatorManager.getCoordinator().getEventBus().registerFor(this, null, IndexerEvent.INDEX_ORES); } @@ -405,7 +408,7 @@ public class SearchServiceImpl implements SearchService, GenericEventListener { if(condQueries != null && !condQueries.isEmpty()) { for(String condQueryString:condQueries) { - QueryParser condQueryParser = new QueryParser(condQueryString, analyzer); + QueryParser condQueryParser = new QueryParser(condQueryString, conditionalQueryAnalyzer); condQueryParser.setLocale(locale); Query condQuery = condQueryParser.parse(condQueryString); query.add(condQuery, Occur.MUST); diff --git a/src/main/java/org/olat/search/service/searcher/ConditionalQueryAnalyzer.java b/src/main/java/org/olat/search/service/searcher/ConditionalQueryAnalyzer.java new file mode 100644 index 0000000000000000000000000000000000000000..d0192a0d91e9719c2cad6a6c5541867552887a68 --- /dev/null +++ b/src/main/java/org/olat/search/service/searcher/ConditionalQueryAnalyzer.java @@ -0,0 +1,76 @@ +/** + * <a href="http://www.openolat.org"> + * OpenOLAT - Online Learning and Training</a><br> + * <p> + * Licensed under the Apache License, Version 2.0 (the "License"); <br> + * you may not use this file except in compliance with the License.<br> + * You may obtain a copy of the License at the + * <a href="http://www.apache.org/licenses/LICENSE-2.0">Apache homepage</a> + * <p> + * Unless required by applicable law or agreed to in writing,<br> + * software distributed under the License is distributed on an "AS IS" BASIS, <br> + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. <br> + * See the License for the specific language governing permissions and <br> + * limitations under the License. + * <p> + * Initial code contributed and copyrighted by<br> + * frentix GmbH, http://www.frentix.com + * <p> + */ +package org.olat.search.service.searcher; + +import java.io.Reader; + +import org.apache.lucene.analysis.LowerCaseFilter; +import org.apache.lucene.analysis.StopFilter; +import org.apache.lucene.analysis.StopwordAnalyzerBase; +import org.apache.lucene.analysis.TokenStream; +import org.apache.lucene.analysis.en.EnglishAnalyzer; +import org.apache.lucene.analysis.standard.StandardTokenizer; +import org.olat.search.model.AbstractOlatDocument; + +/** + * This analyzer is based on the StandardAnalyzer with the english + * stop words, but it doesn't normalize the fields resourceurl and + * resourceurlmd5 to lower case. This is essential because the business + * path is case sensitive and the standard analyzers of Lucene lower + * case the search queries. + * + * + * Initial date: 17 janv. 2019<br> + * + * @author srosse, stephane.rosse@frentix.com, http://www.frentix.com + * + */ +public class ConditionalQueryAnalyzer extends StopwordAnalyzerBase { + + private static final int maxTokenLength = 255; + public ConditionalQueryAnalyzer() { + super(EnglishAnalyzer.ENGLISH_STOP_WORDS_SET); + } + + @Override + protected TokenStreamComponents createComponents(final String fieldName) { + final StandardTokenizer src = new StandardTokenizer(); + src.setMaxTokenLength(maxTokenLength); + TokenStream tok = new LowerCaseFilter(src); + tok = new StopFilter(tok, stopwords); + return new TokenStreamComponents(src, tok) { + @Override + protected void setReader(final Reader reader) { + // So that if maxTokenLength was changed, the change takes + // effect next time tokenStream is called: + src.setMaxTokenLength(maxTokenLength); + super.setReader(reader); + } + }; + } + + @Override + protected TokenStream normalize(String fieldName, TokenStream in) { + if(AbstractOlatDocument.RESOURCEURL_FIELD_NAME.equals(fieldName) || AbstractOlatDocument.RESOURCEURL_MD5_FIELD_NAME.equals(fieldName)) { + return in; + } + return new LowerCaseFilter(in); + } +}