프로젝트

일반

사용자정보

통계
| 브랜치(Branch): | 개정판:

markus / ConvertService / ServiceBase / Markus.Service.Station / StationService / ServiceStationTask.cs @ ff4b1e6e

이력 | 보기 | 이력해설 | 다운로드 (14 KB)

1
using Markus.Service.Interface;
2
using Markus.Message;
3
using System;
4
using System.Collections.Generic;
5
using System.Diagnostics;
6
using System.Linq;
7
using System.Text;
8
using System.Threading;
9
using System.Threading.Tasks;
10
using System.Management;
11
using static Markus.Service.Extensions.Encrypt;
12
using Markus.Service.Extensions;
13
using Markus.Service.Helper;
14

    
15
namespace Markus.Service
16
{
17
    /// <summary>
18
    /// 컨버터 큐 처리 
19
    /// </summary>
20
    public partial class ServiceStation
21
    {
22
        /// <summary>
23
        /// 컨버터 실행중인 item
24
        /// </summary>
25
        private static List<ConvertItem> AliveConvertQueue = new List<ConvertItem>();
26

    
27
        /// <summary>
28
        /// 컨버터 프로세스 실행
29
        /// </summary>
30
        /// <param name="convertitem"></param>
31
        public bool ConvertProcessStart(ConvertItem convertitem)
32
        {
33
            bool result = false;
34
            try
35
            {
36
           
37
                Process ConvertProcess = new Process();
38

    
39
                ProcessContext processSendData = new ProcessContext
40
                {
41
                    ConvertID = convertitem.ConvertID,
42
                    ConnectionString = MarkusDBConnectionString,
43
                    ServiceStationUri = gServiceHostAddress.ToString(),
44
                    OriginFilePath = convertitem.OriginfilePath,
45
                    SaveDirectory = convertitem.ConvertPath,
46
                    TempDirectory = DownloadTempFolder,
47
                    ReleaseWorkMemory = ReleaseWorkMemory,
48
                    MultiThreadMaxPages = MultiThreadMaxPages,
49
                    MinFontSize = MinFontSize,
50
                    SendStatusInterval = SaveStatusInterval,
51
                    UseResolution = UseResolution
52
                };
53

    
54
                var sendData = ObjectToBytesStringConvert.ObjectToBytesString(processSendData);
55
                
56
                ProcessStartInfo startInfo = new ProcessStartInfo
57
                {
58
                    UseShellExecute = false,
59
                    FileName = "Markus.Service.ConvertProcess.exe",
60
                    WindowStyle = ProcessWindowStyle.Hidden,
61
                    CreateNoWindow = true,
62
                    ErrorDialog = false,
63
                    RedirectStandardError = false,
64
                    Arguments = $"{convertitem.ConvertID.ToString()} {AESEncrypter.Encrypt(sendData)}"
65
                    //Arguments = $"{convertitem.ConvertID.ToString()} {convertitem.ProjectNumber} {AESEncrypter.Encrypt(MarkusDBConnectionString)} {gServiceHostAddress} {DownloadTempFolder} {MultiThreadMaxPages}"
66
                };
67

    
68
                ConvertProcess.StartInfo = startInfo;
69
                ConvertProcess.EnableRaisingEvents = false;
70

    
71
                System.Diagnostics.Debug.WriteLine("convert process run : " + startInfo.Arguments);
72

    
73

    
74
                if (ConvertProcess.Start())
75
                {
76
                    try
77
                    {
78
                        var processAffinity = ProcessorAffinityList.Except(AliveConvertQueue.Select(f => (long)f.ProcessorAffinity));
79

    
80
                        if (processAffinity.Count() > 0)
81
                        {
82
                            convertitem.ProcessorAffinity = processAffinity.First();
83

    
84
                            //int bitMask = 1 << (convertitem.ProcessorAffinity - 1);
85
                            //bitMask |= 1 << (anotherUserSelection - 1); / //  프로세스 두개 이상 선택
86

    
87
                            ConvertProcess.ProcessorAffinity = new IntPtr(convertitem.ProcessorAffinity);
88

    
89
                        }
90
                        else
91
                        {
92
                            // 모두 사용중일때 점유율이 작은 걸로 사용
93
                            var CurrentProcessAffinity = AliveConvertQueue.Select(f =>f.ProcessorAffinity).Distinct();
94

    
95
                            var affinity = CurrentProcessAffinity.Min();
96

    
97
                            convertitem.ProcessorAffinity = affinity;
98
                            ConvertProcess.ProcessorAffinity = new IntPtr(affinity);
99
                        }
100
                    }
101
                    catch (Exception ex)
102
                    {
103
                        System.Diagnostics.Debug.WriteLine(ex);
104
                    }
105
        
106

    
107
                    ServiceStation.AliveConvertQueue.Add(convertitem);
108
                    result = true;
109
                }
110
            }
111
            catch (Exception ex)
112
            {
113
                throw new Exception("ConvertThread " + $"{convertitem.ConvertID.ToString()} {convertitem.ProjectNumber} {AESEncrypter.Encrypt(MarkusDBConnectionString)} {gServiceHostAddress} {DownloadTempFolder} {MultiThreadMaxPages}", ex.InnerException);
114
            }
115
            finally
116
            {
117
                //GC.WaitForPendingFinalizers();
118
                //GC.Collect(2);
119
                //GC.Collect(2);
120
            }
121

    
122
            return result;
123
        }
124
    
125
        /// <summary>
126
        /// DB에 있는 대기중인 Item을 가져온다.
127
        /// </summary>
128
        public void setDataBaseWaitingList()
129
        {
130
            using (DataBase.ConvertDatabase database = new DataBase.ConvertDatabase(MarkusDBConnectionString))
131
            {
132
                var convertItems = database.GetWaitConvertItems(this.RunProjectList, StationServiceList.Where(x=>x.IsOnline).Sum(f=>f.Properties.PROCESS_COUNT));
133

    
134
                foreach (var convert in convertItems)
135
                {
136
                    //ReflashSubService();
137

    
138
                    if (convert.STATUS > (int)StatusCodeType.None)
139
                    {
140
                        database.SetCleanUpItem(convert.ID);
141
                    }
142

    
143
                    PassConvertItem(convert.PROJECT_NO, convert.ID,convert.DOCUMENT_ID);
144
                }
145
            }
146
        }
147

    
148
        public bool IsDataBaseWaitingList(int overListCount)
149
        {
150
            bool result = false;
151

    
152
            try
153
            {
154
                using (DataBase.ConvertDatabase database = new DataBase.ConvertDatabase(MarkusDBConnectionString))
155
                {
156
                    var convertItems = database.GetWaitConvertItems(this.RunProjectList, overListCount);
157

    
158
                    if (convertItems.Count() > 0)
159
                    {
160
                        result = true;
161
                    }
162
                }
163
            }
164
            catch (Exception ex)
165
            {
166
                logger.Error($"IsDataBaseWaitingList Error",ex);
167
            }
168

    
169
            return result;
170
        }
171

    
172
        private async void ReflashSubService()
173
        {
174
            foreach (var subservice in StationServiceList)
175
            {
176
                try
177
                {
178
                    subservice.IsOnline = SytemNet.Ping(subservice.Properties.SERVICE_ADDRESS);
179

    
180
                    if (subservice.IsOnline)
181
                    {
182
                        var result = await subservice.Service.AliveConvertListAsync();
183

    
184
                        subservice.ConvertItems = result.ToList();
185
                        subservice.AliveCount = result.Count();
186
                    }
187
                    else
188
                    {
189
                        logger.Error($"Connection Error {subservice.Properties.SERVICE_ADDRESS}");
190
                        subservice.ConvertItems = new List<WcfClient.StationServiceTask.ConvertItem>();
191
                        subservice.AliveCount = subservice.Properties.PROCESS_COUNT;
192
                    }
193
                }
194
                catch (Exception ex)
195
                {
196
                    logger.Error($"ReflashSubService error - Service ID : {subservice.Properties.ID} ", ex);
197
                }
198
            }
199
        }
200

    
201

    
202
        /// <summary>
203
        /// 
204
        /// </summary>
205
        /// <param name="ProjectNo"></param>
206
        /// <param name="ConvertID"></param>
207
        /// <param name="UniqueKey">Document ID(문서의 유일키)</param>
208
        private async void PassConvertItem(string ProjectNo,string ConvertID,string UniqueKey)
209
        {
210

    
211
            try
212
            {
213
                var stationList = StationServiceList.Where(x => x.IsOnline);
214

    
215
                if (stationList.SelectMany(x => x.ConvertItems).Count(c => c.ProjectNumber == ProjectNo && c.UniqueKey == UniqueKey) == 0)
216
                {
217
                    var station = stationList.OrderByDescending(x => x.Properties.PROCESS_COUNT - x.AliveCount).FirstOrDefault();
218

    
219
                    if (station != null)
220
                    {
221
                        System.Diagnostics.Debug.WriteLine($"{station.Properties.SERVICE_ADDRESS}  {station.Properties.PROCESS_COUNT}/{station.AliveCount}");
222
                        var result = await station.Service.ConvertAddAsync(ProjectNo, ConvertID);
223
                        
224
                        if(result.ToLower() == true.ToString().ToLower())
225
                        {
226
                            station.AliveCount++;
227
                        }
228

    
229
                        logger.Info($"PassConvertItem - Service ID : {station.Properties.ID} ConvertID : {ConvertID}");
230
                    }
231
                }
232
            }
233
            catch (Exception ex)
234
            {
235
                logger.Error($"setDataBaseWaitingList", ex);
236
            }
237
        }
238

    
239
        /// <summary>
240
        /// 컨버터 프로세스와 AiliveItems을 비교하여 AliveItems에 없는 경우 AliveItems을 제거
241
        /// </summary>
242
        private void CleanUpAliveQueueItems()
243
        {
244
            if (AliveConvertQueue.Count() > 0)
245
            {
246
                var processList = Process.GetProcessesByName("Markus.Service.ConvertProcess");
247

    
248
                if (processList.Length == 0)
249
                {
250
                    AliveConvertQueue.Clear();
251
                    System.Diagnostics.Debug.WriteLine("AliveConvertQueue.Clear()");
252
                }
253
                else
254
                {
255
                    var argumentList = processList.Select(f => f.Arguments().CommandLine).SelectMany(f => f);
256

    
257
                    for (int i = AliveConvertQueue.Count - 1; i >= 0; --i)
258
                    {
259
                        if (argumentList.Count(x => x == AliveConvertQueue[i].ConvertID) == 0)
260
                        {
261
                            AliveConvertQueue.RemoveAt(i);
262
                        }
263
                    }
264
                }
265
            }
266
        }
267

    
268
        /// <summary>
269
        /// 데이터베이스에서 멈춰있거나 
270
        /// </summary>
271
        private async void CleanUpDataBaseItems()
272
        {
273
            using (DataBase.ConvertDatabase database = new DataBase.ConvertDatabase(MarkusDBConnectionString))
274
            {
275
                var items = database.GetConvertingItems(RunProjectList);
276

    
277
                List< WcfClient.StationServiceTask.ConvertItem> aliveItems = new List<WcfClient.StationServiceTask.ConvertItem>();
278

    
279
                foreach (var item in StationServiceList.Where(x=>x.IsOnline))
280
                {
281
                    try
282
                    {
283
                        var serviceItems = await item.Service.AliveConvertListAsync();
284
                        aliveItems.AddRange(serviceItems);
285
                    }
286
                    catch (Exception)
287
                    {
288
                    }
289
                }
290

    
291
                foreach (var item in items)
292
                {
293
                    if(aliveItems.Count(x=>x.ConvertID == item.ID) == 0)
294
                    {
295
                        database.SetCleanUpItem(item.ID);
296
                    }
297
                }
298
            }
299
        }
300

    
301

    
302
        public void Stopprocess()
303
        {
304
            var process = Process.GetProcessesByName("Markus.Service.ConvertProcess");
305

    
306
            for (int i = process.Count() - 1; i >= 0 ; i--)
307
            {
308
                try
309
                {
310
                    Console.WriteLine($"{i} Process Kill");
311
                    process[i].Kill();
312
                }
313
                catch (Exception ex)
314
                {
315
                    System.Diagnostics.Debug.WriteLine(ex.ToString());
316
                }
317
            }
318
        }
319

    
320
        /// <summary>
321
        /// finish가 호출되고 살아있는 프로세스라고 추정됨
322
        /// </summary>
323
        public void DeadLockProcessKill()
324
        {
325
            var process = Process.GetProcessesByName("Markus.Service.ConvertProcess");
326

    
327
            for (int i = process.Count() - 1; i >= 0; i--)
328
            {
329
                try
330
                {
331
                    var commandLines = process[i].Arguments().CommandLine;
332

    
333
                    if (commandLines.Count() > 0)
334
                    {
335
                        if (ServiceStation.AliveConvertQueue.Count(f => f.ConvertID == commandLines[0]) == 0)
336
                        {
337
                            process[i].Kill();
338
                        }
339
                    }
340
                }
341
                catch (Exception ex)
342
                {
343
                    System.Diagnostics.Debug.WriteLine(ex.ToString());
344
                }
345
            }
346
        }
347

    
348
        private void ConvertFinish(ConvertItem convertitem)
349
        {
350
            try
351
            {
352
                System.Diagnostics.Debug.WriteLine("Convert Finish : " + convertitem.ConvertID);
353

    
354
                System.Diagnostics.Debug.WriteLine("ServiceStation.AliveConvertQueue.Count() : " + ServiceStation.AliveConvertQueue.Count());
355
                
356
                ServiceStation.AliveConvertQueue.Remove(convertitem);
357

    
358
                System.Diagnostics.Debug.WriteLine("ServiceStation.AliveConvertQueue.Count() : " + ServiceStation.AliveConvertQueue.Count());
359

    
360
                if (IsStation)
361
                {
362
                    if (!IsReleaseItems)
363
                    {
364
                        System.Diagnostics.Debug.WriteLine("ConvertFinish ReleaseItems call");
365
                        ReleaseItems();
366
                    }
367
                }
368
                else
369
                {
370
                    if (StationClient != null)
371
                    {
372
                        StationClient.ReleaseConvertItems();
373
                    }
374
                }
375
                //if (ServiceStation.AliveConvertQueue.Count() < MultiProcessCount)
376
                //{
377
                //    setDataBaseWaitingList();
378
                //}
379
            }
380
            catch (Exception ex)
381
            {
382
                logger.Error("ConvertFinish Error",ex);
383
            }
384
        }
385
    }
386
}
클립보드 이미지 추가 (최대 크기: 500 MB)