123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369 |
- #include <sstream>
- #include "card_car.h"
- #include "card_message_handle.h"
- #include "his_location.h"
- #include "area.h"
- #include "mine.h"
- #include "three_rates.h"
- #include "select_tool.h"
- #include "websocket/ws_common.h"
- #include "special_area.h"
- #include "common_tool.h"
- #include "tool_time.h"
- #include "mine_business.h"
- #include "loc_point.h"
- #include "module_service/module_call.h"
- #include "sys_setting.h"
- #include "event.h"
- car::car(const std::string&type,uint32_t cardid,uint16_t needdisplay,int16_t t,int32_t deptid,
- int32_t categoryid, int type_id,int32_t level_id,uint32_t cid)
- :card_location_base(type,cardid,needdisplay,t,deptid,level_id,cid)
- ,m_vehicle_category_id(categoryid)
- ,m_vehicle_type_id(type_id)
- {
- m_message_handle.reset(new card_message_handle(this));
- m_speeds.set_capacity(30);
- //m_his_location_card.reset(new location_vehicle(m_id,m_type,cid));
- }
- car::~car(){}
- std::shared_ptr<mine_tool> car::get_mine_tool()
- {
- return m_mine_tool;
- }
- void car::set_area_info(int mapid,double scale,int areaid,uint64_t t,int type)
- {
- m_area_tool->set_area_info(mapid, scale, areaid, *this, t, type);
- }
- /*
- * @brief
- * 车辆业务
- * @param
- * const std::shared_ptr<site>& site 基站对象
- * const point& pt 车辆定位坐标
- * double acc 加速度值
- * @return
- * 无
- * @note
- * @warning
- * @bug
- * */
- void car::do_business(const std::shared_ptr<site>&site,const point &pt,double acc)
- {
- m_acc = acc;
- m_area_tool->on_point(shared_from_this(),pt);
- m_timeval = m_time;
- handle_traffic_light(pt, site->m_id);
-
- if(CYaSetting::m_sys_setting.m_enable_anti_coll){
- handle_anti_coll(pt, site->m_id);
- }
-
- uint64_t id=tool_other::type_id_to_u64(m_type,m_id);
- mine_business::inst()->make_arg(id,pt,m_time);
- //超速告警业务
- handle_over_speed();
- }
- /*
- * @brief
- * 超速告警,频率为1hz,定位一秒一次;
- * 开始:连续30个定位点超过35km认为开始超速告警
- * 结束:连续10个定位点小于35km认为结束超速告警
- * @param
- * 无
- * @return
- * @note
- * @warning
- * @bug
- * */
- void car::handle_over_speed()
- {
- // 超速告警,频率为1hz,定位一秒一次;
- // 开始:连续30个定位点超过35km认为开始超速告警
- // 结束:连续10个定位点小于35km认为结束超速告警
- m_speeds.push_back(m_speed);
- if(m_speeds.size() < 30){
- return;
- }
- bool status = true;
- for(size_t i = 0;i < m_speeds.size();i++){
- if(m_speed < 35){
- status = false;
- }
- }
- uint64_t id = m_type;
- id = ((id<<32) | m_id);
- if(status){
- // 产生告警
- log_info("[v_over_speed] %d begin warning...", m_id);
- event_tool::instance()->handle_event(OT_CARD, ET_CARD_OVER_SPEED, id, 0, 0, status);
- }
- status = true;
- for(size_t i = 20;i < m_speeds.size();i++){
- if(m_speed>35){
- status = false;
- }
- }
- if(status){
- // 结束告警
- log_info("[v_over_speed] %d end warning...", m_id);
- event_tool::instance()->handle_event(OT_CARD, ET_CARD_OVER_SPEED, id, 0, 0, false);
- }
- }
- int car::get_vehicle_type_id()
- {
- return m_vehicle_type_id;
- }
- /*
- * @brief
- * 车辆防碰撞判断
- * 车辆定位后,检查车辆附近(根据防碰撞的三个档位距离阈值)是否有人,有人则产生防碰撞告警
- * 如果车辆在距离最大档位内没人,则结束告警;
- * 以车为单位,通知web端,进入一档二档三挡会将三个档位每个档位有多少人通知web端,
- * 当人从一档变为二档也会将人档位的变化发送给web端;
- * 发送告警的级别会根据档位的变化动态变化;产生一条告警
- * 允许重复发送
- * 在本方法中实现上述逻辑;
- * @param
- * const point& pt 车辆定位位置
- * const int& sid
- * @return
- * 无
- * @note
- * modified by zhuyf,2022-04-24
- * @warning
- * @bug
- * */
- void car::handle_anti_coll(const point& pt, const int& sid)
- {
- // 车卡下发最紧急的呼叫类型
- std::map<int, float> cd; // 人卡与车卡的距离,key为人卡id,value为距离
- std::map<int, call_card> cards;
- auto tmp_cards = card_list::instance()->m_map;
- log_info("[anti_coll] card'size = %d", tmp_cards.size());
- double min_d = 9999999.9;
- double cur_v = 0.0;
- for(auto k : CYaSetting::m_sys_setting.mp_anti_collision){
- //log_info("[anti_coll] key=%d, value=%.2f", k.first, k.second);
- for(auto& c : tmp_cards)
- {
- // 车卡不参与车车之间防碰撞
- if(c.second->m_type == CT_VEHICLE){
- continue;
- }
- // 司机卡不参与人车之间的防碰撞
- if(c.second->m_id>=9000&&c.second->m_id<=9999){
- continue;
- }
- // 避免历史定位数据的影响
- uint64_t now = time(0)*1000;
- if(m_time - now > 60){
- continue;
- }
- bool s = false;
- int d = (c.second->m_timeval >= m_timeval ? (c.second->m_timeval - m_timeval) : (m_timeval - c.second->m_timeval)) / 1000.0;
- s = ((c.second->m_timeval >= m_timeval ? (c.second->m_timeval - m_timeval) : (m_timeval - c.second->m_timeval)) /1000.0 <= 30);
- //float dist = pt.dist(*c.second);
- float dist = pt.dist(c.second->m_v_point);
- log_info("[anti_coll] vid=%d, pid=%d, dist=%.2f", m_id, c.second->m_id, dist);
- if(dist < k.second && s){
- int call_level = 5 - k.first;
- log_info("[anti_coll] distance=%.3f, level=%d, thre_value=%.3f, time_diff=%d", dist, call_level, k.second, d);
-
- if(min_d > k.second){
- min_d = k.second;
- cur_v = dist;
- }
- auto itc = cd.find(c.second->m_id);
- if(itc == cd.end()){
- cd.insert(std::make_pair(c.second->m_id, dist));
- }else{
- itc->second = dist;
- }
- auto it = cards.find(c.second->m_id);
- if(it != cards.end()){
- if(c.second->m_call_level > call_level){
- c.second->m_call_level = call_level;
- }
- }else{
- cards.insert(std::make_pair(c.second->m_id, call_card(c.second->m_id, c.second->m_type, call_level, CCT_CALL_APOINT, sid)));
- }
- log_info("[anti_coll] card_id=%d, ctype=%d, call_level=%d, call_type_id=%d, site_id=%d",c.second->m_id, c.second->m_type, call_level, CCT_CALL_APOINT, sid);
- }
- }
- }
- std::string desc = "";
- uint64_t id = tool_other::type_id_to_u64(m_type, m_id);
- if(cards.size() == 0){
- // delete event from event_list
- event_tool::instance()->handle_event(OT_CARD, ET_PERSON_VEHICLE_ANTI_COLLISION, id, 0, 0, false);
- }else{
- size_t i = 0;
- for(auto c : cd)
- {
- desc += std::to_string(c.first);
- std::ostringstream buf;
- buf<<c.second;
- desc += "," + buf.str();
- if(i != (cards.size() - 1)){
- desc += ";";
- }
- ++i;
- }
- log_info("[anti_coll] the distance's list between person and vehicle : %s", desc.c_str());
- event_tool::instance()->handle_event(OT_CARD, ET_PERSON_VEHICLE_ANTI_COLLISION, id, min_d, cur_v, true, DT_COMMON, desc);
- }
- if(cards.size() > 0){
- // find the best emengency call
- int call_level = 6;
- for(auto c : cards){
- if(c.second.call_level_id < call_level){
- call_level = c.second.call_level_id;
- }
- }
- cards.insert(std::make_pair(m_id, call_card(m_id, 2, call_level, CCT_CALL_APOINT, sid)));
- module_call::instance()->send_anti_collision(cards);
- }else{
- //log_info("[anti_coll] no card trigger anti collision rules.");
- return;
- }
- //module_call::instance()->send_anti_collision(cards);
- }
- /*
- * @brief 三率业务处理模块
- * @param const point& pt 定位数据
- * @return 无
- * @note
- * @warning
- * @bug
- * */
- void car::handle_three_rates(const point &pt)
- {
- card_pos cp;
- m_biz_stat = get_stat();
- cp.biz_stat = m_biz_stat;
- cp.x = pt.x;
- cp.y = pt.y;
- cp.z = pt.z;
- cp.map_id = m_area_tool->get_mapid();
- cp.vibration = m_acc;
- //put_three_rates(cp);
- }
- /*
- * @brief 红绿灯业务模块
- * @param const point& p 定位数据
- * @param const int& sid 分站id
- * @return 无
- * @note
- * @warning
- * @bug
- * */
- void car::handle_traffic_light(const point& p, const int& sid)
- {
- card_pos cp;
- cp.x = p.x;
- cp.y = p.y;
- cp.z = p.z;
- cp.biz_stat = get_stat();
- cp.map_id = m_area_tool->get_mapid();
- cp.vibration = m_acc;
- cp.reader_id = sid;
- put_traffic_light(cp);
- }
- void car::on_timer()
- {
- if(!empty())
- make_package();
- //1.找到73号卡,如果计数小于100,更新速度为40;计数大于100,更新速度为20
- /* if(m_cache_nums < 40){
- m_cache_nums++;
- m_speed = 40;
- }else{
- m_speed = 20;
- }
- log_info("[v_over_speed] card_id=10004, speed=%.2f", m_speed);
- handle_over_speed();
- */
- //2.调用业务处理
- }
- int car::get_area()
- {
- int status = m_biz_stat;
- int special_id = -1;
- if(status == STATUS_LOST)
- {
- special_id = special_area_list::instance()->get_special_id(m_id,*this,m_vehicle_category_id);
- log_info("enter_special_area:%.2f,%2.f,id:%d,special_area_id:%d",x,y,m_id,special_id);
- }
- return special_id;
- }
- void car::make_package()
- {
- sys::_CARD_POS_ cp;
- loc_point pt = getSmoothPoint();
- cp.area_info = m_area_tool->m_area_info;
- cp.map_id = m_area_tool->get_mapid();
- cp.biz_stat = get_stat();
- //cp.down_time = m_mine_tool->get_down_time();
- cp.work_time = m_mine_tool->get_work_time();
- cp.is_on_duty= m_mine_tool->is_on_duty();
-
- cp.down_time = m_last_point.x;
- cp.z = m_last_point.y;
-
- upt_card_pos(cp, pt);
- uint64_t _now = tool_time::now_to_ms();
- pt.m_time = _now;
- make_his_location(pt.m_time, pt);
- uint64_t t = _now>m_timeval?_now-m_timeval:m_timeval-_now;
- if(t>10*1000)
- {
- m_area_tool->on_point(shared_from_this(),pt);
- m_biz_stat=get_stat();
- }
- }
- void car::get_card(bool f)
- {
- if(f)
- mine_business::inst()->put(shared_from_this());
- }
- loc_point car::getSmoothPoint()
- {
- loc_point lp = m_smo_tool->smooth_strategy();
- m_speed = lp.m_speed;
- m_stat = lp.m_stat;
- lp.y = -lp.y;
- return lp;
- }
|