zhangzengfei
2025-02-07 17f45fcc0a062a15372883de8909953071c51b3c
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
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
package models
 
import (
    "errors"
    "fmt"
    "strconv"
    "strings"
    "time"
 
    "model-engine/db"
    "model-engine/pkg/logger"
    "model-engine/service"
)
 
type DisappearModel struct {
    AreaIds       map[string]struct{}
    Building      string       // 楼栋
    Floor         string       // 楼层
    AlarmType     db.AlarmType // 预警方式
    KeyPersonType string       // 人员类型
    PersonLabel   string
    DisappearTime int    // 消失时间, 单位小时
    AlarmInterval int    // 报警时间间隔, 单位天
    LastDirection string // 最后一次抓拍
    MaxAge        int    // 年龄
    MinAge        int    // 年龄
    StartTime     int64  // 起始时间
    Task          *db.ModelTask
}
 
func (m *DisappearModel) Init(task *db.ModelTask) error {
    m.AreaIds = make(map[string]struct{})
    for _, a := range task.DomainUnitIds {
        m.AreaIds[a] = struct{}{}
    }
 
    m.Task = task
    m.Building = task.Building
    m.Floor = task.Floor
    m.AlarmType = task.AlarmType
    m.KeyPersonType = task.PersonType
    m.PersonLabel = task.PersonLabel
    m.StartTime = task.BeginTime.Unix()
 
    for _, v := range task.Rules {
        if v.Alias == "disappearTime" {
            if val, ok := v.Value.(float64); ok {
                m.DisappearTime = int(val)
            }
        }
 
        if v.Alias == "alarmInterval" {
            if val, ok := v.Value.(float64); ok {
                m.AlarmInterval = int(val)
            }
        }
 
        if v.Alias == "age" {
            if val, ok := v.Value.(string); ok {
                ages := strings.Split(val, ",")
                m.MinAge, _ = strconv.Atoi(ages[0])
                m.MaxAge, _ = strconv.Atoi(ages[1])
            }
        }
 
        if v.Alias == "lastDirection" {
            if val, ok := v.Value.(string); ok {
                m.LastDirection = val
            }
        }
    }
 
    logger.Debugf("DisappearModel init finish ...task id:%s, name:%s, rule:%+v", task.ID, task.Name, m)
 
    if m.DisappearTime == 0 {
        logger.Warnf("invalid parameters. task id:%s, name:%s", task.ID, task.Name)
        return errors.New("invalid parameters")
    }
 
    return nil
}
 
type PersonInfo struct {
    DocumentNumber     string `json:"document_number"`
    CommunityId        string `json:"community_id"`
    OrgId              string `json:"org_id"`
    PersonName         string `json:"person_name"`
    IdCard             string `json:"id_card"`
    LastAppearanceTime int64  `json:"last_appearance_time"`
    LastDirection      string `json:"last_direction"`
    LastLocation       string `json:"last_location"`
}
 
