2024年的最后三天 - 甲流来袭

28号 - 抗体 28号晚上,我觉得嗓子有点刺痛。因为前两周发过一次烧,我觉得这次嗓子痛应该不会是发烧的前奏了,毕竟身体是有抗体的。 我高估了之前产生的抗体。 29号 - 梦 从29号上午开始,身体就好像在温水煮青蛙,最高到达惊人的峰值40度。 为了控制体温,我先洗个热水澡,吃了布洛芬,躺在床上,拿了一块冰袋,裹在卫生纸里,搁在脑门上。身体挺直,裹着两层厚被。 长夜慢慢,睡睡醒醒。 我梦见自己变成了一棵树,脚上好像长出了根须,缓慢的往地下生长。穿过泥土,与蚯蚓擦肩而过。穿过岩石,被石油染黑。然而到达的却是一块冰山,冰冷坚硬。 手上长出了树干,像天空伸展,摊开自己的枝叶,还没等我享受片刻温暖,太阳仿佛飞速向我飞来,炙热的阳光让我瞬间枯黄,灰飞烟灭。 长夜慢慢,我不敢辗转反侧。我头上还顶着一块冰。 30号 - 发热门诊 终于熬到早上,再测试一遍体温,38度。 算了,还是去医院吧。 穿戴整齐,刚下楼,迎面吹来一阵冷风。我有种无法压抑的,想要咳嗽的冲动,但是我必须忍住,我知道这咳嗽必然“感天动地"。 风继续吹,我忍不住咳了出来,那感觉,仿佛有人伸手把我的气管从我的嘴巴里扯了出来。 到了医院,发热门诊是单独一层的小房子,和气派的十几层的门诊大楼相比,简直像个保安室。 发热门诊虽小,但也五脏俱全。 进门先做鼻拭子,量体温,挂号。 再做血液分析,然后就排队等叫号。医生看了报告,说我是甲流,开了两盒药,一盒抗病毒,一盒用来退烧。 从医院出来,已经中午。我走进医院旁边的永X大王,准备随便吃点。 我挑了一个座位,扫码下单,点了馄炖、蒸蛋、豆浆、银耳莲子羹。 等餐期间, 我发现对面有个中年人大叔,穿这黑色的宽大的羽绒服,胡子拉碴、头发稀少, 他时不时的巡视着其他的人。 没过过久,我等到自己的餐。 馄炖上撒了淡黄的鸡蛋丝和黑色海苔碎,鸡蛋丝吃起来像绳子,海苔碎味同嚼蜡。混度汤非常浑浊,像是用了一天的澡堂池子水。馄炖我就吃了一个,就放弃了。 蒸蛋应该是预制菜,放在黑色塑料小杯中,应该是微波炉加热的。我吃了一口,味道奇怪。 豆浆没什么好说的,毕竟也味道也没有下降空间了。 银耳莲子羹还不错,我都个喝完了。 吃饱喝足,顺便我把医生给我开的药也吃了,我起身离开,刚走到餐桌不到2米, 就瞥见那个大叔匆匆走向我的餐桌。 急诊室 我吃完饭,回到家,每隔一个小时测一次体温,体温很稳定,稳定在38度左右。 一直到晚上,我的体温还是没怎么下降。 老婆给我打电话,说让我赶紧去急诊,去输液,光吃药效果不好,高烧不退会要人命的。 但是我觉得没必要,因为尽然发热门诊的医生都没有让我输液,说明我不需要输液,或者说输液也没有多大作用。 老婆说:“你想让我中年丧夫吗?” 我无话可说,只能默默穿上衣服,带上口罩,去了早上那家医院的急诊。 晚上9点急诊室人来人往,络绎不绝,仿佛是白天的门诊。 我挂上号,接着又等了将近1个小时,终于等到我了。 给我看病的是位女医生。 ”医生,我在你们医院发生门诊早上就看过了,诊断是甲流,药也吃了,到现在还是38度,还是给我输液吧。“,我说 ”甲流不是一天两天能好的,至少要发热三天,而且输液效果也不大“,医生说 “那我也烧了两天了,再烧下去人要烧没了。你给我开个输液的吧” “好的吧,那我就给你开一次输液,你看看效果” 一顿拉扯,我终于能输液了。其实输的也不是什么特殊的东西,就是一带左氧和一袋维C。 输完液,已经晚上11点50,叫了车,回到家里。感觉嘴巴好苦,还好家里有甜的冰糖心苹果。 我啃完第一个苹果,每一口都是苦味。接着我再啃一个苹果,每一口都还是苦味。 如果不是因为发烧,我简直立即想去吃点火锅底料漱漱口。 归梦 睡觉前,我又量了一次体温。体温恢复到37度,看起来正常正常了。 我不知道这是吃药起的效果还是输液的效果。 但是两盒药的价格是270,其中一盒西药50,另一盒重要220。 输液呢,一袋左氧+维C,总共也不过才30。 让我想起了电影《大腕》的名言 31号 这是24年的最后一天,美丽的烟花在天空中绽放璀璨的光芒。 有些人觉得烟花美丽,有些人只觉得吵。

