制作NSIS命令行窗口输出插件

在上一篇中,介绍了如何让NSIS生成的安装包在静默安装时从命令行窗口输出安装信息,但是产生了很多问题,于是我们想换一个实现方法,
毕竟静默安装时的输出信息并不需要那么多,多了反而让人眼花。静默安装时,只要输出生成还是失败,失败了有个失败原因就可以了。
所以另一个解决方案就是自己写个插件,向CMD窗口输出一些信息就是了。这篇文章就是要介绍一下如何自己编写这样的一个NSIS插件。
很奇怪的是,NSIS的官方网站上居然没有介绍如何自己编写Plugin的文章,搜索到在OpenWatcom有一篇介绍了NSIS插件要符合的要求。
按照它给出的描述——“NSIS plugins are simply DLLs. The functions provided by these functions are exported functions that have a specific array of parameters.
That signature is as follows.”NSIS插件不过是简单的DLL,只要是符合下面参数列表的函数,就会被认为是合法的NSIS插件。
__declspec(dllexport) void __cdecl TestFunc(HWND hwndParent, int string_size, char *variables, stack_t **stacktop)
好啊。那就照葫芦画瓢写个DLL喽。Hello world级的。下面是hello.cpp的源代码。

#include <windows.h>
 #include <stdio.h>
 #define NSISAPI __declspec(dllexport) void __cdecl
 NSISAPI TestFunc(HWND hwndParent, int string_size, char *variables, stack_t **stacktop)
 {
 printf("Hello");
 }

很简单吧。用下面的命令编译一下。

CL /LD Hello.cpp

编译不过啊!
error C2061: syntax error : identifier ‘stack_t’
stack_t未定义!这让我去哪里找啊?只当是外面的吧。把代码改成下面的样子。

#include <windows.h>
 #include <stdio.h>
 #define NSISAPI __declspec(dllexport) void __cdecl
 struct stack_t;
 NSISAPI TestFunc(HWND hwndParent, int string_size, char *variables, stack_t **stacktop)
 {
 printf("Hello");
 }

这下就过了。把这个DLL放到NSIS的Plugins文件夹中,写一个最简单的NSIS脚本测试一下。

OutFile "Test.exe"
 Section
 Hello::TestFunc
 SectionEnd

结果发现脚本Build不过啊。错误为“Invalid command: Hello::TestFunc”
Command找不到。
看看MakeNSISW上完整的输出信息。列出了NSIS找到的所有Plugin里的可作为Command的函数。如下

Processing config:
Processing plugin dlls: “D:Program FilesNSISPlugins*.dll”
– AdvSplash::show
– Hello::?TestFunc@@YAXPAUHWND__@@HPADPAPAUstack_t@@@Z
的确是有我们刚加入的DLL的,不过函数名不对啊。“?TestFunc@@YAXPAUHWND__@@HPADPAPAUstack_t@@@Z”怎么会有这么个恶心的函数名?
也许有人已经猜到了,由于使用的是C++,而C++是支持函数重载的,但是导出函数表的函数名不能重名啊。所以CL在编译C++的时候,
会自动对函数名进行修饰。这个恶心的名字就是修饰过的函数名。有没有办法不让CL修饰呢?
一个办法就是告诉CL我们的代码是C代码,不支持重载,CL就不会修饰了。代码再改如下。

#include <windows.h>
 #include <stdio.h>
 #define NSISAPI extern "C" __declspec(dllexport) void __cdecl
 struct stack_t;
 NSISAPI TestFunc(HWND hwndParent, int string_size, char *variables, stack_t **stacktop)
 {
 printf("Hello");
 }

再把编译好的DLL复制到插件文件夹,再编译Test.nsi,这次就编译过了。
双击运行Test.exe,不会看到任何Hello,因为printf是输出到Console的。
从命令行运行Test.exe,也不会在控制台看到”Hello”,因为这个Test.exe是个GUI程序,是没有Console的Output的。寒。那怎么才能有呢?
有三个办法,NSIS也是都可以做到了。三种方法可以参考我的另一篇文章让WPF窗体程序支持命令行方式运行的三种方式。方法是通用的。
“| more”的方法就不用解释了。让NSIS生成一个CUI程序,可以参考让NSIS生成的安装包在静默安装时从命令行窗口输出安装信息,但是这种方法弊端多多,
不然也就不会有这篇文章了。最后一种方法就是AttachConsole了,这种方法也有不好的地方,不过比较而言,还是可以的。
这个DLL的代码很简单,只实现了最基本的功能,大家有需要可以自己添加。

#include <windows.h>
 #include <stdio.h>
 #pragma comment (lib, "user32.lib")
 #pragma comment (lib, "kernel32.lib")
 #define NSISAPI extern "C" __declspec(dllexport) void __cdecl
 HANDLE hOutput;
 typedef struct _stack_t {
 struct _stack_t *next;
 char text[1024];
 } stack_t;
 NSISAPI Attach(HWND hwndParent, int string_size, char *variables, stack_t **stacktop)
 {
 if (AttachConsole(-1) == 0)
 printf("failed to attach parent console.");
 else
 hOutput = GetStdHandle(STD_OUTPUT_HANDLE);
 }
 NSISAPI Show(HWND hwndParent, int string_size, char *variables, stack_t **stacktop)
 {
 if (hOutput == NULL)
 printf((*stacktop)->text);
 else
 WriteConsole(hOutput, (*stacktop)->text, strlen((*stacktop)->text), NULL, NULL);
 }
THE END
喜欢就支持一下吧
点赞15 分享
评论 抢沙发

请登录后发表评论

    暂无评论内容