Skip to content

Commit

Permalink
Merge branch 'stefanolsen-feature/replace-jsonconverter' into v9
Browse files Browse the repository at this point in the history
  • Loading branch information
valdisiljuconoks committed Dec 5, 2024
2 parents 7fce628 + 7f62cd5 commit e1e9f67
Show file tree
Hide file tree
Showing 2 changed files with 99 additions and 26 deletions.
31 changes: 5 additions & 26 deletions common/src/DbLocalizationProvider/LocalizationProvider.cs
Original file line number Diff line number Diff line change
Expand Up @@ -9,12 +9,9 @@
using System.Text.RegularExpressions;
using DbLocalizationProvider.Abstractions;
using DbLocalizationProvider.Internal;
using DbLocalizationProvider.Json;
using DbLocalizationProvider.Queries;
using DbLocalizationProvider.Sync;
using Microsoft.Extensions.Options;
using Newtonsoft.Json;
using JsonConverter = DbLocalizationProvider.Json.JsonConverter;

namespace DbLocalizationProvider;

Expand All @@ -28,8 +25,7 @@ public class LocalizationProvider : ILocalizationProvider
private readonly ResourceKeyBuilder _keyBuilder;
internal readonly IQueryExecutor _queryExecutor;
private readonly ScanState _scanState;
private readonly JsonConverter _converter;
private readonly JsonSerializer _serializer;
private readonly ReflectionConverter _reflectionConverter;

/// <summary>
/// Creates new localization provider with all the required settings and services injected.
Expand All @@ -54,12 +50,8 @@ public LocalizationProvider(
_fallbackCollection = context.Value._fallbackCollection;
_queryExecutor = queryExecutor;
_scanState = scanState;

_converter = new JsonConverter(_queryExecutor, _scanState);
_serializer = new JsonSerializer
{
ContractResolver = new StaticPropertyContractResolver()
};

_reflectionConverter = new ReflectionConverter(_queryExecutor, _scanState, _keyBuilder);
}

/// <summary>
Expand Down Expand Up @@ -221,27 +213,14 @@ public T Translate<T>()
}

/// <summary>
/// Give a type to this method and it will return instance of the type but translated
/// Give a type to this method, and it will return instance of the type but translated
/// </summary>
/// <typeparam name="T">Type of the target class you want to translate</typeparam>
/// <param name="language">Language to use during translation</param>
/// <returns>Translated class</returns>
public T Translate<T>(CultureInfo language)
{
var className = typeof(T).FullName;

var json = _converter.GetJson(className, language.Name, _fallbackCollection);

// get the actual class Json representation (we need to select token through FQN of the class)
// to supported nested classes - we need to fix a bit resource key name
var jsonToken = json.SelectToken(className.Replace('+', '.'));

if (jsonToken == null)
{
return (T)Activator.CreateInstance(typeof(T), new object[] { });
}

return jsonToken.ToObject<T>(_serializer);
return _reflectionConverter.Convert<T>(language.Name, _fallbackCollection);
}

/// <summary>
Expand Down
94 changes: 94 additions & 0 deletions common/src/DbLocalizationProvider/ReflectionConverter.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
// Copyright (c) Stefan Holm Olsen. All rights reserved.
// Licensed under Apache-2.0. See the LICENSE file in the project root for more information

using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using DbLocalizationProvider.Abstractions;
using DbLocalizationProvider.Queries;
using DbLocalizationProvider.Sync;

namespace DbLocalizationProvider;

/// <summary>
/// Much faster translations to object converter based on reflection
/// </summary>
public class ReflectionConverter(IQueryExecutor queryExecutor, ScanState scanState, ResourceKeyBuilder keyBuilder)
{
/// <summary>
/// Creates an object of <typeparam name="T"></typeparam> and fills with translations
/// </summary>
/// <param name="languageName">Specify in which language you are going to use</param>
/// <param name="fallbackCollection">Fallback languages collection</param>
/// <typeparam name="T">Specify target object type</typeparam>
/// <returns>If all is good, will return object of type <typeparam name="T"></typeparam> filled with translations of matching keys</returns>
public T Convert<T>(string languageName, FallbackLanguagesCollection fallbackCollection)
{
// TODO: Can the dictionary be cached?
// TODO: Can we go around query execution and use repository directly?
var resources = queryExecutor
.Execute(new GetAllResources.Query())
.ToDictionary(x => x.ResourceKey, StringComparer.Ordinal);

var newObject = Activator.CreateInstance<T>();

FillProperties(newObject!, languageName, resources, fallbackCollection);

return newObject;
}

private void FillProperties(
object instance,
string languageName,
Dictionary<string, LocalizationResource> resources,
FallbackLanguagesCollection fallbackCollection)
{
var type = instance!.GetType();
var properties = type.GetProperties(BindingFlags.Instance | BindingFlags.Public | BindingFlags.Static);
foreach (var propertyInfo in properties)
{
if (propertyInfo.MemberType == MemberTypes.NestedType)
{
var nestedObject = Activator.CreateInstance(propertyInfo.PropertyType);
if (nestedObject == null)
{
continue;
}

FillProperties(nestedObject, languageName, resources, fallbackCollection);

propertyInfo.SetValue(instance, nestedObject);
}

if (propertyInfo.PropertyType != typeof(string))
{
continue;
}

string? translation;
var key = keyBuilder.BuildResourceKey(type, propertyInfo.Name);
if (scanState.UseResourceAttributeCache.TryGetValue(key, out var targetResourceKey)
&& resources.TryGetValue(targetResourceKey, out var foundResource))
{
translation = foundResource.Translations.GetValueWithFallback(
languageName,
fallbackCollection.GetFallbackLanguages(languageName));

propertyInfo.SetValue(instance, translation);
continue;
}

if (!resources.TryGetValue(key, out var resource))
{
continue;
}

translation = resource.Translations.GetValueWithFallback(
languageName,
fallbackCollection.GetFallbackLanguages(languageName));

propertyInfo.SetValue(instance, translation);
}
}
}

0 comments on commit e1e9f67

Please sign in to comment.