Projekt

Obecné

Profil

Stáhnout (10.2 KB) Statistiky
| Větev: | Tag: | Revize:
1 cdf3c217 A-Konig
//
2
// Author: A. Konig
3
//
4
5
using System;
6 5d9a5bd9 A-Konig
using System.Collections.Generic;
7
using System.IO;
8 734533a8 A-Konig
using ServerApp.Parser.InputData;
9
using ServerApp.Parser.OutputInfo;
10 5d9a5bd9 A-Konig
11 734533a8 A-Konig
namespace ServerApp.Parser.Parsers
12 5d9a5bd9 A-Konig
{
13 734533a8 A-Konig
    /// <summary>
14
    /// Class parsing jis files into instances of ActivityInfo divided by given time interval
15
    /// Data parsed from 7am (included) to 18pm (included)
16
    /// </summary>
17 cdf3c217 A-Konig
    /// <author>A. Konig</author>
18 5d9a5bd9 A-Konig
    class JisParser
19
    {
20 98b568bc A-Konig
21
        /// <summary> Datafile loader  </summary>
22 cdf3c217 A-Konig
        IDataLoader loader;
23 98b568bc A-Konig
24
        /// <summary>
25
        /// Constructor
26
        /// </summary>
27
        /// <param name="loader">Datafile loader</param>
28 cdf3c217 A-Konig
        public JisParser(IDataLoader loader)
29 98b568bc A-Konig
        {
30
            this.loader = loader;
31
        }
32
33 734533a8 A-Konig
        /// <summary>
34
        /// Parses jis data to ActivityInfo instances
35
        /// Data parsed from 7am (included) to 18pm (included)
36
        /// </summary>
37 cdf3c217 A-Konig
        /// <param name="jisFiles">Paths to files with login data files</param>
38
        /// <param name="endTime">End time of related data</param>
39
        /// <param name="startTime">Start time of related data</param>
40 734533a8 A-Konig
        /// <param name="wholeDay">Should data be parsed as one instance per day (if true parameter interval will be ignored)</param>
41
        /// <param name="interval">Time interval to divide days by, minimum is 1h</param>
42
        /// <returns></returns>
43 cdf3c217 A-Konig
        public List<ActivityInfo> ParseJisData(List<string> jisFiles, DateTime startTime, DateTime endTime, bool wholeDay = true, int interval = 1)
44 5d9a5bd9 A-Konig
        {
45 734533a8 A-Konig
            List<ActivityInfo> list = new List<ActivityInfo>();
46
47 cdf3c217 A-Konig
            if (jisFiles == null || startTime == null || endTime == null || interval <= 0)
48
                return list;
49
50 734533a8 A-Konig
            // parse all files
51 cdf3c217 A-Konig
            foreach (string fileName in jisFiles)
52 5d9a5bd9 A-Konig
            {
53 cdf3c217 A-Konig
                if (!File.Exists(fileName))
54
                    continue;
55
                
56 734533a8 A-Konig
                // parse as one instance per day
57 cdf3c217 A-Konig
                List<ActivityInfo> loadedData = null;
58 5d9a5bd9 A-Konig
                if (wholeDay)
59 cdf3c217 A-Konig
                    loadedData = ProcessOneJisFileAsDays(fileName, startTime, endTime);
60 734533a8 A-Konig
                // parse by interval length
61 5d9a5bd9 A-Konig
                else
62 cdf3c217 A-Konig
                    loadedData = ProcessOneJisFileAsIntervals(fileName, interval, startTime, endTime);
63 5d9a5bd9 A-Konig
64
                list.AddRange(loadedData);
65
            }
66
67
            return list;
68
        }
69
70 734533a8 A-Konig
        /// <summary>
71
        /// Parses data from one data file as one instance per day
72
        /// </summary>
73 cdf3c217 A-Konig
        /// <param name="path">Path to file</param>
74
        /// <param name="endTime">End time of related data</param>
75
        /// <param name="startTime">Start time of related data</param>
76 734533a8 A-Konig
        /// <returns>List with ActivityInfo instances</returns>
77 cdf3c217 A-Konig
        private List<ActivityInfo> ProcessOneJisFileAsDays(string path, DateTime startTime, DateTime endTime)
78 5d9a5bd9 A-Konig
        {
79 cdf3c217 A-Konig
            List<ActivityInfo> jisInfo = new List<ActivityInfo>();
80
            
81 734533a8 A-Konig
            if (!File.Exists(path))
82 cdf3c217 A-Konig
                return jisInfo;
83 5d9a5bd9 A-Konig
84 98b568bc A-Konig
            List<JisInstance> list =  loader.LoadJisFile(path);
85 5d9a5bd9 A-Konig
86 734533a8 A-Konig
            // data for each faculty
87 d39750f3 A-Konig
            int[] recordedAmount = new int[TagInfo.buildings.Length];
88 734533a8 A-Konig
            // min/max hour taken into account
89 0da0ac88 A-Konig
            int[] minmaxHour = new int[] { 7, 18 };
90 734533a8 A-Konig
            // interval length
91 0da0ac88 A-Konig
            int range = minmaxHour[1] - minmaxHour[0];
92 5d9a5bd9 A-Konig
93 734533a8 A-Konig
            // first day
94
            DateTime lastStartTime = new DateTime(list[0].dateTime.Year, list[0].dateTime.Month, list[0].dateTime.Day, minmaxHour[0], 0, 0);
95 5d9a5bd9 A-Konig
            for (int i = 0; i < list.Count; i++)
96
            {
97 0da0ac88 A-Konig
                int currHour = list[i].dateTime.Hour;
98
                if (currHour < minmaxHour[0] || currHour > minmaxHour[1])
99
                    continue;
100
101 734533a8 A-Konig
                // start of the day -> make an instance
102 0da0ac88 A-Konig
                string place = list[i].placeTag;
103 734533a8 A-Konig
                DateTime date = new DateTime(list[i].dateTime.Year, list[i].dateTime.Month, list[i].dateTime.Day, minmaxHour[0], 0, 0);
104 0da0ac88 A-Konig
                if (!date.Equals(lastStartTime)) 
105 5d9a5bd9 A-Konig
                {
106 734533a8 A-Konig
                    // data for each faculty separate
107 d39750f3 A-Konig
                    for (int k = 0; k < TagInfo.buildings.Length; k++)
108 5d9a5bd9 A-Konig
                    {
109 d39750f3 A-Konig
                        ActivityInfo dayInfo = new ActivityInfo(TagInfo.buildings[k], recordedAmount[k], lastStartTime, range);
110 5d9a5bd9 A-Konig
                        jisInfo.Add(dayInfo);
111
                    }
112
113 d39750f3 A-Konig
                    recordedAmount = new int[TagInfo.buildings.Length];
114 0da0ac88 A-Konig
                    lastStartTime = date;
115 5d9a5bd9 A-Konig
                }
116
117 cdf3c217 A-Konig
                // if not in allowed time window -> discard
118
                if (list[i].dateTime < startTime || list[i].dateTime > endTime)
119
                    continue;
120
121 734533a8 A-Konig
                // aggregate data
122 0da0ac88 A-Konig
                if (TagInfo.jisPlaces.ContainsKey(place))
123
                {
124
                    int index = TagInfo.jisPlaces[place];
125
                    if (index == -1)
126 d39750f3 A-Konig
                        for (int l = 0; l < TagInfo.buildings.Length; l++)
127 0da0ac88 A-Konig
                            recordedAmount[l] += list[i].amount;
128
                    else
129
                        recordedAmount[index] += list[i].amount;
130
131
                }
132 5d9a5bd9 A-Konig
                else
133
                {
134 d39750f3 A-Konig
                    // TODO uknown code handling -> to file?
135
                    // Console.WriteLine("Unknown code " + list[i].placeTag);
136 5d9a5bd9 A-Konig
                }
137
138
            }
139
140 734533a8 A-Konig
            // last day
141 d39750f3 A-Konig
            for (int k = 0; k < TagInfo.buildings.Length; k++)
142 0da0ac88 A-Konig
            {
143 d39750f3 A-Konig
                ActivityInfo dayInfo = new ActivityInfo(TagInfo.buildings[k], recordedAmount[k], lastStartTime, range);
144 0da0ac88 A-Konig
                jisInfo.Add(dayInfo);
145
            }
146
147
            return jisInfo;
148
        }
149
150 98b568bc A-Konig
        /// <summary>
151
        /// Parses data from one data file as instance per interval length in a day
152
        /// </summary>
153
        /// <param name="path">Path ti file</param>
154
        /// <param name="interval">Interval length</param>
155 cdf3c217 A-Konig
        /// <param name="endTime">End time of related data</param>
156
        /// <param name="startTime">Start time of related data</param>
157 98b568bc A-Konig
        /// <returns>List with ActivityInfo instances</returns>
158 cdf3c217 A-Konig
        private List<ActivityInfo> ProcessOneJisFileAsIntervals(string path, int interval, DateTime startTime, DateTime endTime)
159 0da0ac88 A-Konig
        {
160 cdf3c217 A-Konig
            List<ActivityInfo> jisInfo = new List<ActivityInfo>();
161
            
162 260b1217 A-Konig
            if (!File.Exists(path))
163 cdf3c217 A-Konig
                return jisInfo;
164 260b1217 A-Konig
165 98b568bc A-Konig
            List<JisInstance> list = loader.LoadJisFile(path);
166 260b1217 A-Konig
167
            // data for each faculty
168 d39750f3 A-Konig
            int[] recordedAmount = new int[TagInfo.buildings.Length];
169 260b1217 A-Konig
            // min/max hour taken into account
170
            int[] minmaxHour = new int[] { 7, 18 };
171
            int range = minmaxHour[1] - minmaxHour[0];
172
173
            if (interval > range)
174
                return null;
175
176
            int indices = (int) Math.Ceiling(range / (double)interval);
177 ecdce631 A-Konig
            int[] to = new int[indices];
178
            for (int i = 0; i < to.Length; i++) 
179
                to[i] = minmaxHour[0] + interval * (i+1);
180 260b1217 A-Konig
181
            // first day
182
            DateTime lastStartTime = new DateTime(list[0].dateTime.Year, list[0].dateTime.Month, list[0].dateTime.Day, minmaxHour[0], 0, 0);
183 ecdce631 A-Konig
            int endtime = to[0];
184 cdf3c217 A-Konig
            int startingTime = minmaxHour[0];
185 1db865ce A-Konig
            int index = 0;
186 260b1217 A-Konig
            for (int i = 0; i < list.Count; i++)
187
            {
188
                int currHour = list[i].dateTime.Hour;
189
                if (currHour < minmaxHour[0] || currHour > minmaxHour[1])
190
                    continue;
191
192
                // start of the day -> make an instance
193
                string place = list[i].placeTag;
194
                bool trigger = false;
195
                DateTime date = new DateTime(list[i].dateTime.Year, list[i].dateTime.Month, list[i].dateTime.Day, list[i].dateTime.Hour, 0, 0);
196 1db865ce A-Konig
197 260b1217 A-Konig
                // end of the day
198
                if (!(date.Year == lastStartTime.Year && date.Month == lastStartTime.Month && date.Day == lastStartTime.Day))
199 1db865ce A-Konig
                {
200 260b1217 A-Konig
                    trigger = true;
201 1db865ce A-Konig
                    index = 0;
202
                }
203 260b1217 A-Konig
204
                // end of interval period
205 ecdce631 A-Konig
                if ( date.Hour >= endtime || trigger) 
206 260b1217 A-Konig
                {
207 1db865ce A-Konig
                    // start time of last interval
208 cdf3c217 A-Konig
                    DateTime stTime = new DateTime(lastStartTime.Year, lastStartTime.Month, lastStartTime.Day, startingTime, 0, 0);
209 ecdce631 A-Konig
210
                    // find end and start time
211 1db865ce A-Konig
                    if (!trigger)
212 ecdce631 A-Konig
                        index++;
213
                    while (date.Hour >= to[index])
214
                        index++;
215 1db865ce A-Konig
216 ecdce631 A-Konig
                    endtime = to[index];
217
                    if (index == 0)
218 cdf3c217 A-Konig
                        startingTime = minmaxHour[0];
219 ecdce631 A-Konig
                    else
220 cdf3c217 A-Konig
                        startingTime = to[index - 1];
221 1db865ce A-Konig
222 260b1217 A-Konig
                    // data for each faculty separate
223 d39750f3 A-Konig
                    for (int k = 0; k < TagInfo.buildings.Length; k++)
224 260b1217 A-Konig
                    {
225 d39750f3 A-Konig
                        ActivityInfo dayInfo = new ActivityInfo(TagInfo.buildings[k], recordedAmount[k], stTime, interval);
226 260b1217 A-Konig
                        jisInfo.Add(dayInfo);
227
                    }
228
229 d39750f3 A-Konig
                    recordedAmount = new int[TagInfo.buildings.Length];
230 260b1217 A-Konig
                    lastStartTime = date;
231
                }
232
233 cdf3c217 A-Konig
                // if not in allowed time window -> discard
234
                if (list[i].dateTime < startTime || list[i].dateTime > endTime)
235
                    continue;
236
237 260b1217 A-Konig
                // aggregate data
238
                if (TagInfo.jisPlaces.ContainsKey(place))
239
                {
240 1db865ce A-Konig
                    int faculty = TagInfo.jisPlaces[place];
241
                    if (faculty == -1)
242 d39750f3 A-Konig
                        for (int l = 0; l < TagInfo.buildings.Length; l++)
243 260b1217 A-Konig
                            recordedAmount[l] += list[i].amount;
244
                    else
245 1db865ce A-Konig
                        recordedAmount[faculty] += list[i].amount;
246 260b1217 A-Konig
247
                }
248
                else
249
                {
250 d39750f3 A-Konig
                    // TODO uknown code handling -> to file?
251
                    // Console.WriteLine("Unknown code " + list[i].placeTag);
252 260b1217 A-Konig
                }
253
254
            }
255
256
            // last day
257 d39750f3 A-Konig
            for (int k = 0; k < TagInfo.buildings.Length; k++)
258 260b1217 A-Konig
            {
259 d39750f3 A-Konig
                ActivityInfo dayInfo = new ActivityInfo(TagInfo.buildings[k], recordedAmount[k], lastStartTime, interval);
260 260b1217 A-Konig
                jisInfo.Add(dayInfo);
261
            }
262
263
            return jisInfo;
264 5d9a5bd9 A-Konig
        }
265
    }
266
}