Skip to content

Commit c3dc71c

Browse files
committed
Refresh resources cache when the file changed
1 parent c83db38 commit c3dc71c

File tree

3 files changed

+105
-21
lines changed

3 files changed

+105
-21
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
using System;
2+
using System.IO;
3+
4+
namespace My.Extensions.Localization.Json.Internal;
5+
6+
public class JsonFileWatcher : IDisposable
7+
{
8+
private const string JsonExtension = "*.json";
9+
10+
private bool _disposed;
11+
12+
private readonly FileSystemWatcher _filesWatcher;
13+
14+
public event FileSystemEventHandler Changed;
15+
16+
public JsonFileWatcher(string rootDirectory)
17+
{
18+
_filesWatcher = new(rootDirectory)
19+
{
20+
Filter = JsonExtension,
21+
NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.Size,
22+
EnableRaisingEvents = true
23+
};
24+
_filesWatcher.Changed += (s, e) => Changed?.Invoke(s, e);
25+
}
26+
27+
~JsonFileWatcher()
28+
{
29+
Dispose(false);
30+
}
31+
32+
public void Dispose()
33+
{
34+
Dispose(true);
35+
GC.SuppressFinalize(true);
36+
}
37+
38+
public virtual void Dispose(bool disposing)
39+
{
40+
if (_disposed)
41+
{
42+
return;
43+
}
44+
45+
if (disposing)
46+
{
47+
_filesWatcher.Dispose();
48+
}
49+
50+
_disposed = true;
51+
}
52+
}

src/My.Extensions.Localization.Json/Internal/JsonResourceManager.cs

+48-14
Original file line numberDiff line numberDiff line change
@@ -6,21 +6,33 @@
66

77
namespace My.Extensions.Localization.Json.Internal;
88