2024-12-31 21:06:09 · 1 min · Eddie Wang

源码笔记 - 自定义事件路由(中)

[[TOC]] route_list route.h定义了几个函数分别用来获取、查找、新增route // src/core/route.h int route_get(struct route_list *rt, char *name); int route_lookup(struct route_list *rt, char *name); void push(struct action *a, struct action **head); struct route_list { struct action **rlist; int idx; /* first empty entry */ int entries; /* total number of entries */ struct str_hash_table names; /* name to route index mappings */ }; rlist 我们对route_list数据模型进行简化: rlist是一个固定长度的一维数组,通过索引来访问对应的值。如果数组的空间不足,那么就创建一个两倍大的空数据,然后先把原始数据复制过去。这种复制方式保持的原始数据的索引位置。有点像golang的切片扩容机制。 这里最为重要的就是保持数组元素的索引位置在扩容后不变。 static inline int route_new_list(struct route_list *rt) { int ret; struct action **tmp; ret = -1; if(rt->idx >= rt->entries) { // 两倍扩容 tmp = pkg_realloc(rt->rlist, 2 * rt->entries * sizeof(struct action *)); if(tmp == 0) { LM_CRIT("out of memory\n"); goto end; } /* init the newly allocated memory chunk */ memset(&tmp[rt->entries], 0, rt->entries * sizeof(struct action *)); rt->rlist = tmp; rt->entries *= 2; } if(rt->idx < rt->entries) { ret = rt->idx; rt->idx++; } end: return ret; } str_hash_table 我们对hash_table的数据模型进行简化,它其实就是一hash表,key是路由的名,值是一个正数,正数代表了路由执行单元的索引位置。...

2024-12-28 09:43:00 · 2 min · Eddie Wang

源码笔记 - 自定义事件路由(上)

[[TOC]] 事件路由简介 在某些模块中,我们看到有一些模块自定义的事件路由。 例如dispatcher模块,或者rtpengine模块。 event_route[dispatcher:dst-down] { xlog("L_ERR", "Destination down: $rm $ru ($du)\n"); } event_route[rtpengine:dtmf-event] { xlog("L_INFO", "callid: $avp(dtmf_event_callid)\n"); xlog("L_INFO", "source_tag: $avp(dtmf_event_source_tag)\n"); xlog("L_INFO", "timestamp: $avp(dtmf_event_timestamp)\n"); xlog("L_INFO", "dtmf: $avp(dtmf_event)\n"); } disapcher模块 在dispatch.c文件中,我们看到如下代码 if(!ds_skip_dst(old_state) && ds_skip_dst(idx->dlist[i].flags)) { ds_run_route(msg, address, "dispatcher:dst-down", rctx); } else { if(ds_skip_dst(old_state) && !ds_skip_dst(idx->dlist[i].flags)) ds_run_route(msg, address, "dispatcher:dst-up", rctx); } ds_run_route还是定义在dispatch.c文件中, static void ds_run_route(sip_msg_t *msg, str *uri, char *route, ds_rctx_t *rctx) 接着又一个重要调用。 这里似乎在查找路由。 route这个参数其实就是dispatcher:dst-down, 或者 dispatcher:dst-up, 那么event_rt又是什么鬼呢? rt = route_lookup(&event_rt, route); event_rt是一个route_list的结构体...

