Skip to content

Commit e0a18ac

Browse files
committed
Refactor common caching semantics into ICacheService
This change reduces the repentitiveness of implementing request-and-cache or check-cache-else-request patterns when requesting provider metadata. This simplified code in each provider, and also centralized testing that behavior on the CacheService class rather than duplicating cache behavior testing. It also removes the GetCatalogAsync and GetMetadataAsync as those names were Cdnjs-specifc, and the catalog is just another form of metadata. Testing was also largely refactored to apply fakes at the ICacheService layer instead of the underlying IWebRequestHandler, as that should be transparent to the Catalog implementations. There are a couple exceptions to verify that no web requests are issued at all during some test scenarios.
1 parent 28225d0 commit e0a18ac

File tree

10 files changed

+444
-569
lines changed

10 files changed

+444
-569
lines changed

src/LibraryManager.Contracts/Caching/ICacheService.cs

+9-8
Original file line numberDiff line numberDiff line change
@@ -12,21 +12,22 @@ namespace Microsoft.Web.LibraryManager.Contracts.Caching
1212
public interface ICacheService
1313
{
1414
/// <summary>
15-
/// Returns the provider's catalog from the provided Url to cacheFile
15+
/// Gets the contents from the specified URL, or if the request fails then from a locally cached copy
1616
/// </summary>
17-
/// <param name="url">Url to the provider catalog</param>
18-
/// <param name="cacheFile">Where to store the provider catalog within the cache</param>
17+
/// <param name="url">The URL to request</param>
18+
/// <param name="cacheFile">The locally cached file</param>
1919
/// <param name="cancellationToken">Cancellation token</param>
2020
/// <returns></returns>
21-
Task<string> GetCatalogAsync(string url, string cacheFile, CancellationToken cancellationToken);
21+
Task<string> GetContentsFromUriWithCacheFallbackAsync(string url, string cacheFile, CancellationToken cancellationToken);
2222

2323
/// <summary>
24-
/// Returns library metadata from provided Url to cacheFile
24+
/// Gets the contents of a local cache file, or if the file does not exist then requests it from the specified URL
2525
/// </summary>
26-
/// <param name="url">Url to the library metadata</param>
27-
/// <param name="cacheFile">Where to store the metadata file within the cache</param>
26+
/// <param name="cacheFile">The locally cached file</param>
27+
/// <param name="url">The URL to request</param>
2828
/// <param name="cancellationToken">Cancellation token</param>
2929
/// <returns></returns>
30-
Task<string> GetMetadataAsync(string url, string cacheFile, CancellationToken cancellationToken);
30+
/// <exception cref="ResourceDownloadException">Thrown when the file doesn't exist and the resource download fails</exception>
31+
Task<string> GetContentsFromCachedFileWithWebRequestFallbackAsync(string cacheFile, string url, CancellationToken cancellationToken);
3132
}
3233
}

src/LibraryManager/CacheService/CacheService.cs

+41-27
Original file line numberDiff line numberDiff line change
@@ -17,9 +17,7 @@ namespace Microsoft.Web.LibraryManager
1717
/// </summary>
1818
public class CacheService : ICacheService
1919
{
20-
// TO DO: Move these expirations to the provider
21-
private const int CatalogExpiresAfterDays = 1;
22-
private const int MetadataExpiresAfterDays = 1;
20+
private const int DefaultCacheExpiresAfterDays = 1;
2321
private const int MaxConcurrentDownloads = 10;
2422

2523
private readonly IWebRequestHandler _requestHandler;
@@ -52,30 +50,6 @@ public static string CacheFolder
5250
}
5351
}
5452

