因爲項目須要,最近研究了一下跨進程通信改寫第三方程序中的方法(運行中),把本身程序中的目標方法直接覆蓋第三方程序中的方法函數;一直沒有頭緒,經過搜索引擎找了一大堆解決方案,資料甚是稀少,最後功夫不負有心人,通過兩天的研究,終於在github 上找到兩個開源的代碼,經過兩個開源代碼結合起來便可實現個人需求。下面進一步來分析實踐原理,後面會把源代碼地址貼上來;
經過該文章分享,你會知道怎樣經過注入一個dll模塊改寫第三方運行的程序中的某個方法,在裏面實現本身的業務,這個場景在作外掛程序中特別實用!!!git
假若有一個第三方應用程序,這時候須要對第三方應用程序進行方法攔截,好比第三方應用程序中的某個操做須要用咱們的業務覆蓋掉他們的業務,那這種狀況下咱們有什麼好的方案解決呢?咱們不可能修改第三方程序的代碼,那有什麼方案能夠解決呢?其實咱們仍是有辦法進行」修改「第三方程序的代碼的,怎麼」修改「呢,請看下面實踐原理,下面帶你走入不同的代碼世界!!!!github
原理簡化圖:
json
這裏實踐我就直接寫兩個客戶端程序來進行代碼上的演示框架
DotNetDetour
,可是開源做者是從.net framework 4.5開始支持,不支持.net framework 4.0, 個人需求須要運行在老爺機xp 上,故必需要支持4.0 的框架,全部我fork了一份把源代碼作了修改支持到了.net framework 4.0 框架,fork 源代碼地址:https://github.com/a312586670/DotNetDetour這裏爲了演示,我本身建立了一個目標客戶端程序,主要有以下核心代碼方法:函數
public class ProcessService { public string GetProcessInfo() { return "這是TargetClient 客戶端(第三方程序)"; } public ProcessResponse GetProcessInfo(ProcessRequest request) { return new ProcessResponse() { Name = "這是TargetClient 客戶端(第三方程序)", Version = request.Version }; } }
UI界面交互代碼以下:ui
/// <summary> /// MainWindow.xaml 的交互邏輯 /// </summary> public partial class MainWindow : Window { public MainWindow() { InitializeComponent(); } private void btnInfo_Click(object sender, RoutedEventArgs e) { var service = new ProcessService(); this.txtInfo.Text = service.GetProcessInfo(); } private void btnComplateInfo_Click(object sender, RoutedEventArgs e) { var service = new ProcessService(); var response = service.GetProcessInfo(new ProcessRequest() { Version = "v-Demo 1.0 版本" }); this.txtInfo.Text = response.Name + response.Version; } }
上面代碼中有兩個按鈕事件,分別調用了ProcessService 的兩個方法,咱們先來運行目標客戶端Demo程序,分別點擊兩個按鈕運行結果以下:
this
好了,上面咱們的目標第三方Demo程序已經寫好了,接下來咱們須要寫一個核心的Jlion.Process.HookCore
類庫 改寫目標的ProcessService
的兩個方法。
我這裏建了一個Jlion.Process.HookCore
類庫,經過nuget包引用我fork 後的DotNetDetour
類庫,以下圖:
應用成功後咱們創建核心的hook 方法,代碼以下:搜索引擎
public class ProcessHookService : IMethodHook { [HookMethod("Jlion.Process.Target.Client.ProcessService", null, null)] public string GetProcessInfo() { TextHelper.LogInfo($"這是Jlion.Process.HookCore.HookService dll. 改寫TargetClient 客戶端 的GetProcessInfo 方法後獲得的結果"); return "這是Jlion.Process.HookCore.HookService dll. 改寫TargetClient 客戶端 的GetProcessInfo 方法後獲得的結果"; } [OriginalMethod] public string GetProcessInfo_Original() { return null; } [HookMethod("Jlion.Process.Target.Client.ProcessService", null, null)] public object GetProcessInfo([RememberType("Jlion.Process.Target.Client.Model.ProcessRequest", false)] object request) { var json = JsonConvert.SerializeObject(request); TextHelper.LogInfo($"json:{json}"); var name = "這是Jlion.Process.HookCore.HookService dll. 改寫TargetClient 客戶端的GetProcessInfo(obj)後獲得的結果"; return new ProcessResponse() { Name = name, Version = "改寫的dll 版本" }; } [OriginalMethod] public object GetProcessInfo_Original([RememberType("Jlion.Process.Target.Client.Model.ProcessRequest", false)] object request) { return null; } }
我這裏就不詳細的寫DotNetDetour
的使用,須要知道它的使用能夠訪問 https://github.com/a312586670/DotNetDetour 查看具體的文檔.net
核心的Jlion.Process.HookCore
hook 類庫 也已經建立完了,接下來還須要建立一個初始化Hook
的服務類(特別重要),而且還必須是靜態
方法,代碼以下:3d
public class HookService { /// <summary> /// Hook 初始化 /// </summary> /// <param name="msg"></param> /// <returns></returns> public static int Start(string msg) { try { TextHelper.LogInfo("開始"+msg); MethodHook.Install(); } catch { return -1; } return 1; } }
到這一步基本上Jlion.Process.HookCore
Hook 核心的類庫已經建立完了
建立客戶端後須要引用FastWin32
類庫,以下圖:
客戶端注入Dll核心代碼以下:
public class InjectService { //注入的核心dll 路徑 public static string path = AppDomain.CurrentDomain.BaseDirectory+ "Jlion.Process.HookCore.dll"; /// <summary> /// 進程id /// </summary> public static uint pid = 0; /// <summary> /// 啓動 /// </summary> public static void Start() { Inject(); } #region 私有方法 private static void Inject() { try { Injector.InjectManaged(pid, path, "Jlion.Process.HookCore.HookService", "Start", "ss", out int returnValue); } catch (Exception ex) { } } #endregion }
代碼中核心的代碼是Injector.InjectManaged()
,該方法有以下兩個重構方法:
/// <summary> /// MainWindow.xaml 的交互邏輯 /// </summary> public partial class MainWindow : Window { public MainWindow() { InitializeComponent(); } private void btnInject_Click(object sender, RoutedEventArgs e) { InjectService.pid = Convert.ToUInt32(txbPid.Text.Trim()); InjectService.Start(); } }
這裏核心的注入Client Demo 也寫完了,咱們把注入的客戶端也運行起來,輸入目標的進程pid(也能夠程序中查找目標進程Id),運行後再來執行上面建立的第三方程序的兩個按鈕,結果以下:
經過編寫客戶端程序點擊注入dll後,再點擊第三方程序的兩個按鈕事件,結果以下:
能夠看到點擊後,運行的結果已經被動態注入的Jlion.Process.HookCore.dll
改寫了,不過上面的代碼也能夠改寫後同時還運行原有目標的方法就是經過調用'_Original'後綴結尾的方法,方法體返回null便可。
經過DotNetDetour
框架能夠編寫對目標進程的方法進行Hook 重寫,使用新的方法覆蓋第三方進程的方法,也能夠繼續執行第三方的方法。
經過FastWin32
調用Win32 API 把開發的dll模塊注入到第三方進程中,同時注入後執行初始化方法,能夠進行原有的Hook方法進行覆蓋。
到這裏是否是感受很神奇,它能夠在如下場景中使用:
感興趣的朋友能夠下載Demo 源代碼玩一玩:
github 源代碼地址:https://github.com/a312586670/processClientDemo