kamailio关于via那点事

如果kamailio作为代理服务器,在转到目的路由时 不删除原始的via信息 会造成信息泄露

如果 Kamailio 作为代理服务器(SIP Proxy)在转发 SIP 请求时不删除原始的 Via 信息,这确实可能会造成信息泄露


📌 为什么不删除 Via 可能会泄露信息?

  1. 暴露内部网络信息

    • Via 头包含了 前一个发送者的 IP 地址、端口和传输方式(UDP/TCP/TLS)。
    • 如果 Via 头未被清理,可能会泄露 内部服务器 IP(如私有地址 192.168.x.x)或运营商信息
    • 例如,假设 Kamailio 运行在 10.10.10.1,但 SIP 服务器在 192.168.1.100,那么 Via 头可能会泄露:
      Via: SIP/2.0/UDP 192.168.1.100:5060;branch=z9hG4bK12345
      
      这个信息 不应该暴露给外网
  2. 攻击者可利用 Via 进行网络拓扑分析

    • 恶意用户可能利用 Via 头中的 IP 地址 扫描 SIP 代理的拓扑结构,寻找潜在的攻击点(如未受保护的 SIP 服务器)。
  3. SIP 设备可能错误处理额外的 Via

    • 某些 SIP 设备(特别是运营商的 SBC 或 PBX)可能不期望看到 额外的 Via,从而导致 SIP 处理异常(如 400 Bad Request483 Too Many Hops)。

✅ 解决方案:让 Kamailio 在转发请求时删除原始 Via

方法 1:使用 remove_hf("Via")

Kamailio 提供了 remove_hf("Via") 指令,可以在 SIP 代理模式下 移除所有 Via 头并重新生成,防止信息泄露。
kamailio.cfg 配置文件中:

request_route {if (!is_method("ACK")) {remove_hf("Via");  # 删除所有 Via 头}append_hf("Via: SIP/2.0/UDP $si:$sp;branch=z9hG4bK$var(branch)\r\n");t_relay();
}

🔹 解释

  • 移除 Via,防止泄露内部 IP。
  • 手动添加新的 Via,只显示 Kamailio 作为代理的 IP($si:$sp)。

方法 2:使用 topos 模块(推荐)

topos 是 Kamailio 提供的 “拓扑隐藏”(Topology Hiding)模块,可以自动删除 Via 头,并在响应时恢复,防止信息泄露。

📌 安装 topos 模块
sudo apt install kamailio-topos-modules
📌 在 kamailio.cfg 启用 topos
loadmodule "topos.so"request_route {if (!is_method("ACK")) {topos_request();}t_relay();
}onreply_route {topos_reply();
}

🔹 解释

  • topos_request() 删除 ViaRecord-Route,隐藏内部拓扑
  • topos_reply() 在响应时恢复正确的路由,确保 SIP 消息能正确返回。

📍 结论

不删除 Via 确实会导致信息泄露,特别是在 NAT 或运营商环境中!
推荐使用 remove_hf("Via")topos 模块来移除 Via,保护 SIP 代理的内部拓扑信息! 🚀

使用topos模块自动更正via

使用 topos 模块后,目的地路由的 Via 头信息

当 Kamailio 作为 SIP 代理(Proxy) 并启用了 topos 模块时,它会自动删除原始的 Via,并在转发 SIP 请求时仅留下 Kamailio 自己的 Via。那么,在**目的地路由(最终 SIP 服务器)**收到的请求中,Via 头的情况如下:


📌 1. 启用了 topos,最终目的地收到的 Via

示例:

📍 客户端 (EP1) 发送初始请求

INVITE sip:13545769997@222.223.190.118 SIP/2.0
Via: SIP/2.0/UDP 192.168.1.100:5060;branch=z9hG4bK111111
Via: SIP/2.0/UDP 10.10.10.1:5060;branch=z9hG4bK222222
Via: SIP/2.0/UDP 222.211.75.75:5060;branch=z9hG4bK333333

🔹 这里 Via 头中包含了所有中间代理的 IP(内部网络信息)


📍 Kamailio 代理(启用了 topos

topos_request() 处理后,Kamailio 会:

  • 删除原始 Via192.168.1.10010.10.10.1222.211.75.75)。
  • 只留下 Kamailio 自己的 Via

🔹 Kamailio 转发后的 SIP 请求

INVITE sip:13545769997@222.223.190.118 SIP/2.0
Via: SIP/2.0/UDP 47.100.62.225:5060;branch=z9hG4bKabcdef

📌 🔹 这里 Via 只保留了 Kamailio 自己的 IP(47.100.62.225),不会暴露任何内部 IP!


📌 2. 目的地收到的 Via

最终,目的地路由(222.223.190.118) 只会看到 Kamailio 的 Via

INVITE sip:13545769997@222.223.190.118 SIP/2.0
Via: SIP/2.0/UDP 47.100.62.225:5060;branch=z9hG4bKabcdef

原始 Via(192.168.1.100、10.10.10.1、222.211.75.75)已被隐藏,保护了 SIP 代理的内部网络结构!


📌 3. 响应中的 Via 处理

当 目的地(222.223.190.118) 发送 200 OK 响应时,它的 Via 头如下:

SIP/2.0 200 OK
Via: SIP/2.0/UDP 47.100.62.225:5060;branch=z9hG4bKabcdef

然后 Kamailio 会使用 topos_reply() 恢复正确的返回路径但不会暴露目的地的 Via,所以客户端最终收到的响应仍然不会看到 222.223.190.118 的 Via


📍 结论

目的地路由(最终 SIP 服务器)收到的 Via 头只包含 Kamailio 的 IP,不包含原始的 Via 头,确保了内部网络信息不被泄露!
响应时 Kamailio 也会正确恢复路径,确保 SIP 消息能够正确返回! 🚀

下面是一个完整的 Kamailio 作为 SIP 代理服务器(Proxy)kamailio.cfg 配置,结合 Lua 脚本 进行 呼叫计费,并使用 topos 模块隐藏 Via 信息,确保 SIP 请求的安全性。


📌 功能概述

  1. 基本 SIP 代理功能
  2. 启用 topos 模块,隐藏 Via
  3. 使用 Lua 进行通话计费
  4. 转发 INVITE、ACK、BYE 等必要的 SIP 请求
  5. 防止环路、错误处理
  6. 超时释放通话

📌 Lua 计费脚本 (/usr/local/kamailio/lua/billing.lua)

-- billing.lua - 计费逻辑(计算通话费用)
local billing = {}function billing.calculate_fee(duration)local rate_per_second = 0.05  -- 每秒 0.05 元local cost = duration * rate_per_secondreturn cost
endfunction billing.log_call(call_id, caller, callee, duration)local cost = billing.calculate_fee(duration)local log_str = string.format("CALL BILLING - Call-ID: %s, From: %s, To: %s, Duration: %d sec, Cost: %.2f 元",call_id, caller, callee, duration, cost)-- 记录日志KSR.info(log_str)
endreturn billing

📌 Kamailio kamailio.cfg 配置

####### 全局配置 #########!KAMAILIO
#!define WITH_LUA
#!define WITH_TOPOloadmodule "sl.so"                # SIP 事务管理
loadmodule "tm.so"                # SIP 事务管理
loadmodule "rr.so"                # 记录路由
loadmodule "maxfwd.so"            # Max-Forwards 处理
loadmodule "siputils.so"          # 处理 SIP 头
loadmodule "topos.so"             # 隐藏 Via 头
loadmodule "topos_redis.so"       # Redis 版本的 topos(推荐)
loadmodule "lua.so"               # Lua 脚本支持modparam("topos", "storage", "redis://127.0.0.1:6379/0")  # 存储 SIP 会话信息到 Redis
modparam("lua", "load", "/usr/local/kamailio/lua/billing.lua")####### SIP 请求路由 ########request_route {if (!mf_process_maxfwd(10)) {sl_send_reply("483", "Too Many Hops");exit;}if (is_method("OPTIONS")) {sl_send_reply("200", "OK");exit;}# 处理 INVITEif (is_method("INVITE")) {record_route();topos_request();  # 删除原始 Via 信息route(1);exit;}# 处理 BYE(通话结束)if (is_method("BYE")) {handle_bye();topos_request();  # 确保 Via 头处理route(1);exit;}route(1);
}route[1] {if (!t_relay()) {sl_reply_error();}exit;
}####### 响应处理 ########onreply_route {topos_reply();  # 确保 Via 头处理
}####### 处理 BYE 进行计费 ########event_route[tm:local-request] {if (is_method("BYE")) {handle_bye();}
}####### 计费逻辑(Lua 计费) ########route[handle_bye] {xlog("L_INFO", "Handling BYE request for call billing");# 获取通话时长if ($Ts - $T_req_time > 0) {$var(duration) = $Ts - $T_req_time;} else {$var(duration) = 0;}# 获取通话 ID、主叫和被叫$var(call_id) = $ci;$var(caller) = $fU;$var(callee) = $tU;# 调用 Lua 计费$var(lua_result) = lua_runstring(string.format("require 'billing'.log_call('%s', '%s', '%s', %d)",$var(call_id), $var(caller), $var(callee), $var(duration)));xlog("L_INFO", "Billing logged: " + $var(lua_result));
}