55-
/// <summary>
56-
/// Returns the provider's catalog from the provided Url to cacheFile
57-
/// </summary>
58-
/// <param name="url"></param>
59-
/// <param name="cacheFile"></param>
60-
/// <param name="cancellationToken"></param>
61-
/// <returns></returns>
62-
public async Task<string> GetCatalogAsync(string url, string cacheFile, CancellationToken cancellationToken)
63-
{
64-
return await GetResourceAsync(url, cacheFile, CatalogExpiresAfterDays, cancellationToken);
65-
}
66-
67-
/// <summary>
68-
/// Returns library metadata from provided Url to cacheFile
69-
/// </summary>
70-
/// <param name="url"></param>
71-
/// <param name="cacheFile"></param>
72-
/// <param name="cancellationToken"></param>
73-
/// <returns></returns>
74-
public async Task<string> GetMetadataAsync(string url, string cacheFile, CancellationToken cancellationToken)
75-
{
76-
return await GetResourceAsync(url, cacheFile, MetadataExpiresAfterDays, cancellationToken);
77-
}
78-
7953
/// <summary>
8054
/// Downloads a resource from specified url to a destination file
8155
/// </summary>
@@ -141,5 +115,45 @@ async Task DownloadFileIfNecessaryAsync(CacheFileMetadata metadata)
141115
}
142116
}
143117
}
118+
119+
/// <inheritdoc />
120+
public async Task<string> GetContentsFromUriWithCacheFallbackAsync(string url, string cacheFile, CancellationToken cancellationToken)
121+
{
122+
string contents;
123+
try
124+
{
125+
contents = await GetResourceAsync(url, cacheFile, DefaultCacheExpiresAfterDays, cancellationToken);
126+
}
127+
catch (ResourceDownloadException)
128+
{
129+
// TODO: Log telemetry
130+
if (File.Exists(cacheFile))
131+
{
132+
contents = await FileHelpers.ReadFileAsTextAsync(cacheFile, cancellationToken);
133+
}
134+
else
135+
{
136+
throw;
137+
}
138+
}
139+
140+
return contents;
141+
}
142+
143+
/// <inheritdoc />
144+
public async Task<string> GetContentsFromCachedFileWithWebRequestFallbackAsync(string cacheFile, string url, CancellationToken cancellationToken)
145+
{
146+
string contents;
147+
if (File.Exists(cacheFile))
148+
{
149+
contents = await FileHelpers.ReadFileAsTextAsync(cacheFile, cancellationToken);
150+
}
151+
else
152+
{
153+
contents = await GetResourceAsync(url, cacheFile, DefaultCacheExpiresAfterDays, cancellationToken);
154+
}
155+
156+
return contents;
157+
}
144158
}
145159
}

src/LibraryManager/Providers/Cdnjs/CdnjsCatalog.cs

