# Conflicts: # build/NuSpecs/UmbracoCms.Web.nuspec # src/SolutionInfo.cs # src/Umbraco.Core/Compose/RelateOnTrashComponent.cs # src/Umbraco.Core/Composing/Current.cs # src/Umbraco.Core/Constants-AppSettings.cs # src/Umbraco.Core/Constants-SqlTemplates.cs # src/Umbraco.Core/Dashboards/ContentDashboardSettings.cs # src/Umbraco.Core/Dashboards/IContentDashboardSettings.cs # src/Umbraco.Core/Extensions/PublishedPropertyExtension.cs # src/Umbraco.Core/HealthChecks/Checks/Services/SmtpCheck.cs # src/Umbraco.Core/Models/IReadOnlyContentBase.cs # src/Umbraco.Core/Models/Mapping/MemberTabsAndPropertiesMapper.cs # src/Umbraco.Core/Models/Mapping/UserMapDefinition.cs # src/Umbraco.Core/Models/ReadOnlyContentBaseAdapter.cs # src/Umbraco.Core/Persistence/NPocoDatabaseExtensions-Bulk.cs # src/Umbraco.Core/PropertyEditors/IPropertyCacheCompression.cs # src/Umbraco.Core/PropertyEditors/IPropertyCacheCompressionOptions.cs # src/Umbraco.Core/PropertyEditors/MediaPicker3Configuration.cs # src/Umbraco.Core/PropertyEditors/NoopPropertyCacheCompressionOptions.cs # src/Umbraco.Core/PropertyEditors/PropertyCacheCompression.cs # src/Umbraco.Core/Routing/UrlProviderExtensions.cs # src/Umbraco.Core/Runtime/CoreRuntime.cs # src/Umbraco.Core/Services/ILocalizedTextService.cs # src/Umbraco.Core/Services/LocalizedTextServiceExtensions.cs # src/Umbraco.Examine.Lucene/BackOfficeExamineSearcher.cs # src/Umbraco.Examine/UmbracoContentIndex.cs # src/Umbraco.Infrastructure/Examine/GenericIndexDiagnostics.cs # src/Umbraco.Infrastructure/IPublishedContentQuery.cs # src/Umbraco.Infrastructure/Migrations/Upgrade/UmbracoPlan.cs # src/Umbraco.Infrastructure/Models/MediaWithCrops.cs # src/Umbraco.Infrastructure/Persistence/NPocoDatabaseExtensions.cs # src/Umbraco.Infrastructure/Persistence/UmbracoDatabase.cs # src/Umbraco.Infrastructure/PropertyEditors/ImageCropperConfiguration.cs # src/Umbraco.Infrastructure/PropertyEditors/MediaPicker3PropertyEditor.cs # src/Umbraco.Infrastructure/PropertyEditors/NestedContentPropertyEditor.cs # src/Umbraco.Infrastructure/PropertyEditors/UploadFileTypeValidator.cs # src/Umbraco.Infrastructure/PropertyEditors/ValueConverters/ImageCropperValue.cs # src/Umbraco.Infrastructure/PropertyEditors/ValueConverters/MediaPickerWithCropsValueConverter.cs # src/Umbraco.Infrastructure/PublishedContentQuery.cs # src/Umbraco.Infrastructure/Search/UmbracoTreeSearcherFields.cs # src/Umbraco.Infrastructure/Services/Implement/LocalizedTextService.cs # src/Umbraco.Persistence.SqlCe/SqlCeSyntaxProvider.cs # src/Umbraco.PublishedCache.NuCache/DataSource/BTree.DictionaryOfPropertyDataSerializer.cs # src/Umbraco.PublishedCache.NuCache/DataSource/BTree.cs # src/Umbraco.PublishedCache.NuCache/DataSource/ContentData.cs # src/Umbraco.PublishedCache.NuCache/DataSource/ContentSourceDto.cs # src/Umbraco.PublishedCache.NuCache/DataSource/PropertyData.cs # src/Umbraco.Tests.Integration/Umbraco.Infrastructure/Services/ContentTypeServiceVariantsTests.cs # src/Umbraco.Tests.UnitTests/Umbraco.Web.Common/ImageCropperTest.cs # src/Umbraco.Tests/App.config # src/Umbraco.Tests/Models/Mapping/ContentWebModelMappingTests.cs # src/Umbraco.Tests/PublishedContent/NuCacheChildrenTests.cs # src/Umbraco.Tests/PublishedContent/NuCacheTests.cs # src/Umbraco.Tests/Scoping/ScopedNuCacheTests.cs # src/Umbraco.Web.BackOffice/Controllers/CodeFileController.cs # src/Umbraco.Web.BackOffice/Controllers/ContentController.cs # src/Umbraco.Web.BackOffice/Controllers/ContentTypeControllerBase.cs # src/Umbraco.Web.BackOffice/Controllers/DashboardController.cs # src/Umbraco.Web.BackOffice/Controllers/DataTypeController.cs # src/Umbraco.Web.BackOffice/Controllers/MediaController.cs # src/Umbraco.Web.BackOffice/Controllers/PackageInstallController.cs # src/Umbraco.Web.BackOffice/Controllers/TemplateQueryController.cs # src/Umbraco.Web.BackOffice/Mapping/ContentMapDefinition.cs # src/Umbraco.Web.BackOffice/Trees/ContentTreeControllerBase.cs # src/Umbraco.Web.BackOffice/Trees/DataTypeTreeController.cs # src/Umbraco.Web.BackOffice/Trees/MediaTypeTreeController.cs # src/Umbraco.Web.Common/Extensions/FriendlyImageCropperTemplateExtensions.cs # src/Umbraco.Web.Common/Extensions/ImageCropperTemplateCoreExtensions.cs # src/Umbraco.Web.Common/Macros/MacroRenderer.cs # src/Umbraco.Web.UI.NetCore/umbraco/config/lang/da.xml # src/Umbraco.Web.UI/Umbraco/config/lang/it.xml # src/Umbraco.Web.UI/web.Template.Debug.config # src/Umbraco.Web.UI/web.Template.config # src/Umbraco.Web/Compose/NotificationsComponent.cs # src/Umbraco.Web/Composing/ModuleInjector.cs # src/Umbraco.Web/Editors/AuthenticationController.cs # src/Umbraco.Web/Editors/BackOfficeController.cs # src/Umbraco.Web/Editors/ContentTypeController.cs # src/Umbraco.Web/Editors/CurrentUserController.cs # src/Umbraco.Web/Editors/DictionaryController.cs # src/Umbraco.Web/Editors/MediaTypeController.cs # src/Umbraco.Web/Editors/MemberController.cs # src/Umbraco.Web/Editors/MemberGroupController.cs # src/Umbraco.Web/Editors/MemberTypeController.cs # src/Umbraco.Web/Editors/NuCacheStatusController.cs # src/Umbraco.Web/Editors/UserGroupsController.cs # src/Umbraco.Web/Editors/UsersController.cs # src/Umbraco.Web/HealthCheck/Checks/Config/AbstractConfigCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Config/CompilationDebugCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Config/ConfigurationService.cs # src/Umbraco.Web/HealthCheck/Checks/Config/CustomErrorsCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Config/MacroErrorsCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Config/NotificationEmailCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Config/TraceCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Config/TrySkipIisCustomErrorsCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Permissions/FolderAndFilePermissionsCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Security/BaseHttpHeaderCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Security/ExcessiveHeadersCheck.cs # src/Umbraco.Web/HealthCheck/Checks/Security/HttpsCheck.cs # src/Umbraco.Web/HealthCheck/NotificationMethods/EmailNotificationMethod.cs # src/Umbraco.Web/Models/Trees/MenuItemList.cs # src/Umbraco.Web/PublishedCache/NuCache/DataSource/ContentCacheDataModel.cs # src/Umbraco.Web/PublishedCache/NuCache/DataSource/DatabaseDataSource.cs # src/Umbraco.Web/PublishedCache/NuCache/DataSource/SerializerBase.cs # src/Umbraco.Web/PublishedCache/NuCache/NuCacheComposer.cs # src/Umbraco.Web/Runtime/WebRuntime.cs # src/Umbraco.Web/Search/ExamineComponent.cs # src/Umbraco.Web/Trees/ApplicationTreeController.cs # src/Umbraco.Web/Trees/MemberTreeController.cs # src/Umbraco.Web/UrlHelperRenderExtensions.cs
472 lines
18 KiB
C#
472 lines
18 KiB
C#
// Copyright (c) Umbraco.
|
|
// See LICENSE for more details.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Text.RegularExpressions;
|
|
using Examine;
|
|
using Examine.Search;
|
|
using Examine.Search;
|
|
using Lucene.Net.QueryParsers.Classic;
|
|
using Umbraco.Cms.Core;
|
|
using Umbraco.Cms.Core.Cache;
|
|
using Umbraco.Cms.Core.Mapping;
|
|
using Umbraco.Cms.Core.Models;
|
|
using Umbraco.Cms.Core.Models.ContentEditing;
|
|
using Umbraco.Cms.Core.Routing;
|
|
using Umbraco.Cms.Core.Security;
|
|
using Umbraco.Cms.Core.Services;
|
|
using Umbraco.Cms.Core.Web;
|
|
using Umbraco.Extensions;
|
|
using Constants = Umbraco.Cms.Core.Constants;
|
|
|
|
namespace Umbraco.Cms.Infrastructure.Examine
|
|
{
|
|
public class BackOfficeExamineSearcher : IBackOfficeExamineSearcher
|
|
{
|
|
private readonly IExamineManager _examineManager;
|
|
private readonly ILocalizationService _languageService;
|
|
private readonly IBackOfficeSecurityAccessor _backOfficeSecurityAccessor;
|
|
private readonly IEntityService _entityService;
|
|
private readonly IUmbracoTreeSearcherFields _treeSearcherFields;
|
|
private readonly AppCaches _appCaches;
|
|
private readonly IUmbracoMapper _umbracoMapper;
|
|
private readonly IPublishedUrlProvider _publishedUrlProvider;
|
|
|
|
public BackOfficeExamineSearcher(IExamineManager examineManager,
|
|
ILocalizationService languageService,
|
|
IBackOfficeSecurityAccessor backOfficeSecurityAccessor,
|
|
IEntityService entityService,
|
|
IUmbracoTreeSearcherFields treeSearcherFields,
|
|
AppCaches appCaches,
|
|
IUmbracoMapper umbracoMapper,
|
|
IPublishedUrlProvider publishedUrlProvider)
|
|
{
|
|
_examineManager = examineManager;
|
|
_languageService = languageService;
|
|
_backOfficeSecurityAccessor = backOfficeSecurityAccessor;
|
|
_entityService = entityService;
|
|
_treeSearcherFields = treeSearcherFields;
|
|
_appCaches = appCaches;
|
|
_umbracoMapper = umbracoMapper;
|
|
_publishedUrlProvider = publishedUrlProvider;
|
|
}
|
|
|
|
public IEnumerable<ISearchResult> Search(string query, UmbracoEntityTypes entityType, int pageSize, long pageIndex, out long totalFound, string searchFrom = null, bool ignoreUserStartNodes = false)
|
|
{
|
|
var sb = new StringBuilder();
|
|
|
|
string type;
|
|
var indexName = Constants.UmbracoIndexes.InternalIndexName;
|
|
var fields = _treeSearcherFields.GetBackOfficeFields().ToList();
|
|
|
|
ISet<string> fieldsToLoad = new HashSet<string>(_treeSearcherFields.GetBackOfficeFieldsToLoad());
|
|
|
|
// TODO: WE should try to allow passing in a lucene raw query, however we will still need to do some manual string
|
|
// manipulation for things like start paths, member types, etc...
|
|
//if (Examine.ExamineExtensions.TryParseLuceneQuery(query))
|
|
//{
|
|
|
|
//}
|
|
|
|
//special GUID check since if a user searches on one specifically we need to escape it
|
|
if (Guid.TryParse(query, out var g))
|
|
{
|
|
query = "\"" + g.ToString() + "\"";
|
|
}
|
|
|
|
var currentUser = _backOfficeSecurityAccessor?.BackOfficeSecurity?.CurrentUser;
|
|
|
|
switch (entityType)
|
|
{
|
|
case UmbracoEntityTypes.Member:
|
|
indexName = Constants.UmbracoIndexes.MembersIndexName;
|
|
type = "member";
|
|
fields.AddRange(_treeSearcherFields.GetBackOfficeMembersFields());
|
|
foreach(var field in _treeSearcherFields.GetBackOfficeMembersFieldsToLoad())
|
|
{
|
|
fieldsToLoad.Add(field);
|
|
}
|
|
|
|
if (searchFrom != null && searchFrom != Constants.Conventions.MemberTypes.AllMembersListId && searchFrom.Trim() != "-1")
|
|
{
|
|
sb.Append("+__NodeTypeAlias:");
|
|
sb.Append(searchFrom);
|
|
sb.Append(" ");
|
|
}
|
|
break;
|
|
case UmbracoEntityTypes.Media:
|
|
type = "media";
|
|
fields.AddRange(_treeSearcherFields.GetBackOfficeMediaFields());
|
|
foreach (var field in _treeSearcherFields.GetBackOfficeMediaFieldsToLoad())
|
|
{
|
|
fieldsToLoad.Add(field);
|
|
}
|
|
|
|
var allMediaStartNodes = currentUser != null
|
|
? currentUser.CalculateMediaStartNodeIds(_entityService, _appCaches)
|
|
: Array.Empty<int>();
|
|
AppendPath(sb, UmbracoObjectTypes.Media, allMediaStartNodes, searchFrom, ignoreUserStartNodes, _entityService);
|
|
break;
|
|
case UmbracoEntityTypes.Document:
|
|
type = "content";
|
|
fields.AddRange(_treeSearcherFields.GetBackOfficeDocumentFields());
|
|
foreach (var field in _treeSearcherFields.GetBackOfficeDocumentFieldsToLoad())
|
|
{
|
|
fieldsToLoad.Add(field);
|
|
}
|
|
var allContentStartNodes = currentUser != null
|
|
? currentUser.CalculateContentStartNodeIds(_entityService, _appCaches)
|
|
: Array.Empty<int>();
|
|
AppendPath(sb, UmbracoObjectTypes.Document, allContentStartNodes, searchFrom, ignoreUserStartNodes, _entityService);
|
|
break;
|
|
default:
|
|
throw new NotSupportedException("The " + typeof(BackOfficeExamineSearcher) + " currently does not support searching against object type " + entityType);
|
|
}
|
|
|
|
if (!_examineManager.TryGetIndex(indexName, out var index))
|
|
throw new InvalidOperationException("No index found by name " + indexName);
|
|
|
|
if (!BuildQuery(sb, query, searchFrom, fields, type))
|
|
{
|
|
totalFound = 0;
|
|
return Enumerable.Empty<ISearchResult>();
|
|
}
|
|
|
|
var result = index.Searcher
|
|
.CreateQuery()
|
|
.NativeQuery(sb.ToString())
|
|
.SelectFields(fieldsToLoad)
|
|
//only return the number of items specified to read up to the amount of records to fill from 0 -> the number of items on the page requested
|
|
.Execute(QueryOptions.SkipTake(Convert.ToInt32(pageSize * pageIndex), pageSize));
|
|
|
|
totalFound = result.TotalItemCount;
|
|
|
|
var pagedResult = result.Skip(Convert.ToInt32(pageIndex));
|
|
|
|
return pagedResult;
|
|
}
|
|
|
|
private bool BuildQuery(StringBuilder sb, string query, string searchFrom, List<string> fields, string type)
|
|
{
|
|
//build a lucene query:
|
|
// the nodeName will be boosted 10x without wildcards
|
|
// then nodeName will be matched normally with wildcards
|
|
// the rest will be normal without wildcards
|
|
|
|
var allLangs = _languageService.GetAllLanguages().Select(x => x.IsoCode.ToLowerInvariant()).ToList();
|
|
|
|
// the chars [*-_] in the query will mess everything up so let's remove those
|
|
query = Regex.Replace(query, "[\\*\\-_]", "");
|
|
|
|
//check if text is surrounded by single or double quotes, if so, then exact match
|
|
var surroundedByQuotes = Regex.IsMatch(query, "^\".*?\"$")
|
|
|| Regex.IsMatch(query, "^\'.*?\'$");
|
|
|
|
if (surroundedByQuotes)
|
|
{
|
|
//strip quotes, escape string, the replace again
|
|
query = query.Trim(Constants.CharArrays.DoubleQuoteSingleQuote);
|
|
|
|
query = QueryParser.Escape(query);
|
|
|
|
//nothing to search
|
|
if (searchFrom.IsNullOrWhiteSpace() && query.IsNullOrWhiteSpace())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
//update the query with the query term
|
|
if (query.IsNullOrWhiteSpace() == false)
|
|
{
|
|
//add back the surrounding quotes
|
|
query = string.Format("{0}{1}{0}", "\"", query);
|
|
|
|
sb.Append("+(");
|
|
|
|
AppendNodeNamePhraseWithBoost(sb, query, allLangs);
|
|
|
|
foreach (var f in fields)
|
|
{
|
|
//additional fields normally
|
|
sb.Append(f);
|
|
sb.Append(": (");
|
|
sb.Append(query);
|
|
sb.Append(") ");
|
|
}
|
|
|
|
sb.Append(") ");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
var trimmed = query.Trim(Constants.CharArrays.DoubleQuoteSingleQuote);
|
|
|
|
//nothing to search
|
|
if (searchFrom.IsNullOrWhiteSpace() && trimmed.IsNullOrWhiteSpace())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
//update the query with the query term
|
|
if (trimmed.IsNullOrWhiteSpace() == false)
|
|
{
|
|
query = QueryParser.Escape(query);
|
|
|
|
var querywords = query.Split(Constants.CharArrays.Space, StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
sb.Append("+(");
|
|
|
|
AppendNodeNameExactWithBoost(sb, query, allLangs);
|
|
|
|
AppendNodeNameWithWildcards(sb, querywords, allLangs);
|
|
|
|
foreach (var f in fields)
|
|
{
|
|
var queryWordsReplaced = new string[querywords.Length];
|
|
|
|
// when searching file names containing hyphens we need to replace the hyphens with spaces
|
|
if (f.Equals(UmbracoExamineFieldNames.UmbracoFileFieldName))
|
|
{
|
|
for (var index = 0; index < querywords.Length; index++)
|
|
{
|
|
queryWordsReplaced[index] = querywords[index].Replace("\\-", " ").Replace("_", " ").Trim(" ");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
queryWordsReplaced = querywords;
|
|
}
|
|
|
|
//additional fields normally
|
|
sb.Append(f);
|
|
sb.Append(":");
|
|
sb.Append("(");
|
|
foreach (var w in queryWordsReplaced)
|
|
{
|
|
sb.Append(w.ToLower());
|
|
sb.Append("* ");
|
|
}
|
|
sb.Append(")");
|
|
sb.Append(" ");
|
|
}
|
|
|
|
sb.Append(") ");
|
|
}
|
|
}
|
|
|
|
//must match index type
|
|
sb.Append("+__IndexType:");
|
|
sb.Append(type);
|
|
|
|
return true;
|
|
}
|
|
|
|
private void AppendNodeNamePhraseWithBoost(StringBuilder sb, string query, IEnumerable<string> allLangs)
|
|
{
|
|
//node name exactly boost x 10
|
|
sb.Append("nodeName: (");
|
|
sb.Append(query.ToLower());
|
|
sb.Append(")^10.0 ");
|
|
|
|
//also search on all variant node names
|
|
foreach (var lang in allLangs)
|
|
{
|
|
//node name exactly boost x 10
|
|
sb.Append($"nodeName_{lang}: (");
|
|
sb.Append(query.ToLower());
|
|
sb.Append(")^10.0 ");
|
|
}
|
|
}
|
|
|
|
private void AppendNodeNameExactWithBoost(StringBuilder sb, string query, IEnumerable<string> allLangs)
|
|
{
|
|
//node name exactly boost x 10
|
|
sb.Append("nodeName:");
|
|
sb.Append("\"");
|
|
sb.Append(query.ToLower());
|
|
sb.Append("\"");
|
|
sb.Append("^10.0 ");
|
|
//also search on all variant node names
|
|
foreach (var lang in allLangs)
|
|
{
|
|
//node name exactly boost x 10
|
|
sb.Append($"nodeName_{lang}:");
|
|
sb.Append("\"");
|
|
sb.Append(query.ToLower());
|
|
sb.Append("\"");
|
|
sb.Append("^10.0 ");
|
|
}
|
|
}
|
|
|
|
private void AppendNodeNameWithWildcards(StringBuilder sb, string[] querywords, IEnumerable<string> allLangs)
|
|
{
|
|
//node name normally with wildcards
|
|
sb.Append("nodeName:");
|
|
sb.Append("(");
|
|
foreach (var w in querywords)
|
|
{
|
|
sb.Append(w.ToLower());
|
|
sb.Append("* ");
|
|
}
|
|
sb.Append(") ");
|
|
//also search on all variant node names
|
|
foreach (var lang in allLangs)
|
|
{
|
|
//node name normally with wildcards
|
|
sb.Append($"nodeName_{lang}:");
|
|
sb.Append("(");
|
|
foreach (var w in querywords)
|
|
{
|
|
sb.Append(w.ToLower());
|
|
sb.Append("* ");
|
|
}
|
|
sb.Append(") ");
|
|
}
|
|
}
|
|
|
|
private void AppendPath(StringBuilder sb, UmbracoObjectTypes objectType, int[] startNodeIds, string searchFrom, bool ignoreUserStartNodes, IEntityService entityService)
|
|
{
|
|
if (sb == null) throw new ArgumentNullException(nameof(sb));
|
|
if (entityService == null) throw new ArgumentNullException(nameof(entityService));
|
|
|
|
UdiParser.TryParse(searchFrom, true, out var udi);
|
|
searchFrom = udi == null ? searchFrom : entityService.GetId(udi).Result.ToString();
|
|
|
|
var entityPath = int.TryParse(searchFrom, out var searchFromId) && searchFromId > 0
|
|
? entityService.GetAllPaths(objectType, searchFromId).FirstOrDefault()
|
|
: null;
|
|
if (entityPath != null)
|
|
{
|
|
// find... only what's underneath
|
|
sb.Append("+__Path:");
|
|
AppendPath(sb, entityPath.Path, false);
|
|
sb.Append(" ");
|
|
}
|
|
else if (startNodeIds.Length == 0)
|
|
{
|
|
// make sure we don't find anything
|
|
sb.Append("+__Path:none ");
|
|
}
|
|
else if (startNodeIds.Contains(-1) == false && ignoreUserStartNodes == false) // -1 = no restriction
|
|
{
|
|
var entityPaths = entityService.GetAllPaths(objectType, startNodeIds);
|
|
|
|
// for each start node, find the start node, and what's underneath
|
|
// +__Path:(-1*,1234 -1*,1234,* -1*,5678 -1*,5678,* ...)
|
|
sb.Append("+__Path:(");
|
|
var first = true;
|
|
foreach (var ep in entityPaths)
|
|
{
|
|
if (first)
|
|
first = false;
|
|
else
|
|
sb.Append(" ");
|
|
AppendPath(sb, ep.Path, true);
|
|
}
|
|
sb.Append(") ");
|
|
}
|
|
}
|
|
|
|
private void AppendPath(StringBuilder sb, string path, bool includeThisNode)
|
|
{
|
|
path = path.Replace("-", "\\-").Replace(",", "\\,");
|
|
if (includeThisNode)
|
|
{
|
|
sb.Append(path);
|
|
sb.Append(" ");
|
|
}
|
|
sb.Append(path);
|
|
sb.Append("\\,*");
|
|
}
|
|
|
|
// TODO: When/Where is this used?
|
|
|
|
/// <summary>
|
|
/// Returns a collection of entities for media based on search results
|
|
/// </summary>
|
|
/// <param name="results"></param>
|
|
/// <returns></returns>
|
|
private IEnumerable<SearchResultEntity> MemberFromSearchResults(IEnumerable<ISearchResult> results)
|
|
{
|
|
//add additional data
|
|
foreach (var result in results)
|
|
{
|
|
var m = _umbracoMapper.Map<SearchResultEntity>(result);
|
|
|
|
//if no icon could be mapped, it will be set to document, so change it to picture
|
|
if (m.Icon == Constants.Icons.DefaultIcon)
|
|
{
|
|
m.Icon = Constants.Icons.Member;
|
|
}
|
|
|
|
if (result.Values.ContainsKey("email") && result.Values["email"] != null)
|
|
{
|
|
m.AdditionalData["Email"] = result.Values["email"];
|
|
}
|
|
if (result.Values.ContainsKey(UmbracoExamineFieldNames.NodeKeyFieldName) && result.Values[UmbracoExamineFieldNames.NodeKeyFieldName] != null)
|
|
{
|
|
if (Guid.TryParse(result.Values[UmbracoExamineFieldNames.NodeKeyFieldName], out var key))
|
|
{
|
|
m.Key = key;
|
|
}
|
|
}
|
|
|
|
yield return m;
|
|
}
|
|
}
|
|
|
|
// TODO: When/Where is this used?
|
|
|
|
/// <summary>
|
|
/// Returns a collection of entities for media based on search results
|
|
/// </summary>
|
|
/// <param name="results"></param>
|
|
/// <returns></returns>
|
|
private IEnumerable<SearchResultEntity> MediaFromSearchResults(IEnumerable<ISearchResult> results)
|
|
=> _umbracoMapper.Map<IEnumerable<SearchResultEntity>>(results);
|
|
|
|
// TODO: When/Where is this used?
|
|
|
|
/// <summary>
|
|
/// Returns a collection of entities for content based on search results
|
|
/// </summary>
|
|
/// <param name="results"></param>
|
|
/// <returns></returns>
|
|
private IEnumerable<SearchResultEntity> ContentFromSearchResults(IEnumerable<ISearchResult> results, string culture = null)
|
|
{
|
|
var defaultLang = _languageService.GetDefaultLanguageIsoCode();
|
|
foreach (var result in results)
|
|
{
|
|
var entity = _umbracoMapper.Map<SearchResultEntity>(result, context =>
|
|
{
|
|
if (culture != null)
|
|
{
|
|
context.SetCulture(culture);
|
|
}
|
|
}
|
|
);
|
|
|
|
var intId = entity.Id.TryConvertTo<int>();
|
|
if (intId.Success)
|
|
{
|
|
//if it varies by culture, return the default language URL
|
|
if (result.Values.TryGetValue(UmbracoExamineFieldNames.VariesByCultureFieldName, out var varies) && varies == "y")
|
|
{
|
|
entity.AdditionalData["Url"] = _publishedUrlProvider.GetUrl(intId.Result, culture: culture ?? defaultLang);
|
|
}
|
|
else
|
|
{
|
|
entity.AdditionalData["Url"] = _publishedUrlProvider.GetUrl(intId.Result);
|
|
}
|
|
}
|
|
|
|
yield return entity;
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|