-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathActorClientService.cs
172 lines (147 loc) · 5.12 KB
/
ActorClientService.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
using Orleans;
using Orleans.Hosting;
using Orleans.Configuration;
using System;
using System.Threading.Tasks;
using System.Threading;
using Microsoft.Extensions.Configuration;
using System.Collections.Generic;
using System.Reflection;
using System.IO;
using System.Diagnostics;
namespace Frontend
{
public class ActorClientService : IHostedService
{
private static List<Assembly> _externalAssemblies = new List<Assembly>();
private readonly ILogger<ActorClientService> _log;
private readonly IConfiguration _config;
public IClusterClient Client { get; }
public ActorClientService(IConfiguration config, ILogger<ActorClientService> log)
{
if (config == null)
{
throw new ArgumentNullException(nameof(config));
}
if (log == null)
{
throw new ArgumentNullException(nameof(log));
}
_config = config;
_log = log;
IClientBuilder builder = new ClientBuilder();
ConfigureOrleansBase(builder);
var env = _config["ORLEANS_CONFIG"];
if (env == "SQL")
{
ConfigureOrleansSQL(builder);
}
else if (env == "STORAGE")
{
ConfigureOrleansStorage(builder);
}
else
{
throw new Exception("ORLEANS_CONFIG envvar not defined.");
}
Client = builder.Build();
}
private void ConfigureOrleansStorage(IClientBuilder builder)
{
builder
.UseAzureStorageClustering(options =>
{
options.TableName = "clusterstate";
options.ConnectionString = Environment.GetEnvironmentVariable("AzureWebJobsStorage");
});
}
private void ConfigureOrleansSQL(IClientBuilder builder)
{
builder
.UseAdoNetClustering(options =>
{
options.Invariant = "System.Data.SqlClient";
options.ConnectionString = Environment.GetEnvironmentVariable("SqlConnectionString");
});
}
private void ConfigureOrleansBase(IClientBuilder builder)
{
builder
.Configure<ClusterOptions>(options =>
{
options.ClusterId = "serverlessorleans";
options.ServiceId = "serverlessorleans";
})
.ConfigureLogging(builder => builder.AddConsole())
.ConfigureApplicationParts(parts =>
{
foreach(var assembly in ExternalAssemblies)
{
System.Console.WriteLine("Loading orleans app parts: " + assembly.FullName);
parts.AddApplicationPart(assembly);
}
})
.UseDashboard();
}
public Task StartAsync(CancellationToken cancellationToken)
{
var attempt = 0;
var maxAttempts = 10;
var delay = TimeSpan.FromSeconds(6);
return Client.Connect(async error =>
{
if (cancellationToken.IsCancellationRequested)
{
return false;
}
if (++attempt < maxAttempts)
{
_log.LogWarning(
error,
$"Failed to connect to Orleans cluster on attempt {attempt} of {maxAttempts}.");
try
{
await Task.Delay(delay, cancellationToken);
}
catch (OperationCanceledException)
{
return false;
}
return true;
}
else
{
_log.LogError(error,
$"Failed to connect to Orleans cluster on attempt {attempt} of {maxAttempts}.");
return false;
}
});
}
public async Task StopAsync(CancellationToken cancellationToken)
{
await Client.Close();
Client.Dispose();
}
public static IEnumerable<Assembly> ExternalAssemblies
{
get
{
lock(_externalAssemblies)
{
if (_externalAssemblies.Count == 0)
{
var appPath = AppDomain.CurrentDomain.BaseDirectory + "/app";
Debug.Assert(!string.IsNullOrWhiteSpace(appPath));
foreach(var assemblyPath in Directory.GetFiles(appPath, "*.dll"))
{
_externalAssemblies.Add(Assembly.LoadFrom(assemblyPath));
}
}
return _externalAssemblies;
}
}
}
}
}