+4-37
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,8 @@ internal class CdnjsCatalog : ILibraryCatalog
1919
{
2020
// TO DO: These should become Provider properties to be passed to CacheService
2121
private const string FileName = "cache.json";
22-
private const string RemoteApiUrl = "https://aka.ms/g8irvu";
23-
private const string MetaPackageUrlFormat = "https://api.cdnjs.com/libraries/{0}"; // https://aka.ms/goycwu/{0}
22+
public const string CatalogUrl = "https://api.cdnjs.com/libraries?fields=name,description,version";
23+
public const string MetaPackageUrlFormat = "https://api.cdnjs.com/libraries/{0}"; // https://aka.ms/goycwu/{0}
2424

2525
private readonly string _cacheFile;
2626
private readonly CdnjsProvider _provider;
@@ -245,24 +245,7 @@ private async Task<bool> EnsureCatalogAsync(CancellationToken cancellationToken)
245245

246246
try
247247
{
248-
string json;
249-
try
250-
{
251-
json = await _cacheService.GetCatalogAsync(RemoteApiUrl, _cacheFile, cancellationToken).ConfigureAwait(false);
252-
}
253-
catch (ResourceDownloadException)
254-
{
255-
// TODO: add telemetry
256-
_provider.HostInteraction.Logger.Log(string.Format(Resources.Text.FailedToDownloadCatalog, _provider.Id), LogLevel.Operation);
257-
if (File.Exists(_cacheFile))
258-
{
259-
json = await FileHelpers.ReadFileAsTextAsync(_cacheFile, cancellationToken);
260-
}
261-
else
262-
{
263-
return false;
264-
}
265-
}
248+
string json = await _cacheService.GetContentsFromUriWithCacheFallbackAsync(CatalogUrl, _cacheFile, cancellationToken);
266249

267250
if (string.IsNullOrWhiteSpace(json))
268251
{
@@ -295,23 +278,7 @@ private async Task<IEnumerable<Asset>> GetAssetsAsync(string groupName, Cancella
295278

296279
try
297280
{
298-
string json;
299-
try
300-
{
301-
json = await _cacheService.GetMetadataAsync(url, localFile, cancellationToken).ConfigureAwait(false);
302-
}
303-
catch (ResourceDownloadException)
304-
{
305-
// TODO: Log telemetry
306-
if (File.Exists(localFile))
307-
{
308-
json = await FileHelpers.ReadFileAsTextAsync(localFile, cancellationToken);
309-
}
310-
else
311-
{
312-
throw;
313-
}
314-
}
281+
string json = await _cacheService.GetContentsFromUriWithCacheFallbackAsync(url, localFile, cancellationToken);
315282

316283
if (!string.IsNullOrEmpty(json))
317284
{

src/LibraryManager/Providers/Unpkg/UnpkgCatalog.cs

+7-34
Original file line numberDiff line numberDiff line change
@@ -48,25 +48,11 @@ public async Task<string> GetLatestVersion(string libraryName, bool includePreRe
4848
try
4949
{
5050
string latestLibraryVersionUrl = string.Format(LatestLibraryVersonUrl, libraryName);
51-
string latestCacheFile = Path.Combine(_cacheFolder, $"{libraryName}-{LatestVersionTag}.json");
51+
string latestCacheFile = Path.Combine(_cacheFolder, libraryName, $"{LatestVersionTag}.json");
5252

53-
string latestJson;
54-
try
55-
{
56-
latestJson = await _cacheService.GetMetadataAsync(latestLibraryVersionUrl, latestCacheFile, cancellationToken);
57-
}
58-
catch (ResourceDownloadException)
59-
{
60-
// TODO: add telemetry
61-
if (File.Exists(latestCacheFile))
62-
{
63-
latestJson = await FileHelpers.ReadFileAsTextAsync(latestCacheFile, cancellationToken);
64-
}
65-
else
66-
{
67-
throw;
68-
}
69-
}
53+
string latestJson = await _cacheService.GetContentsFromUriWithCacheFallbackAsync(latestLibraryVersionUrl,
54+
latestCacheFile,
55+
cancellationToken);
7056

7157
var packageObject = (JObject)JsonConvert.DeserializeObject(latestJson);
7258

@@ -123,22 +109,9 @@ private async Task<IEnumerable<string>> GetLibraryFilesAsync(string libraryName,
123109
string libraryFileListUrl = string.Format(LibraryFileListUrlFormat, libraryName, version);
124110
string libraryFileListCacheFile = Path.Combine(_cacheFolder, libraryName, $"{version}-filelist.json");
125111

126-
string fileList;
127-
try
128-
{
129-
fileList = await _cacheService.GetMetadataAsync(libraryFileListUrl, libraryFileListCacheFile, cancellationToken);
130-
}
131-
catch (ResourceDownloadException)
132-
{
133-
if (File.Exists(libraryFileListCacheFile))
134-
{
135-
fileList = await FileHelpers.ReadFileAsTextAsync(libraryFileListCacheFile, cancellationToken);
136-
}
137-
else
138-
{
139-
throw;
140-
}
141-
}
112+
string fileList = await _cacheService.GetContentsFromCachedFileWithWebRequestFallbackAsync(libraryFileListCacheFile,
113+
libraryFileListUrl,
114+
cancellationToken);
142115

143116
var fileListObject = (JObject)JsonConvert.DeserializeObject(fileList);
144117

src/LibraryManager/Providers/jsDelivr/JsDelivrCatalog.cs

+10-45
Original file line numberDiff line numberDiff line change
@@ -61,28 +61,13 @@ public async Task<string> GetLatestVersion(string libraryId, bool includePreRele
6161
bool isGitHub = IsGitHub(libraryId);
6262
string latestLibraryVersionUrl = string.Format(isGitHub ? LatestLibraryVersionUrlGH : LatestLibraryVersionUrl, name);
6363
string cacheFileType = isGitHub ? "github" : "npm";
64-
string latestLibraryVersionCacheFile = Path.Combine(_cacheFolder, $"{name.Replace("/", "_")}-{cacheFileType}-{LatestVersionTag}.json");
64+
string latestLibraryVersionCacheFile = Path.Combine(_cacheFolder, name, $"{cacheFileType}-{LatestVersionTag}.json");
6565

66-
string latestVersionContent;
67-
try
68-
{
69-
latestVersionContent = await _cacheService.GetMetadataAsync(latestLibraryVersionUrl, latestLibraryVersionCacheFile, cancellationToken);
70-
}
71-
catch (ResourceDownloadException)
72-
{
73-
// TODO: Log telemetry
74-
if (File.Exists(latestLibraryVersionCacheFile))
75-
{
76-
latestVersionContent = await FileHelpers.ReadFileAsTextAsync(latestLibraryVersionCacheFile, cancellationToken);
77-
}
78-
else
79-
{
80-
throw;
81-
}
82-
}
66+
string latestVersionContent = await _cacheService.GetContentsFromUriWithCacheFallbackAsync(latestLibraryVersionUrl,
67+
latestLibraryVersionCacheFile,
68+
cancellationToken);
8369

8470
var packageObject = (JObject)JsonConvert.DeserializeObject(latestVersionContent);
85-
8671
if (packageObject != null)
8772
{
8873
var versions = packageObject["tags"] as JObject;
@@ -129,16 +114,10 @@ private async Task<IEnumerable<string>> GetLibraryFilesAsync(string libraryId, C
129114

130115
(string libraryName, string libraryVersion) = _libraryNamingScheme.GetLibraryNameAndVersion(libraryId);
131116
string libraryFileListCacheFile = Path.Combine(_cacheFolder, libraryName, $"{libraryVersion}-filelist.json");
132-
string fileListJson;
133-
if (File.Exists(libraryFileListCacheFile))
134-
{
135-
fileListJson = await FileHelpers.ReadFileAsTextAsync(libraryFileListCacheFile, cancellationToken);
136-
}
137-
else
138-
{
139-
string libraryFileListUrl = string.Format(IsGitHub(libraryId) ? LibraryFileListUrlFormatGH : LibraryFileListUrlFormat, libraryId);
140-
fileListJson = await _cacheService.GetMetadataAsync(libraryFileListUrl, libraryFileListCacheFile, cancellationToken);
141-
}
117+
string libraryFileListUrl = string.Format(IsGitHub(libraryId) ? LibraryFileListUrlFormatGH : LibraryFileListUrlFormat, libraryId);
118+
string fileListJson = await _cacheService.GetContentsFromCachedFileWithWebRequestFallbackAsync(libraryFileListCacheFile,
119+
libraryFileListUrl,
120+
cancellationToken);
142121

143122
if ((JObject)JsonConvert.DeserializeObject(fileListJson) is var fileListObject)
144123
{
@@ -326,22 +305,8 @@ private async Task<IEnumerable<string>> GetGithubLibraryVersionsAsync(string nam
326305
{
327306
var versions = new List<string>();
328307
string versionsCacheFile = Path.Combine(_cacheFolder, name, "github-versions-cache.json");
329-
string versionsJson;
330-
try
331-
{
332-
versionsJson = await _cacheService.GetMetadataAsync(string.Format(LatestLibraryVersionUrlGH, name), versionsCacheFile, cancellationToken);
333-
}
334-
catch (ResourceDownloadException)
335-
{
336-
if (File.Exists(versionsCacheFile))
337-
{
338-
versionsJson = await FileHelpers.ReadFileAsTextAsync(versionsCacheFile, cancellationToken);
339-
}
340-
else
341-
{
342-
throw;
343-
}
344-
}
308+
string versionsUrl = string.Format(LatestLibraryVersionUrlGH, name);
309+
string versionsJson = await _cacheService.GetContentsFromUriWithCacheFallbackAsync(versionsUrl, versionsCacheFile, cancellationToken);
345310

346311
var versionsObject = (JObject)JsonConvert.DeserializeObject(versionsJson);
347312
var versionsArray = versionsObject["versions"] as JArray;

0 commit comments

Comments
 (0)