📌 代码解析

  1. topos 模块

    • topos_request(); 删除 Via,防止信息泄露
    • topos_reply(); 确保返回路径正确
    • 使用 Redis 存储会话信息,防止 Kamailio 重启导致 Via 丢失
  2. Lua 计费

    • calculate_fee(duration) 计算通话费用
    • log_call(call_id, caller, callee, duration) 记录计费日志
  3. 计费触发时机

    • onreply_route {} 处理 SIP 响应
    • event_route[tm:local-request] {} 监听 BYE 事件并触发 Lua 计费
  4. SIP 代理功能

    • 转发 INVITE、ACK、BYE
    • 防止环路(Max-Forwards 机制)
    • 记录路由(Record-Route 机制)

📌 运行流程

📞 呼叫过程:

  1. EP1 -> Kamailio 发送 INVITE
  2. Kamailio 使用 topos_request() 删除原始 Via
  3. Kamailio -> 目的地 转发 INVITE
  4. 目的地 -> Kamailio 发送 200 OK
  5. Kamailio 使用 topos_reply() 恢复路径 并转发给 EP1
  6. EP1 发送 BYE
  7. Kamailio 调用 Lua 计算通话时长和费用
  8. Kamailio 记录计费日志,并转发 BYE目的地

📌 示例日志

CALL BILLING - Call-ID: c12345abcd, From: 01089672389, To: 13545769997, Duration: 120 sec, Cost: 6.00 元

📌 结论

隐藏 Via 头,保护代理服务器信息
使用 Lua 实现通话计费
支持 topos,防止 Via 泄露
支持基本 SIP 代理功能(INVITE、BYE、ACK) 🚀

已经使用了lua脚本,为什么还要把路由相关代码写入到kamailio.cfg中呢?

这是因为 Kamailio 的路由逻辑必须在 kamailio.cfg 中定义,Lua 主要是用于 业务逻辑扩展(如计费),而不是完全替代 kamailio.cfg 的 SIP 处理能力。下面详细解释 为什么 kamailio.cfg 仍然需要路由逻辑


📌 为什么 kamailio.cfg 仍然需要路由逻辑?

Kamailio 主要用于 SIP 代理、注册和路由控制,其核心逻辑(如 INVITE 处理、路由转发、事务管理等)都是由 kamailio.cfg 配置完成的。Lua 只是一个辅助工具,不能完全代替 Kamailio 处理 SIP 消息。