2024-12-27 20:45:03 · 2 min · Eddie Wang

DMQ模块源码学习笔记

背景 多个SIP注册服务器之间,如何同步分机的注册信息呢? 简单的方案就是使用共享数据库的方式同步注册信息,这个方案实现起来简单,但是分机的注册信息本身就是个需要频繁增删改查的,数据库很可能在大量注册分机的压力下,成为性能的瓶颈。 除了数据库之外,OpenSIPS和kamailio分别提供了不同的方案。 OpenSIPS提供的方案是使用cluster模块,cluster模块在多个实例之间同步分机的注册信息,注册信息的格式是OpenSIPS自定义的格式。 Kamailio的方案是DMQ模块, DMQ听起来高大上,放佛是依赖外部的一个服务。 但它其实就是扩展SIP消息,通过SIP消息来广播分机的注册信息。 KDMQ sip:notification_peer@192.168.40.15:5090 SIP/2.0 Via: SIP/2.0/UDP 192.168.40.15;branch=z9hG4bK55e5.423d95110000 To: <sip:notification_peer@192.168.40.15:5090> From: <sip:notification_peer@192.168.40.15:5060>;tag=2cdb7a33a7f21abb98fd3a44968e3ffd-5b01 CSeq: 10 KDMQ Call-ID: 1fe138e07b5d0a7a-50419@192.168.40.15 Content-Length: 116 User-Agent: kamailio (4.3.0 (x86_64/linneaus)) Max-Forwards: 1 Content-Type: text/plain sip:192.168.40.16:5060;status=active sip:192.168.40.15:5060;status=disabled sip:192.168.40.17:5060;status=active 源码分析 该模块一共暴露了8个参数,其中7个参数都是简单类型,INT和STR,就直接取对应变量的地址就可以了。 其中notification_address参数是用来配置集群中其他节点的通信地址的,因为要配置多次,所以需要用一个函数来解析。 // dmq.c static param_export_t params[] = { {"num_workers", PARAM_INT, &dmq_num_workers}, {"ping_interval", PARAM_INT, &dmq_ping_interval}, {"server_address", PARAM_STR, &dmq_server_address}, {"server_socket", PARAM_STR, &dmq_server_socket}, {"notification_address", PARAM_STR|PARAM_USE_FUNC, dmq_add_notification_address}, {"notification_channel", PARAM_STR, &dmq_notification_channel}, {"multi_notify", PARAM_INT, &dmq_multi_notify}, {"worker_usleep", PARAM_INT, &dmq_worker_usleep}, {0, 0, 0} }; 这些参数都没有加上static关键词,主要目的为了在dmq模块的其他c文件能使用。...

2024-12-22 18:07:08 · 2 min · Eddie Wang

路由执行的顺序

1. 请求消息处理过程 请求可以 直接丢弃,不返回任何响应。对于恶意请求,SIP Flood攻击,最好不要返回任何响应。 直接返回状态码,不做转发,例如直接返回301重定向 无状态转发 有状态转发 执行分支路由,分支路由也可以将消息丢弃 无论有无状态,请求发出去前都会执行onsend_route路由,在onsend_route内部,已经不能对SIP消息再做拦截 2. 响应消息处理过程 首先执行reply_route{}, 在这个路由里可以将消息丢弃 然后判断消息是否有状态的 有状态,这执行onreply_route[ID]路由 如果响应是失败的,还可以执行failure_route[ID], 当前前提是在请求路由里是否设置了钩子 在失败路由可以,可以再次设置新的目标地址,进行转发; 设置了新的目标地址后,还可以设置分支路由 Tip 这里要注意的是,响应路由在失败路由之前执行。 3. 重传处理