9-
public class JsonResourceManager(string resourcesPath, string resourceName = null)
9+
public class JsonResourceManager
1010
{
11+
private readonly JsonFileWatcher _jsonFileWatcher;
1112
private readonly ConcurrentDictionary<string, ConcurrentDictionary<string, string>> _resourcesCache = new();
1213

13-
public string ResourceName { get; } = resourceName;
1414

15-
public string ResourcesPath { get; } = resourcesPath;
15+
public JsonResourceManager(string resourcesPath, string resourceName = null)
16+
{
17+
ResourcesPath = resourcesPath;
18+
ResourceName = resourceName;
19+
20+
_jsonFileWatcher = new(resourcesPath);
21+
_jsonFileWatcher.Changed += RefreshResourcesCache;
22+
}
23+
24+
public string ResourceName { get; }
25+
26+
public string ResourcesPath { get; }
1627

1728
public string ResourcesFilePath { get; private set; }
1829

1930
public virtual ConcurrentDictionary<string, string> GetResourceSet(CultureInfo culture, bool tryParents)
2031
{
2132
TryLoadResourceSet(culture);
2233

23-
if (!_resourcesCache.ContainsKey(culture.Name))
34+
var key = $"{ResourceName}.{culture.Name}";
35+
if (!_resourcesCache.ContainsKey(key))
2436
{
2537
return null;
2638
}
@@ -30,7 +42,7 @@ public virtual ConcurrentDictionary<string, string> GetResourceSet(CultureInfo c
3042
var allResources = new ConcurrentDictionary<string, string>();
3143
do
3244
{
33-
if (_resourcesCache.TryGetValue(culture.Name, out var resources))
45+
if (_resourcesCache.TryGetValue(key, out var resources))
3446
{
3547
foreach (var entry in resources)
3648
{
@@ -45,7 +57,7 @@ public virtual ConcurrentDictionary<string, string> GetResourceSet(CultureInfo c
4557
}
4658
else
4759
{
48-
_resourcesCache.TryGetValue(culture.Name, out var resources);
60+
_resourcesCache.TryGetValue(key, out var resources);
4961

5062
return resources;
5163
}
@@ -63,7 +75,8 @@ public virtual string GetString(string name)
6375

6476
do
6577
{
66-
if (_resourcesCache.TryGetValue(culture.Name, out var resources))
78+
var key = $"{ResourceName}.{culture.Name}";
79+
if (_resourcesCache.TryGetValue(key, out var resources))
6780
{
6881
if (resources.TryGetValue(name, out var value))
6982
{
@@ -86,7 +99,8 @@ public virtual string GetString(string name, CultureInfo culture)
8699
return null;
87100
}
88101

89-
if (!_resourcesCache.TryGetValue(culture.Name, out var resources))
102+
var key = $"{ResourceName}.{culture.Name}";
103+
if (!_resourcesCache.TryGetValue(key, out var resources))
90104
{
91105
return null;
92106
}
@@ -102,7 +116,7 @@ private void TryLoadResourceSet(CultureInfo culture)
102116
{
103117
var file = Path.Combine(ResourcesPath, $"{culture.Name}.json");
104118

105-
GetOrAddResourceCache(file);
119+
TryAddResources(file);
106120
}
107121
else
108122
{
@@ -129,7 +143,7 @@ private void TryLoadResourceSet(CultureInfo culture)
129143

130144
culture = CultureInfo.GetCultureInfo(cultureName);
131145

132-
GetOrAddResourceCache(file);
146+
TryAddResources(file);
133147
}
134148
}
135149

@@ -147,14 +161,34 @@ IEnumerable<string> GetResourceFiles(string culture)
147161
: [];
148162
}
149163

150-
ConcurrentDictionary<string, string> GetOrAddResourceCache(string resourceFile)
164+
void TryAddResources(string resourceFile)
151165
{
152-
return _resourcesCache.GetOrAdd(culture.Name, _ =>
166+
var key = $"{ResourceName}.{culture.Name}";
167+
if (!_resourcesCache.ContainsKey(key))
153168
{
154169
var resources = JsonResourceLoader.Load(resourceFile);
155170

156-
return new ConcurrentDictionary<string, string>(resources.ToDictionary(r => r.Key, r => r.Value));
157-
});
171+
_resourcesCache.TryAdd(key, new ConcurrentDictionary<string, string>(resources));
172+
}
173+
}
174+
}
175+
176+
private void RefreshResourcesCache(object sender, FileSystemEventArgs e)
177+
{
178+
var key = Path.GetFileNameWithoutExtension(e.FullPath);
179+
if (_resourcesCache.TryGetValue(key, out var resources))
180+
{
181+
if (!resources.IsEmpty)
182+
{
183+
resources.Clear();
184+
185+
var freshResources = JsonResourceLoader.Load(e.FullPath);
186+
187+
foreach (var item in freshResources)
188+
{
189+
_resourcesCache[key].TryAdd(item.Key, item.Value);
190+
}
191+
}
158192
}
159193
}
160194
}
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
using BenchmarkDotNet.Attributes;
22
using My.Extensions.Localization.Json.Internal;
33
using System.Globalization;
4+
using System.IO;
45

56
namespace My.Extensions.Localization.Json.Benchmarks;
67

@@ -12,19 +13,16 @@ public class JsonResourceManagerBenchmark
1213

1314
static JsonResourceManagerBenchmark()
1415
{
15-
_jsonResourceManager = new JsonResourceManager("Resources\\fr-FR.json");
16+
var resources = "Resources";
17+
_jsonResourceManager = new JsonResourceManager(resources, Path.Combine("fr-FR.json"));
1618
_frenchCulture = CultureInfo.GetCultureInfo("fr-FR");
1719
}
1820

1921
[Benchmark]
2022
public void EvaluateGetResourceSetWithoutCultureFallback()
21-
{
22-
_jsonResourceManager.GetResourceSet(_frenchCulture, tryParents: false);
23-
}
23+
=> _jsonResourceManager.GetResourceSet(_frenchCulture, tryParents: false);
2424

2525
[Benchmark]
2626
public void EvaluateGetResourceSetWithCultureFallback()
27-
{
28-
_jsonResourceManager.GetResourceSet(_frenchCulture, tryParents: true);
29-
}
27+
=> _jsonResourceManager.GetResourceSet(_frenchCulture, tryParents: true);
3028
}

0 commit comments

Comments
 (0)