🚀 Kamailio kamailio.cfg 主要职责

  1. 解析 SIP 请求

    • 识别 SIP 方法(如 INVITEBYE
    • 处理 Max-Forwards 以防止循环
    • 记录路由 (Record-Route) 以保持通话链路
  2. 管理 SIP 事务

    • tm(事务管理)模块用于 SIP 事务控制
    • sl(状态回复)模块用于无事务性消息回复
  3. 控制 SIP 代理逻辑

    • 进行 t_relay()(SIP 请求转发)
    • 处理 onreply_route{}(SIP 响应处理)
    • 通过 topos 隐藏 Via

📌 Lua 主要职责

  1. 提供业务逻辑(如计费、日志记录)
  2. 与外部系统交互(如数据库、Redis、REST API)
  3. 增强 Kamailio,但不能完全取代 kamailio.cfg

Lua 不能直接控制 Kamailio 的 SIP 代理行为,它只是一个扩展工具。例如,Lua 不能执行 t_relay() 这样的 SIP 代理操作,必须在 kamailio.cfg 中完成。


📌 为什么不能只用 Lua 处理 SIP?

  • Lua 不能直接处理 SIP 事务(如 t_relay(),必须在 kamailio.cfg 内调用。
  • Lua 不能替代 kamailio.cfg 的核心 SIP 处理逻辑(如 record_route())。
  • Kamailio 是一个高性能 SIP 代理,C 语言模块比 Lua 解析更快,直接在 kamailio.cfg 处理 SIP 逻辑可以提升性能。

📌 最佳实践:结合 kamailio.cfg 和 Lua

  • kamailio.cfg 负责 SIP 代理、转发、事务管理
  • Lua 负责计费、日志、数据库交互

例如:

  • Kamailio 处理 BYE 请求并调用 Lua 计费
  • Lua 计算通话时长,并将费用存入数据库
  • Kamailio 继续处理 SIP 路由

📌 结论

不能完全用 Lua 替代 kamailio.cfg,因为 Lua 不能直接处理 SIP 代理逻辑。
最佳方式是:kamailio.cfg 处理 SIP 代理,Lua 负责计费、日志等业务逻辑。 🚀

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/16369.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MapReduce到底是个啥?

在聊 MapReduce 之前不妨先看个例子:假设某短视频平台日活用户大约在7000万左右,若平均每一个用户产生3条行为日志:点赞、转发、收藏;这样就是两亿条行为日志,再假设每条日志大小为100个字节,那么一天就会产…

LeetCode --- 435周赛

题目列表 3442. 奇偶频次间的最大差值 I 3443. K 次修改后的最大曼哈顿距离 3444. 使数组包含目标值倍数的最少增量 3445. 奇偶频次间的最大差值 II 一、奇偶频次间的最大差值I 统计字母出现次数,然后分别统计出现偶数次的最小值和出现奇数次的最大值,…

chrome-mojo C++ Bindings API

概述 Mojo C 绑定 API 利用C 系统 API提供一组更自然的原语,用于通过 Mojo 消息管道进行通信。结合从Mojom IDL 和绑定生成器生成的代码,用户可以轻松地跨任意进程内和进程间边界连接接口客户端和实现。 本文档通过示例代码片段提供了绑定 API 用法的详…

目标检测数据集合集(持续更新中)

第1期 高压输电线塔鸟巢数据集 第2期 特种工程车辆检测数据集 第3期 金桔目标检测数据集 第4期 金属锈蚀识别检测数据集 第5期 苦瓜目标检测数据集 第6期 石榴目标检测数据集YOLO格式 第7期 光伏电池板缺陷检测数据集YOLO格式 第8期 铁路轨道异物入侵检测数据集YOLO格式…

活动预告 | 为 AI 新纪元做好准备:助力安全的业务转型

课程介绍 随着现代办公模式的不断演变和 AI 技术的迅速发展,企业在享受效率提升的同时,也面临着信息安全与数据保护的严峻挑战。在利用 AI 技术释放业务潜力的同时,如何确保数据质量与安全已成为企业发展的关键议题。 在本次线上课程中&…

语义分割文献阅读——SETR:使用Transformer从序列到序列的角度重新思考语义分割

目录 摘要 Abstract 1 引言 2 Vision Transformer(ViT) 2.1 图片预处理:分块和降维 2.2 Patch Embedding 2.3 位置编码 2.4 Transformer Encoder的前向过程 3 SETR 3.1 图像序列化处理 3.2 Transformer 3.3 解码器 总结 摘要 本周阅读的论文题目是《R…

深度学习入门--python入门1

以前学的python全部还给老师了,所以现在重新开始学习了。目标是每天至少学习一点点吧。 目录 1.1 python是什么 1.2 python安装 1.3 python解释器 1.3.1 算术计算 1.3.2 数据类型 1.3.3 变量 1.3.4 列表(数组) 1.3.5 字典 1.3.6 布…

【2024最新Java面试宝典】—— SpringBoot面试题(44道含答案)_java spingboot 面试题

37. 如何重新加载 Spring Boot 上的更改,而无需重新启动服务器?Spring Boot项目如何热部署?38. SpringBoot微服务中如何实现 session 共享 ?39. 您使用了哪些 starter maven 依赖项?40. Spring Boot 中的 starter 到底是什么 ?4…

【动态规划】风扫枯杨,满地堆黄叶 - 9. 完全背包问题

本篇博客给大家带来的是完全背包问题之动态规划解法技巧. 🐎文章专栏: 动态规划 🚀若有问题 评论区见 ❤ 欢迎大家点赞 评论 收藏 分享 如果你不知道分享给谁,那就分享给薯条. 你们的支持是我不断创作的动力 . 王子,公主请阅🚀 要开心要快乐顺…

python-leetcode-单词搜索

79. 单词搜索 - 力扣(LeetCode) class Solution:def exist(self, board: List[List[str]], word: str) -> bool:if not board or not board[0]:return Falserows, cols len(board), len(board[0])def backtrack(r, c, index):if index len(word):re…

游戏引擎学习第98天

仓库:https://gitee.com/mrxiao_com/2d_game_2 开始进行一点回顾 今天的目标是继续实现正常贴图的操作,尽管目前我们还没有足够的光照信息来使其完全有用。昨日完成了正常贴图相关的基础工作,接下来将集中精力实现正常贴图的基本操作,并准备…

PH热榜 | 2025-02-10

1. 2pr 标语:人工智能帮你把想法变成LinkedIn爆款 或者更口语化一点: AI帮你把点子变成LinkedIn上的热门帖子 介绍:用AI主持的访谈,把你的想法变成LinkedIn爆款帖子。录制你的想法,让AI帮你创作个性化、引人入胜的…

django配置跨域

1、第一种 from django.views.decorators.csrf import csrf_exemptcsrf_exempt第二种 安装 pip install django-cors-headers在配置文件settings.py进入 INSTALLED_APPS [..."corsheaders", # 添加 ]MIDDLEWARE [corsheaders.middleware.CorsMiddleware, # 添加…

使用C语言实现MySQL数据库的增删改查操作指南

使用C语言与MySQL数据库进行交互,通常涉及使用MySQL提供的C API库。这套API允许开发者在C/C++程序中执行SQL查询,从而实现数据库的增删改查操作。下面,我将详细介绍如何在C语言中实现这些基本操作。 准备工作 安装MySQL开发库:确保你的系统上安装了MySQL服务器以及MySQL开发…

25考研电子信息复试面试常见核心问题真题汇总,电子信息考研复试没有项目怎么办?电子信息考研复试到底该如何准备?

你是不是在为电子信息考研复试焦虑?害怕被老师问到刁钻问题、担心专业面答不上来?别慌!作为复试面试92分逆袭上岸的学姐,今天手把手教你拆解电子信息类复试通关密码!看完这篇,让你面试现场直接开大&#xf…

vite + axios 代理不起作用 404 无效

vite axios 代理不起作用 先看官方示例 export default defineConfig({server: {proxy: {// 字符串简写写法/foo: http://localhost:4567,// 选项写法/api: {target: http://jsonplaceholder.typicode.com,changeOrigin: true,rewrite: (path) > path.replace(/^\/api/, )…

【设计模式】【行为型模式】模板方法模式(Template Method)

👋hi,我不是一名外包公司的员工,也不会偷吃茶水间的零食,我的梦想是能写高端CRUD 🔥 2025本人正在沉淀中… 博客更新速度 📫 欢迎V: flzjcsg2,我们共同讨论Java深渊的奥秘 &#x1f…

基础设施在平台工程中的作用

平台工程侧重于设计和构建自助服务工具和环境,以简化软件开发和部署。通过简化和隐藏底层系统的复杂性,我们可以将精力集中在提供有意义的价值上。 从传统的 IT 运营过渡到集成的 DevOps 基础设施实践优先考虑团队合作、简化的流程和持续交付&#xff0…

Unity3D实现显示模型线框(shader)

系列文章目录 unity工具 文章目录 系列文章目录👉前言👉一、效果展示👉二、第一种方式👉二、第二种方式👉壁纸分享👉总结👉前言 在 Unity 中显示物体线框主要基于图形渲染管线和特定的渲染模式。 要显示物体的线框,通常有两种常见的方法:一种是利用内置的渲染…

活动预告 |【Part1】Microsoft Azure 在线技术公开课:AI 基础知识

课程介绍 参加“Azure 在线技术公开课:AI 基础知识”活动,了解 AI 核心概念。参加我们举办的本次免费培训活动,了解组织如何使用 AI 技术克服实际挑战,以及如何借助 Azure AI 服务构建智能应用程序。本次培训适用于任何对 AI 解决…