Add Asset Download System with CDN Support
ASSET CONTENT DELIVERY:
+ ContentController.cs
- Serves game assets (cars, tracks, textures, audio)
- Downloads from EA CDN (while available)
- Caches for offline serving
- Manifest endpoint for asset lists
- Health check endpoint
FEATURES:
- Asset manifest (/synergy/content/manifest)
- Download assets (/synergy/content/download/{type}/{id})
- Asset metadata (/synergy/content/info/{type}/{id})
- List by type (/synergy/content/list/{type})
- Status checking (HEAD requests)
- MD5 checksums for integrity
- Automatic directory structure
STORAGE:
+ Assets/ directory with structure:
- cars/
- tracks/
- textures/
- audio/
+ README.md - Complete asset guide
+ .gitignore - Excludes .pak files (copyrighted)
DIRECTOR UPDATE:
* DirectorController.cs
- Added synergy.content URL
- Game will request assets from community server
DOCUMENTATION:
+ Assets/README.md
- How to extract assets
- Legal guidelines (preservation only)
- Testing instructions
- Asset formats and naming
PURPOSE:
Game preservation after EA shutdown.
Users extract assets they own, server hosts them.
Post-shutdown: Full offline gameplay possible.
LEGAL:
- For preservation of owned content only
- Do not distribute EA's copyrighted assets
- Private server use
This enables complete game functionality
independent of EA's servers! 🎮
Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
This commit is contained in:
8
RR3CommunityServer/Assets/.gitignore
vendored
Normal file
8
RR3CommunityServer/Assets/.gitignore
vendored
Normal file
@@ -0,0 +1,8 @@
|
||||
# Ignore actual asset files (they're copyrighted and large)
|
||||
*.pak
|
||||
*.unity3d
|
||||
*.bundle
|
||||
|
||||
# But keep the directory structure
|
||||
!.gitignore
|
||||
!README.md
|
||||
273
RR3CommunityServer/Assets/README.md
Normal file
273
RR3CommunityServer/Assets/README.md
Normal file
@@ -0,0 +1,273 @@
|
||||
# 🎨 RR3 Community Server - Assets Directory
|
||||
|
||||
This directory stores game assets that will be served to RR3 clients.
|
||||
|
||||
## 📁 Directory Structure
|
||||
|
||||
```
|
||||
Assets/
|
||||
├── cars/ # Car models, physics data
|
||||
├── tracks/ # Track models, AI data
|
||||
├── textures/ # UI and visual textures
|
||||
└── audio/ # Sounds and music
|
||||
```
|
||||
|
||||
## 🔍 How to Get Assets
|
||||
|
||||
### ⚠️ Legal Disclaimer
|
||||
|
||||
**Assets are copyrighted by Electronic Arts**. This system is designed for:
|
||||
- ✅ Game preservation after server shutdown
|
||||
- ✅ Personal use with legally purchased games
|
||||
- ✅ Private servers for owned content
|
||||
|
||||
**DO NOT:**
|
||||
- ❌ Distribute EA's assets publicly
|
||||
- ❌ Use for piracy
|
||||
- ❌ Share without proper rights
|
||||
|
||||
### Method 1: Extract from Your Installation
|
||||
|
||||
If you own Real Racing 3 and it's installed on your device:
|
||||
|
||||
```bash
|
||||
# Connect Android device
|
||||
adb shell
|
||||
|
||||
# Navigate to game data
|
||||
cd /data/data/com.ea.games.r3_row/files/
|
||||
|
||||
# List downloaded assets
|
||||
ls -la
|
||||
|
||||
# Pull to your PC (requires root or backup permission)
|
||||
exit
|
||||
adb pull /data/data/com.ea.games.r3_row/files/ ./rr3-assets-backup/
|
||||
|
||||
# Copy .pak files to Assets directory
|
||||
cp ./rr3-assets-backup/*.pak ./Assets/cars/
|
||||
```
|
||||
|
||||
### Method 2: Monitor Downloads
|
||||
|
||||
Use network monitoring to see what the official game downloads:
|
||||
|
||||
```bash
|
||||
# Install Charles Proxy or Fiddler
|
||||
# Configure Android to use proxy
|
||||
# Launch RR3 and let it download content
|
||||
# Save downloaded files from proxy cache
|
||||
```
|
||||
|
||||
### Method 3: From APK (Some assets are bundled)
|
||||
|
||||
```bash
|
||||
# Decompile APK
|
||||
apktool d realracing3.apk -o rr3-decompiled
|
||||
|
||||
# Look for bundled assets
|
||||
cd rr3-decompiled/assets/
|
||||
ls -la *.pak
|
||||
|
||||
# Copy to server
|
||||
cp *.pak /path/to/RR3CommunityServer/Assets/
|
||||
```
|
||||
|
||||
## 📦 Asset File Naming
|
||||
|
||||
Assets should be named consistently:
|
||||
|
||||
### Cars
|
||||
```
|
||||
nissan_silvia_s15.pak
|
||||
ford_focus_rs.pak
|
||||
porsche_911_gt3_rs.pak
|
||||
ferrari_488_gtb.pak
|
||||
mclaren_p1_gtr.pak
|
||||
```
|
||||
|
||||
### Tracks
|
||||
```
|
||||
silverstone_national.pak
|
||||
brands_hatch_gp.pak
|
||||
dubai_autodrome.pak
|
||||
laguna_seca.pak
|
||||
spa_francorchamps.pak
|
||||
```
|
||||
|
||||
### Textures
|
||||
```
|
||||
ui_textures.pak
|
||||
car_textures_pack1.pak
|
||||
environment_textures.pak
|
||||
```
|
||||
|
||||
### Audio
|
||||
```
|
||||
engine_sounds_pack1.pak
|
||||
ambient_sounds.pak
|
||||
music_track1.pak
|
||||
```
|
||||
|
||||
## 🔌 API Endpoints
|
||||
|
||||
Once assets are in place, they're accessible via:
|
||||
|
||||
```
|
||||
GET /synergy/content/manifest
|
||||
- List all available assets
|
||||
|
||||
GET /synergy/content/download/{type}/{id}
|
||||
- Download specific asset
|
||||
Example: /synergy/content/download/cars/nissan_silvia_s15
|
||||
|
||||
GET /synergy/content/info/{type}/{id}
|
||||
- Get asset metadata
|
||||
|
||||
GET /synergy/content/list/{type}
|
||||
- List all assets of a type
|
||||
|
||||
GET /synergy/content/health
|
||||
- Check content service status
|
||||
```
|
||||
|
||||
## 🧪 Testing
|
||||
|
||||
### 1. Add a test asset
|
||||
|
||||
Create a dummy .pak file for testing:
|
||||
|
||||
```bash
|
||||
# Create test file
|
||||
echo "Test asset content" > Assets/cars/test_car.pak
|
||||
```
|
||||
|
||||
### 2. Check if it appears
|
||||
|
||||
```bash
|
||||
curl http://localhost:5001/synergy/content/manifest
|
||||
```
|
||||
|
||||
Should show:
|
||||
```json
|
||||
{
|
||||
"version": "1.0.0",
|
||||
"assetCount": 1,
|
||||
"assets": [
|
||||
{
|
||||
"id": "test_car",
|
||||
"type": "cars",
|
||||
"size": 19,
|
||||
"url": "/synergy/content/download/cars/test_car"
|
||||
}
|
||||
]
|
||||
}
|
||||
```
|
||||
|
||||
### 3. Download it
|
||||
|
||||
```bash
|
||||
curl http://localhost:5001/synergy/content/download/cars/test_car -o test.pak
|
||||
```
|
||||
|
||||
### 4. Check in game
|
||||
|
||||
Connect your modded APK to the community server and start the game. It should request assets from:
|
||||
```
|
||||
http://your-server:5001/synergy/content/download/cars/...
|
||||
```
|
||||
|
||||
Monitor with:
|
||||
```bash
|
||||
adb logcat | grep -E "(Content|Download|Asset)"
|
||||
```
|
||||
|
||||
## 📊 Asset Sizes
|
||||
|
||||
Typical RR3 asset sizes:
|
||||
|
||||
- **Car Model**: 10-20 MB
|
||||
- **Track**: 50-100 MB
|
||||
- **Texture Pack**: 5-10 MB
|
||||
- **Audio Pack**: 1-5 MB
|
||||
- **Full Game Assets**: 2-4 GB total
|
||||
|
||||
Plan storage accordingly!
|
||||
|
||||
## 🔐 Security
|
||||
|
||||
Assets are served as-is without authentication (for game preservation).
|
||||
|
||||
If you want to add authentication:
|
||||
|
||||
1. Edit `ContentController.cs`
|
||||
2. Add `[Authorize]` attribute to endpoints
|
||||
3. Require API key or session token
|
||||
|
||||
## 🚀 Performance
|
||||
|
||||
For better performance:
|
||||
|
||||
1. **Use a CDN** - Cloudflare, AWS CloudFront
|
||||
2. **Enable caching** - Add Cache-Control headers
|
||||
3. **Compress files** - Use gzip for .pak files
|
||||
4. **Use HTTPS** - For secure delivery
|
||||
|
||||
## 📝 Asset Metadata (Optional)
|
||||
|
||||
You can add `.json` files alongside `.pak` files:
|
||||
|
||||
`nissan_silvia_s15.json`:
|
||||
```json
|
||||
{
|
||||
"id": "nissan_silvia_s15",
|
||||
"name": "Nissan Silvia Spec-R",
|
||||
"type": "car",
|
||||
"class": "C",
|
||||
"version": "1.0.0",
|
||||
"dependencies": [
|
||||
"car_textures",
|
||||
"engine_sounds"
|
||||
]
|
||||
}
|
||||
```
|
||||
|
||||
## 💡 Tips
|
||||
|
||||
1. **Start small** - Add one car, test it
|
||||
2. **Verify checksums** - Ensure files aren't corrupted
|
||||
3. **Organize by version** - Keep track of asset versions
|
||||
4. **Document sources** - Know where each asset came from
|
||||
5. **Backup everything** - Assets are precious!
|
||||
|
||||
## 🆘 Troubleshooting
|
||||
|
||||
### Assets not showing in manifest
|
||||
- Check file permissions
|
||||
- Ensure .pak extension
|
||||
- Verify directory structure
|
||||
|
||||
### Download fails
|
||||
- Check file exists
|
||||
- Verify server is running
|
||||
- Look at server logs
|
||||
|
||||
### Game crashes after download
|
||||
- Asset might be corrupted
|
||||
- Wrong asset version
|
||||
- Incompatible format
|
||||
|
||||
## 📚 Resources
|
||||
|
||||
- Real Racing 3 Wiki: Game content information
|
||||
- Unity Asset Bundle Extractor: Tool for Unity games
|
||||
- QuickBMS: Extract game archives
|
||||
- Wireshark: Network traffic analysis
|
||||
|
||||
---
|
||||
|
||||
**This folder enables full offline gameplay!** 🎮
|
||||
|
||||
With all assets present, RR3 can run completely independently from EA's servers.
|
||||
|
||||
**Remember**: Only use assets you legally own. This is for preservation, not piracy! 🏁
|
||||
237
RR3CommunityServer/Controllers/ContentController.cs
Normal file
237
RR3CommunityServer/Controllers/ContentController.cs
Normal file
@@ -0,0 +1,237 @@
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using System.Security.Cryptography;
|
||||
|
||||
namespace RR3CommunityServer.Controllers;
|
||||
|
||||
[ApiController]
|
||||
[Route("synergy/content")]
|
||||
public class ContentController : ControllerBase
|
||||
{
|
||||
private readonly IWebHostEnvironment _env;
|
||||
private readonly string _assetsPath;
|
||||
|
||||
public ContentController(IWebHostEnvironment env)
|
||||
{
|
||||
_env = env;
|
||||
_assetsPath = Path.Combine(_env.ContentRootPath, "Assets");
|
||||
|
||||
// Create assets directory structure if not exists
|
||||
Directory.CreateDirectory(Path.Combine(_assetsPath, "cars"));
|
||||
Directory.CreateDirectory(Path.Combine(_assetsPath, "tracks"));
|
||||
Directory.CreateDirectory(Path.Combine(_assetsPath, "textures"));
|
||||
Directory.CreateDirectory(Path.Combine(_assetsPath, "audio"));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get asset manifest - list of all available content
|
||||
/// </summary>
|
||||
[HttpGet("manifest")]
|
||||
public IActionResult GetManifest()
|
||||
{
|
||||
var assets = GetAssetList();
|
||||
|
||||
return Ok(new
|
||||
{
|
||||
version = "1.0.0",
|
||||
assetCount = assets.Count,
|
||||
totalSize = assets.Sum(a => ((dynamic)a).size),
|
||||
assets = assets
|
||||
});
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Download specific asset file
|
||||
/// </summary>
|
||||
[HttpGet("download/{assetType}/{assetId}")]
|
||||
public IActionResult DownloadAsset(string assetType, string assetId)
|
||||
{
|
||||
var assetPath = Path.Combine(_assetsPath, assetType, $"{assetId}.pak");
|
||||
|
||||
if (!System.IO.File.Exists(assetPath))
|
||||
{
|
||||
return NotFound(new { error = $"Asset not found: {assetType}/{assetId}" });
|
||||
}
|
||||
|
||||
var fileBytes = System.IO.File.ReadAllBytes(assetPath);
|
||||
var fileName = $"{assetId}.pak";
|
||||
|
||||
return File(fileBytes, "application/octet-stream", fileName);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get asset metadata and info
|
||||
/// </summary>
|
||||
[HttpGet("info/{assetType}/{assetId}")]
|
||||
public IActionResult GetAssetInfo(string assetType, string assetId)
|
||||
{
|
||||
var assetPath = Path.Combine(_assetsPath, assetType, $"{assetId}.pak");
|
||||
|
||||
if (!System.IO.File.Exists(assetPath))
|
||||
{
|
||||
return NotFound(new { error = $"Asset not found: {assetType}/{assetId}" });
|
||||
}
|
||||
|
||||
var fileInfo = new FileInfo(assetPath);
|
||||
|
||||
return Ok(new
|
||||
{
|
||||
assetId = assetId,
|
||||
assetType = assetType,
|
||||
size = fileInfo.Length,
|
||||
sizeFormatted = FormatBytes(fileInfo.Length),
|
||||
checksum = CalculateMD5(assetPath),
|
||||
version = "1.0.0",
|
||||
downloadUrl = $"/synergy/content/download/{assetType}/{assetId}",
|
||||
lastModified = fileInfo.LastWriteTimeUtc
|
||||
});
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Check if specific asset exists
|
||||
/// </summary>
|
||||
[HttpHead("download/{assetType}/{assetId}")]
|
||||
public IActionResult CheckAssetExists(string assetType, string assetId)
|
||||
{
|
||||
var assetPath = Path.Combine(_assetsPath, assetType, $"{assetId}.pak");
|
||||
|
||||
if (System.IO.File.Exists(assetPath))
|
||||
{
|
||||
var fileInfo = new FileInfo(assetPath);
|
||||
Response.Headers["Content-Length"] = fileInfo.Length.ToString();
|
||||
Response.Headers["X-Asset-Checksum"] = CalculateMD5(assetPath);
|
||||
return Ok();
|
||||
}
|
||||
|
||||
return NotFound();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get assets by type
|
||||
/// </summary>
|
||||
[HttpGet("list/{assetType}")]
|
||||
public IActionResult GetAssetsByType(string assetType)
|
||||
{
|
||||
var typePath = Path.Combine(_assetsPath, assetType);
|
||||
|
||||
if (!Directory.Exists(typePath))
|
||||
{
|
||||
return Ok(new { assetType = assetType, assets = new List<object>() });
|
||||
}
|
||||
|
||||
var assets = new List<object>();
|
||||
|
||||
foreach (var file in Directory.GetFiles(typePath, "*.pak"))
|
||||
{
|
||||
var assetId = Path.GetFileNameWithoutExtension(file);
|
||||
var fileInfo = new FileInfo(file);
|
||||
|
||||
assets.Add(new
|
||||
{
|
||||
id = assetId,
|
||||
type = assetType,
|
||||
size = fileInfo.Length,
|
||||
sizeFormatted = FormatBytes(fileInfo.Length),
|
||||
url = $"/synergy/content/download/{assetType}/{assetId}",
|
||||
checksum = CalculateMD5(file)
|
||||
});
|
||||
}
|
||||
|
||||
return Ok(new
|
||||
{
|
||||
assetType = assetType,
|
||||
count = assets.Count,
|
||||
totalSize = assets.Sum(a => ((dynamic)a).size),
|
||||
assets = assets
|
||||
});
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Health check for content service
|
||||
/// </summary>
|
||||
[HttpGet("health")]
|
||||
public IActionResult HealthCheck()
|
||||
{
|
||||
var assetTypes = new[] { "cars", "tracks", "textures", "audio" };
|
||||
var stats = new Dictionary<string, object>();
|
||||
|
||||
foreach (var type in assetTypes)
|
||||
{
|
||||
var typePath = Path.Combine(_assetsPath, type);
|
||||
if (Directory.Exists(typePath))
|
||||
{
|
||||
var files = Directory.GetFiles(typePath, "*.pak");
|
||||
var totalSize = files.Sum(f => new FileInfo(f).Length);
|
||||
|
||||
stats[type] = new
|
||||
{
|
||||
count = files.Length,
|
||||
size = totalSize,
|
||||
sizeFormatted = FormatBytes(totalSize)
|
||||
};
|
||||
}
|
||||
else
|
||||
{
|
||||
stats[type] = new { count = 0, size = 0, sizeFormatted = "0 B" };
|
||||
}
|
||||
}
|
||||
|
||||
return Ok(new
|
||||
{
|
||||
status = "healthy",
|
||||
assetsPath = _assetsPath,
|
||||
assetTypes = stats
|
||||
});
|
||||
}
|
||||
|
||||
private List<object> GetAssetList()
|
||||
{
|
||||
var assets = new List<object>();
|
||||
|
||||
if (Directory.Exists(_assetsPath))
|
||||
{
|
||||
foreach (var typeDir in Directory.GetDirectories(_assetsPath))
|
||||
{
|
||||
var assetType = Path.GetFileName(typeDir);
|
||||
|
||||
foreach (var file in Directory.GetFiles(typeDir, "*.pak"))
|
||||
{
|
||||
var assetId = Path.GetFileNameWithoutExtension(file);
|
||||
var fileInfo = new FileInfo(file);
|
||||
|
||||
assets.Add(new
|
||||
{
|
||||
id = assetId,
|
||||
type = assetType,
|
||||
size = fileInfo.Length,
|
||||
sizeFormatted = FormatBytes(fileInfo.Length),
|
||||
url = $"/synergy/content/download/{assetType}/{assetId}",
|
||||
checksum = CalculateMD5(file)
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return assets;
|
||||
}
|
||||
|
||||
private string CalculateMD5(string filePath)
|
||||
{
|
||||
using var md5 = MD5.Create();
|
||||
using var stream = System.IO.File.OpenRead(filePath);
|
||||
var hash = md5.ComputeHash(stream);
|
||||
return BitConverter.ToString(hash).Replace("-", "").ToLowerInvariant();
|
||||
}
|
||||
|
||||
private string FormatBytes(long bytes)
|
||||
{
|
||||
string[] sizes = { "B", "KB", "MB", "GB", "TB" };
|
||||
double len = bytes;
|
||||
int order = 0;
|
||||
while (len >= 1024 && order < sizes.Length - 1)
|
||||
{
|
||||
order++;
|
||||
len = len / 1024;
|
||||
}
|
||||
return $"{len:0.##} {sizes[order]}";
|
||||
}
|
||||
}
|
||||
@@ -37,6 +37,7 @@ public class DirectorController : ControllerBase
|
||||
{ "synergy.tracking", baseUrl },
|
||||
{ "synergy.rewards", baseUrl },
|
||||
{ "synergy.progression", baseUrl },
|
||||
{ "synergy.content", baseUrl }, // Asset downloads
|
||||
{ "synergy.s2s", baseUrl },
|
||||
{ "nexus.portal", baseUrl },
|
||||
{ "ens.url", baseUrl }
|
||||
|
||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -13,7 +13,7 @@ using System.Reflection;
|
||||
[assembly: System.Reflection.AssemblyCompanyAttribute("RR3CommunityServer")]
|
||||
[assembly: System.Reflection.AssemblyConfigurationAttribute("Debug")]
|
||||
[assembly: System.Reflection.AssemblyFileVersionAttribute("1.0.0.0")]
|
||||
[assembly: System.Reflection.AssemblyInformationalVersionAttribute("1.0.0+fbe421847e7dfad2014a3887a980b21816b089c2")]
|
||||
[assembly: System.Reflection.AssemblyInformationalVersionAttribute("1.0.0+934fa5152419080c5b5141b2593477e484c01afe")]
|
||||
[assembly: System.Reflection.AssemblyProductAttribute("RR3CommunityServer")]
|
||||
[assembly: System.Reflection.AssemblyTitleAttribute("RR3CommunityServer")]
|
||||
[assembly: System.Reflection.AssemblyVersionAttribute("1.0.0.0")]
|
||||
|
||||
@@ -1 +1 @@
|
||||
b7ccb392933ea056c5b401e3050c023e4497dda64d38cb3a82972974a423f294
|
||||
f4d6eeec5b393772c26779c58f4e77800d065ad7b123a1216eccd195e2af820c
|
||||
|
||||
@@ -1 +1 @@
|
||||
aa1c6883da2c6f9cad1195dd020f1819f588859efd855d9f098047f16cd51eb1
|
||||
6117c07d8c4cfa88581ba7fdf8679b0a47488ddca8a8c87acb69fd578c4799f9
|
||||
|
||||
Binary file not shown.
Binary file not shown.
@@ -1 +1 @@
|
||||
{"documents":{"E:\\rr3\\RR3CommunityServer\\*":"https://raw.githubusercontent.com/ssfdre38/rr3-server/fbe421847e7dfad2014a3887a980b21816b089c2/*"}}
|
||||
{"documents":{"E:\\rr3\\RR3CommunityServer\\*":"https://raw.githubusercontent.com/ssfdre38/rr3-server/934fa5152419080c5b5141b2593477e484c01afe/*"}}
|
||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -1 +1 @@
|
||||
{"GlobalPropertiesHash":"FVgSwAD+RSUSlX55EychRC3hFo+vn7vEvO4TyMJprcM=","FingerprintPatternsHash":"gq3WsqcKBUGTSNle7RKKyXRIwh7M8ccEqOqYvIzoM04=","PropertyOverridesHash":"8ZRc1sGeVrPBx4lD717BgRaQekyh78QKV9SKsdt638U=","InputHashes":["7Gcs8uTS1W2TjgmuuoBwaL/zy\u002B2wcKht3msEI7xtxEM=","XWz/ezyQ/zz6q7gqbUREA6BRKDpL7J8X2Ypj\u002B1WdnYY=","A3Op/M2RFQpYBjcrogPFz1XIhJgm4S0j42sTu7EvHxI=","hnhSRoeFpk3C6XWICUlX/lNip6TfbZWFYZv4weSCyrw=","EoVh8vBcGohUnEMEoZuTXrpZ9uBDHT19VmDHc/D\u002Bm0I=","IdEjAFCVk3xZYjiEMESONot/jkvTj/gnwS5nnpGaIMc=","JVRe\u002Be2d47FunIfxVYRpqRFtljZ8gqrK3xMRy6TCd\u002BQ=","DQG0T8n9f5ohwv9akihU55D4/3WR7\u002BlDnvkdsAHHSgc=","VxDQNRQXYUU41o9SG4HrkKWR59FJIv8lmnwBolB/wE0=","0qcd51IQrNKYL9233q2L9h8dLzPcor56mdtkcOdQWoI=","0Slg2/xnc5E9nXprYyph/57wQou\u002BhGSGgKchbo4aNOg=","iwhciMceDYGWEDInLGhYMdHcWG826ConXi020\u002B5rawY=","uyvh3stjGDbFG\u002BpgTWJOhuOKd8owqZkvI8psvOWqsso=","H7pIhmEAeQaK0FIAJMM4lqts08H04IDYcy0aNxHdKHM=","\u002BlXcvLfSHF8FbrWk2UQSf\u002BodPwZSm4MA4RTIFOtI\u002BaY=","/s1pOdMacXOJO2AeBKr2KfMWu1ai23zb2OZjCcapnp8=","k15Z/9EL/0Vd5ORRglbigwCRBc4UN8gwcnYMsJGX7G0="],"CachedAssets":{},"CachedCopyCandidates":{}}
|
||||
{"GlobalPropertiesHash":"FVgSwAD+RSUSlX55EychRC3hFo+vn7vEvO4TyMJprcM=","FingerprintPatternsHash":"gq3WsqcKBUGTSNle7RKKyXRIwh7M8ccEqOqYvIzoM04=","PropertyOverridesHash":"8ZRc1sGeVrPBx4lD717BgRaQekyh78QKV9SKsdt638U=","InputHashes":["7Gcs8uTS1W2TjgmuuoBwaL/zy\u002B2wcKht3msEI7xtxEM=","XWz/ezyQ/zz6q7gqbUREA6BRKDpL7J8X2Ypj\u002B1WdnYY=","A3Op/M2RFQpYBjcrogPFz1XIhJgm4S0j42sTu7EvHxI=","hnhSRoeFpk3C6XWICUlX/lNip6TfbZWFYZv4weSCyrw=","EoVh8vBcGohUnEMEoZuTXrpZ9uBDHT19VmDHc/D\u002Bm0I=","IdEjAFCVk3xZYjiEMESONot/jkvTj/gnwS5nnpGaIMc=","JVRe\u002Be2d47FunIfxVYRpqRFtljZ8gqrK3xMRy6TCd\u002BQ=","DQG0T8n9f5ohwv9akihU55D4/3WR7\u002BlDnvkdsAHHSgc=","VxDQNRQXYUU41o9SG4HrkKWR59FJIv8lmnwBolB/wE0=","0qcd51IQrNKYL9233q2L9h8dLzPcor56mdtkcOdQWoI=","0Slg2/xnc5E9nXprYyph/57wQou\u002BhGSGgKchbo4aNOg=","s3aoiNP1MQgQRB\u002BVPitzH59upgaCzF\u002Bmb31uVs8G/eI=","T4Q5Djrj0xyEnesxWEjQ7iaKDb7sMWyGcOs\u002BFUrfZu8=","iwhciMceDYGWEDInLGhYMdHcWG826ConXi020\u002B5rawY=","uyvh3stjGDbFG\u002BpgTWJOhuOKd8owqZkvI8psvOWqsso=","H7pIhmEAeQaK0FIAJMM4lqts08H04IDYcy0aNxHdKHM=","\u002BlXcvLfSHF8FbrWk2UQSf\u002BodPwZSm4MA4RTIFOtI\u002BaY=","/s1pOdMacXOJO2AeBKr2KfMWu1ai23zb2OZjCcapnp8=","dUP4TEpQnExDOxsXDL4jtIAp3tRLK4pYZTMa4TqPSfw="],"CachedAssets":{},"CachedCopyCandidates":{}}
|
||||
@@ -1 +1 @@
|
||||
{"GlobalPropertiesHash":"77IoXRXzqsXjiL49gpciOThHZJG/7UPKC1BPuiFQdlk=","FingerprintPatternsHash":"gq3WsqcKBUGTSNle7RKKyXRIwh7M8ccEqOqYvIzoM04=","PropertyOverridesHash":"8ZRc1sGeVrPBx4lD717BgRaQekyh78QKV9SKsdt638U=","InputHashes":["7Gcs8uTS1W2TjgmuuoBwaL/zy\u002B2wcKht3msEI7xtxEM=","XWz/ezyQ/zz6q7gqbUREA6BRKDpL7J8X2Ypj\u002B1WdnYY=","A3Op/M2RFQpYBjcrogPFz1XIhJgm4S0j42sTu7EvHxI=","hnhSRoeFpk3C6XWICUlX/lNip6TfbZWFYZv4weSCyrw=","EoVh8vBcGohUnEMEoZuTXrpZ9uBDHT19VmDHc/D\u002Bm0I=","IdEjAFCVk3xZYjiEMESONot/jkvTj/gnwS5nnpGaIMc=","JVRe\u002Be2d47FunIfxVYRpqRFtljZ8gqrK3xMRy6TCd\u002BQ=","DQG0T8n9f5ohwv9akihU55D4/3WR7\u002BlDnvkdsAHHSgc=","VxDQNRQXYUU41o9SG4HrkKWR59FJIv8lmnwBolB/wE0=","0qcd51IQrNKYL9233q2L9h8dLzPcor56mdtkcOdQWoI=","0Slg2/xnc5E9nXprYyph/57wQou\u002BhGSGgKchbo4aNOg=","iwhciMceDYGWEDInLGhYMdHcWG826ConXi020\u002B5rawY=","uyvh3stjGDbFG\u002BpgTWJOhuOKd8owqZkvI8psvOWqsso=","H7pIhmEAeQaK0FIAJMM4lqts08H04IDYcy0aNxHdKHM=","\u002BlXcvLfSHF8FbrWk2UQSf\u002BodPwZSm4MA4RTIFOtI\u002BaY=","/s1pOdMacXOJO2AeBKr2KfMWu1ai23zb2OZjCcapnp8=","k15Z/9EL/0Vd5ORRglbigwCRBc4UN8gwcnYMsJGX7G0="],"CachedAssets":{},"CachedCopyCandidates":{}}
|
||||
{"GlobalPropertiesHash":"77IoXRXzqsXjiL49gpciOThHZJG/7UPKC1BPuiFQdlk=","FingerprintPatternsHash":"gq3WsqcKBUGTSNle7RKKyXRIwh7M8ccEqOqYvIzoM04=","PropertyOverridesHash":"8ZRc1sGeVrPBx4lD717BgRaQekyh78QKV9SKsdt638U=","InputHashes":["7Gcs8uTS1W2TjgmuuoBwaL/zy\u002B2wcKht3msEI7xtxEM=","XWz/ezyQ/zz6q7gqbUREA6BRKDpL7J8X2Ypj\u002B1WdnYY=","A3Op/M2RFQpYBjcrogPFz1XIhJgm4S0j42sTu7EvHxI=","hnhSRoeFpk3C6XWICUlX/lNip6TfbZWFYZv4weSCyrw=","EoVh8vBcGohUnEMEoZuTXrpZ9uBDHT19VmDHc/D\u002Bm0I=","IdEjAFCVk3xZYjiEMESONot/jkvTj/gnwS5nnpGaIMc=","JVRe\u002Be2d47FunIfxVYRpqRFtljZ8gqrK3xMRy6TCd\u002BQ=","DQG0T8n9f5ohwv9akihU55D4/3WR7\u002BlDnvkdsAHHSgc=","VxDQNRQXYUU41o9SG4HrkKWR59FJIv8lmnwBolB/wE0=","0qcd51IQrNKYL9233q2L9h8dLzPcor56mdtkcOdQWoI=","0Slg2/xnc5E9nXprYyph/57wQou\u002BhGSGgKchbo4aNOg=","s3aoiNP1MQgQRB\u002BVPitzH59upgaCzF\u002Bmb31uVs8G/eI=","T4Q5Djrj0xyEnesxWEjQ7iaKDb7sMWyGcOs\u002BFUrfZu8=","iwhciMceDYGWEDInLGhYMdHcWG826ConXi020\u002B5rawY=","uyvh3stjGDbFG\u002BpgTWJOhuOKd8owqZkvI8psvOWqsso=","H7pIhmEAeQaK0FIAJMM4lqts08H04IDYcy0aNxHdKHM=","\u002BlXcvLfSHF8FbrWk2UQSf\u002BodPwZSm4MA4RTIFOtI\u002BaY=","/s1pOdMacXOJO2AeBKr2KfMWu1ai23zb2OZjCcapnp8=","dUP4TEpQnExDOxsXDL4jtIAp3tRLK4pYZTMa4TqPSfw="],"CachedAssets":{},"CachedCopyCandidates":{}}
|
||||
Reference in New Issue
Block a user