2024-12-22 16:20:42 · 1 min · Eddie Wang

kamailio 启动参数控制

-a mode Auto aliases mode: enable with yes or on, disable with no or off 一般都是关闭 --alias=val Add an alias, the value has to be '[proto:]hostname[:port]' (like for 'alias' global parameter) 设置对外别名, 在多个对外别名时,相比于在脚本中写死, 更好的方式 是在启动时传入, alias一般都是服务的对外域名或者IP 如果km有多个对外域名,并且不同的环境都不同,这块配置就合适在脚本里写死 --atexit=val Control atexit callbacks execution from external libraries which may access destroyed shm memory causing crash on shutdown. Can be y[es] or 1 to enable atexit callbacks, n[o] or 0 to disable, default is no....

2024-12-22 15:46:13 · 3 min · Eddie Wang

你不怕暴露自己的无知吗?

公开自己的错误 我在写博客时,有时候脑海里总会蹦出一个小人,面露鄙夷的脸色对我说:你写这么多没啥技术含量的垃圾,公开在网上,难道不怕暴露自己的无知吗? 说实话,我是有这样的担忧。 因为我是有自知之明的,我知道自己估计也是黄老师那种"样样通,样样松"的人。 写的东西也都是一些表面的东西,甚至有错误的可能。这并不是自谦。 我一直无法找到反击脑海里小人的理由。 今天,我在读一本书的时候,学到了一个概念,这个概念叫做坎宁安定律。 在互联网上获得正确答案的最好方法并不是提出问题,而是发布错误的答案 也许我的答案是错误的,但是它并没有被隐藏我脑海的某个角落,二是被公开在了网上。 即使我的小破站再小,必然也会有几个阅读量吧,或许能有读者对错误的答案提出自己的异议。 学习金字塔理论 如果仅仅是通过阅读学习,学习内容的平均存留率只有5%。 如果把学习内容公开,这其中就暗示了你可能需要把自己学到的内容教授给他人这一心理。 那么在记录笔记的时候,就会想办法把问题讲解的让别人更清楚,从而加深了自己的学习知识吸收。 参考 https://baike.baidu.com/item/%E5%AD%A6%E4%B9%A0%E9%87%91%E5%AD%97%E5%A1%94/9515094

2024-12-07 14:17:28 · 1 min · Eddie Wang

TypeSctipt类型实战  [draft]

2024-11-22 13:04:35 · 0 min · Eddie Wang

第二份工作的辞职总结

总结 第二份工作快到一年了,我最近申请离职了。 同事问我离职原因,我说被领导骂了,感觉自尊受辱,索性不干了。 实际上被领导骂只是压死骆驼的最后一根稻草,我也不是那种会心血来潮说干就不干的。 这份工作,我做了不到一年,大部分的时间我是做的不开心的。 所以离职是一直在心里酝酿。 为什么会工作不开心? 主要原因有以下四点: 行业前景暗淡:S公司处在行业的中游,上游受到运营商管控,下游在客户层面话语权也不大。这是一个资源限制型的行业,行业的命脉就在运营商,可能运营商有个风吹草动,整个行业就要翻江倒海。 自研能力薄弱:S公司大部分业务都是购买的第三方的软件,各种变更都受到制约,添加一个小功能都要付费升级。自研能力弱的一个原因可能就是前些年赚钱太容易了,投资研发哪有赚快钱来的舒服呢。同时也是这个原因,多个第三方软件之间也造成了数据割裂,运行维护困难。 家族企业制约:S公司大部分人员都是老板的亲朋好友,家族式企业的优点是内聚力强,但是包容性弱。之前也有一些大厂进来的员工,没多久就辞职了。 我曾经也跟老板沟通过,似乎老板也有心做些变革。但是变革哪有那么容易,要变革不早都变了,还要等到现在? 短期主义:似乎S公司没有什么具体的愿景,唯一感觉到的愿景就是赚快钱。一旦发现投入的回报太少,就立即切换目标。 企业关怀差:说实在,我没有感觉到多少员工关怀。在S公司的工作和在地主家做短工也没啥区别,老板想什么都要管,但是即管不了那么多,又管的不好。所以给人的感觉就是东一棒槌,西一榔头,看起来很卖力,实际上并没有什么产出。几乎每个月都有人离职,对员工都吝啬,不考虑员工的自尊,员工怎么可能对公司有归属感。 工作压力大:受限于资源收紧、产品优势小,老板想提高营业额的方法是提高员工的服务水平。说实在,这真是杯水车薪。不想着从底层系统的解决问题,只想着通过给员工压力、通过加班来解决问题,只会造成工作压力大、人员流失大。 所以,我的离职,实际上也是考虑再三的一个选择。 当然,我在S公司也不是什么收获都没有。 收获总结 深入了解了行业的运行原理、客户、资源、政策等信息 对于数据分析方面有了较大的进步,毕竟我做了40+个数据报表来分析运行数据 深入学习了python, 主要用来做数据分析。数据分析方面的协议栈主要有pandas、duckdb、doris(MySQL)、grafana来组成 深入学习了python的GUI开发,主要用到的框架是wxPython,开发的两个工具分别给到运维人员和客服去使用,来提高运行效率