func (m *DisappearModel) Run() error {
    results := make([]*db.ModelTaskResults, 0)
    var ageFilter, labelFilter, keyFilter, lastFilter []PersonInfo
 
    if m.MinAge > 0 {
        err := db.GetDB().Raw(`
        SELECT
            s.document_number,
            s.community_id,
            s.org_id,
            p.person_name,
            p.id_card,
            s.last_appearance_time,
            s.last_direction,
            s.last_location 
        FROM
            snapshot_count_summary AS s
            JOIN person AS p ON p.id = s.document_number 
        WHERE
            s.last_appearance_time > ?
            AND    p.id_card != "" 
            AND TIMESTAMPDIFF(
                YEAR,
                STR_TO_DATE( CASE WHEN LENGTH( id_card ) = 18 THEN SUBSTRING( id_card, 7, 8 ) ELSE NULL END, '%Y%m%d' ),
                CURDATE( ) 
            ) >= ? 
            AND TIMESTAMPDIFF(
                YEAR,
                STR_TO_DATE( CASE WHEN LENGTH( id_card ) = 18 THEN SUBSTRING( id_card, 7, 8 ) ELSE NULL END, '%Y%m%d' ),
            CURDATE( ) 
            ) <= ?
        `, m.StartTime, m.MinAge, m.MaxAge).Scan(&ageFilter).Error
        if err != nil {
            logger.Warnf(err.Error())
        }
 
        if len(ageFilter) == 0 {
            return fmt.Errorf("no results found that match the age condition %s - %s", m.MinAge, m.MaxAge)
        }
 
        logger.Debugf("task %s match age result %d", m.Task.Name, len(ageFilter))
    }
 
    if m.PersonLabel != "" {
        labels := strings.Split(m.PersonLabel, ",")
        err := db.GetDB().Raw(`
        SELECT
            s.document_number,
            s.community_id,
            s.org_id,
            p.person_name,
            p.id_card,
            s.last_appearance_time,
            s.last_direction,
            s.last_location 
        FROM
            snapshot_count_summary AS s
            JOIN person AS p ON p.id = s.document_number
            JOIN person_label AS l ON p.id = l.person_id 
        WHERE
            s.last_appearance_time > ?
            AND    l.label_id IN ?
        `, m.StartTime, labels).Scan(&labelFilter).Error
        if err != nil {
            logger.Warnf(err.Error())
        }
 
        if len(labelFilter) == 0 {
            return fmt.Errorf("no results found that match the label condition %s", m.PersonLabel)
        }
 
        logger.Debugf("task %s match label result %d", m.Task.Name, len(labelFilter))
    }
 
    // 合并一下条件
    if m.MinAge > 0 && m.PersonLabel != "" {
        lastFilter = intersectPersonInfo(ageFilter, labelFilter)
    } else if m.MinAge > 0 {
        lastFilter = ageFilter
    } else if m.PersonLabel != "" {
        lastFilter = labelFilter
    }
 
    if m.KeyPersonType != "" {
        keyTypes := strings.Split(m.KeyPersonType, ",")
        err := db.GetDB().Raw(`
        SELECT
            s.document_number,
            s.community_id,
            s.org_id,
            p.person_name,
            p.id_card,
            s.last_appearance_time,
            s.last_direction,
            s.last_location 
        FROM
            snapshot_count_summary AS s
            JOIN person AS p ON p.id = s.document_number
            JOIN key_person AS k ON k.id_card = p.id_card 
        WHERE
            s.last_appearance_time > ?
            AND s.key_status = 1 
            AND p.id_card != "" 
            AND k.person_type IN ?
        `, m.StartTime, keyTypes).Scan(&keyFilter).Error
        if err != nil {
            logger.Warnf(err.Error())
        }
 
        logger.Debugf("task %s match key person result %d", m.Task.Name, len(keyFilter))
 
        if len(lastFilter) > 0 {
            lastFilter = intersectPersonInfo(lastFilter, keyFilter)
        } else {
            lastFilter = keyFilter
        }
    }
 
    logger.Debugf("task %s last result %d", m.Task.Name, len(lastFilter))
 
    var personIds = make(map[string]struct{}, 0)
    for _, p := range lastFilter {
        if len(m.AreaIds) > 0 {
            _, o1 := m.AreaIds[p.CommunityId]
            _, o2 := m.AreaIds[p.OrgId]
            if !o1 && !o2 {
                continue
            }
        }
 
        if m.LastDirection != "" {
            if p.LastDirection != m.LastDirection {
                continue
            }
        }
 
        if isOlderThanGivenHours(p.LastAppearanceTime, m.DisappearTime) {
            result := &db.ModelTaskResults{
                Title:         m.Task.Name,
                Event:         m.eventFormat(p.LastAppearanceTime, p.LastDirection),
                ModelID:       m.Task.ModelID,
                ModelTaskID:   m.Task.ID,
                CommunityId:   p.CommunityId,
                OrgID:         p.OrgId,
                ObjectIds:     p.DocumentNumber,
                Location:      p.LastLocation,
                PicDate:       time.Unix(p.LastAppearanceTime, 0).Format("2006-01-02 15:04:05"),
                FirstPersonID: p.DocumentNumber,
            }
 
            // 同一个人报警一次
            if _, ok := personIds[p.DocumentNumber]; ok {
                continue
            } else {
                personIds[p.DocumentNumber] = struct{}{}
            }
 
            results = append(results, result)
        }
    }
 
    logger.Debugf("task %s last filter result %d", m.Task.Name, len(results))
    return service.SaveTaskResults(results)
}
 
func (m *DisappearModel) KeepAlive() error {
    db.GetDB().Model(m.Task).Where("id = ?", m.Task.ID).Update("last_run_time", time.Now())
    return nil
}
 
func (m *DisappearModel) Shutdown() error {
    // 清理资源
    fmt.Println("Shutting down Disappear Model")
    return nil
}
 
func intersectPersonInfo(arr1, arr2 []PersonInfo) []PersonInfo {
    // 使用 map 来存储 arr1 中的 DocumentNumber
    resultMap := make(map[string]PersonInfo)
    var result []PersonInfo
 
    // 将第一个数组的 DocumentNumber 添加到 map 中
    for _, person := range arr1 {
        resultMap[person.DocumentNumber] = person
    }
 
    // 遍历第二个数组,检查 DocumentNumber 是否在 map 中
    for _, person := range arr2 {
        if _, exists := resultMap[person.DocumentNumber]; exists {
            // 如果存在交集,将该元素加入结果数组
            result = append(result, person)
            // 可选:如果希望每个交集元素只出现一次,可以删除 map 中的元素
            delete(resultMap, person.DocumentNumber)
        }
    }
 
    return result
}
 
func isOlderThanGivenHours(timestamp int64, hours int) bool {
    // 将时间戳转换为时间对象
    timestampTime := time.Unix(timestamp, 0)
 
    // 获取当前时间
    currentTime := time.Now()
 
    // 计算当前时间减去指定小时数的时间
    timeThreshold := currentTime.Add(-time.Duration(hours) * time.Hour)
 
    // 判断时间戳是否早于该时间
    return timestampTime.Before(timeThreshold)
}
 
func (m *DisappearModel) eventFormat(lastAppearTime int64, lastDirection string) string {
    lastTime := time.Unix(lastAppearTime, 0)
    currentTime := time.Now()
 
    duration := currentTime.Sub(lastTime)
 
    // 输出时间差的小时数
    var durationStr = "小时"
    hours := duration.Hours()
    if hours > 24 {
        hours = hours / 24
        durationStr = "天"
    }
 
    var direction string
    var reverse = "未出现"
    if lastDirection == "in" {
        direction = "进"
        reverse = "未出"
    } else if lastDirection == "out" {
        direction = "出"
        reverse = "未归"
    }
 
    return fmt.Sprintf("%s%s,持续%.1f%s%s", lastTime.Format("2006-01-02 15:04:05"), direction, hours, durationStr, reverse)
}