2024-09-10 21:08:51 · 1 min · Eddie Wang

算法 - 入侵脑细胞

我以前看过一部电影,叫做《入侵脑细胞》,这部电影提供一种可能性。类似的电影还有《盗梦空间》,都是将一种思想植入别人的思想中。 这两部电影虽然提供了可能性,但是却无法提供实际可操作的方案。 最近在阅读的一本书也于此有关,叫做《对不起,我操控了你的大脑》,是一个叫做剑桥分析、曾轰动世界的公司,利用基于大数据、算法、人工智能、心理学来做思想植入的。常见的领域如操纵大选,并且提供成功的案例。 但是作为普通人,大规模的思想植入似乎里我们还是比较遥远。 但是真的那么遥远吗? 昨天我爸来看我,之前他总是抱怨手机流量不够用。我给他换了每个月180G的流量卡,才过没两三天,他说已经用了20多G了。按照这个速度,180G的流量也不够用一个月。 我问:“你怎么会用那么多流量!,你用手机看视频的吗?” 爸说:“我经常刷抖音,刷好久,刷的眼疼” 我说:“你还是少刷点手机吧”,虽然这只是苍白无力的 一句关心。 我爸生于70年代,没怎么读过书,小学文凭。现在年近五十,经常带着鸭舌帽,依然遮不住后脑勺斑白的头发。虽然他经常用手机,但是很多东西依然搞不明白,需要找身边的年轻人来帮忙设置手机。 一个50多岁的人,人生过了大半。面对现在飞速发展的社会,很多东西都已经无法适应。对于他自己来说,也基本上无法改变自己。从第七次人口普查的数据中可以看出,在2020年60+的人口已经占比接近14亿人口的20%了,随着时间的推移,这个占比只会增大。 对于我爸这类的人,即使刷的眼睛疼,也无法控制自己,还是一直不停的刷。 由此可见,像这种大公司,制作的超级App有多可怕。能够一直持续的让大脑保持短期兴奋。这种超级App,背后都是心理学家、计算机学家、数据科学家等。 这些专家对待普通大众,就像哥伦布对待印第安人。 因为我发现他们没有什么信仰,也没有任何崇拜的形象,人们又都很温和,他们无邪得不知道什么是邪恶,从不侵害别人,也不懂得偷窃,没有打斗,特别的温顺 《哥伦布日记》 或许几十年以后,大部分人的大脑已经退化,也不需要有思考能里。只需要机械的按照刺激->反应去行动。 这种进化是一种潮流,绝大多数人都难以幸免。作为一个独立思考的人,我只能尽量避免被影响,远离社交网络。

2023-11-21 20:35:02 · 1 min · Eddie Wang