有没有一种使用JavaScript创建lnk文件的方法?

11
我希望在我的网站中给用户提供下载“lnk”文件的能力。 我的想法是生成此文件,其中包含一个只能使用一次的地址。 是否有办法在Javascript中生成此文件? 流程如下:
1. 用户按下按钮 2. Javascript 生成该文件并将其下载到用户计算机 3. 用户将此文件发送给另一个用户以从他的计算机使用该一次性地址
从客户端的角度来看,是否可以像这样在Javascript中完成?还是我需要使用Java服务器端来生成此文件?

1
@Cory 当然可以 - 看看Blob,你可以在JS中创建任意文件并将其下载到计算机。 - Benjamin Gruenbaum
5
@user1322801 这是你需要做的 - 不会很有趣。首先,阅读.lnk文件格式的工作原理(微软发布了规范)。其次,使用Blob在JavaScript中编写二进制数据。最后,将该Blob转换为所有浏览器的下载文件,例如查看此页面或https://dev59.com/wHA65IYBdhLWcg3wuhIR。 - Benjamin Gruenbaum
1
@TomSarduy 我不这么认为,因为它没有具体说明如何完成这些事情。它只是给OP(以及可能的未来答案)指导,告诉他们如何尝试解决问题 - 答案应该是问题的实际解决方案(不是生产就绪,但至少是概念验证级别)。我认为一个概念验证大约需要80行代码(也许少一点),如果你想拾起手套并实现它,那么它会为写它的人带来很好的声誉。 - Benjamin Gruenbaum
1
@BenjaminGruenbaum:OP确实问了“是否可行?有没有一种在JS中生成文件的方法?”,而不是“如何创建lnk文件?”。你的评论确实回答了这个问题 - 我认为要求一个实际的解决方案可能会超出范围或太广泛。 - Bergi
你想要针对.URL链接(如远程/互联网链接)还是.lnk链接(如本地链接)进行定位? - user13500
显示剩余4条评论
3个回答

8

这是mslink.sh的忠实翻译。

我只在Windows 8.1上测试了我的答案,但我认为它也适用于早期版本的Windows。

function create_lnk_blob(lnk_target) {
    function hex_to_arr(s) {
        var result = Array(s.length / 2);
        for (var i = 0; i < result.length; ++i) {
            result[i] = +('0x' + s.substr(2*i, 2));
        }
        return result;
    }

    function str_to_arr(s) {
        var result = Array(s.length);
        for (var i = 0; i < s.length; ++i) {
            var c = s.charCodeAt(i);
            if (c >= 128) {
                throw Error("Only ASCII paths are suppored :-(");
            }
            result[i] = c;
        }
        return result;
    }

    function convert_CLSID_to_DATA(s) {
        var idx = [[6,2], [4,2], [2,2], [0,2],
                   [11,2], [9,2], [16,2], [14,2],
                   [19,4], [24,12]];
        var s = idx.map(function (ii) {
            return s.substr(ii[0], ii[1]);
        });
        return hex_to_arr(s.join(''));
    }

    function gen_IDLIST(s) {
        var item_size = (0x10000 + s.length + 2).toString(16).substr(1);
        return hex_to_arr(item_size.replace(/(..)(..)/, '$2$1')).concat(s);
    }

    var HeaderSize = [0x4c, 0x00,0x00,0x00],
        LinkCLSID = convert_CLSID_to_DATA("00021401-0000-0000-c000-000000000046"),
        LinkFlags = [0x01,0x01,0x00,0x00], // HasLinkTargetIDList ForceNoLinkInfo

        FileAttributes_Directory = [0x10,0x00,0x00,0x00],
        FileAttributes_File = [0x20,0x00,0x00,0x00],

        CreationTime = [0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],
        AccessTime = [0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],
        WriteTime = [0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],

        FileSize = [0x00,0x00,0x00,0x00],
        IconIndex = [0x00,0x00,0x00,0x00],
        ShowCommand = [0x01,0x00,0x00,0x00], //SW_SHOWNORMAL
        Hotkey = [0x00,0x00], // No Hotkey
        Reserved = [0x00,0x00],
        Reserved2 = [0x00,0x00,0x00,0x00],
        Reserved3 = [0x00,0x00,0x00,0x00],
        TerminalID = [0x00,0x00],

        CLSID_Computer = convert_CLSID_to_DATA("20d04fe0-3aea-1069-a2d8-08002b30309d"),
        CLSID_Network = convert_CLSID_to_DATA("208d2c60-3aea-1069-a2d7-08002b30309d"),

        PREFIX_LOCAL_ROOT = [0x2f],
        PREFIX_FOLDER = [0x31,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],
        PREFIX_FILE = [0x32,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],
        PREFIX_NETWORK_ROOT = [0xc3,0x01,0x81],
        PREFIX_NETWORK_PRINTER = [0xc3,0x02,0xc1],

        END_OF_STRING = [0x00];

    if (/.*\\+$/.test(lnk_target)) {
        lnk_target = lnk_target.replace(/\\+$/g, '');
        var target_is_folder = true;
    }

    var prefix_root, item_data, target_root, target_leaf;
    if (lnk_target.substr(0, 2) === '\\\\') {
        prefix_root = PREFIX_NETWORK_ROOT;
        item_data = [0x1f, 0x58].concat(CLSID_Network);
        target_root = lnk_target.subtr(lnk_target.lastIndexOf('\\'));
        if (/\\\\.*\\.*/.test(lnk_target)) {
            target_leaf = lnk_target.substr(lnk_target.lastIndexOf('\\') + 1);
        }
        if (target_root === '\\') {
            target_root = lnk_target;
        }
    } else {
        prefix_root = PREFIX_LOCAL_ROOT;
        item_data = [0x1f, 0x50].concat(CLSID_Computer);
        target_root = lnk_target.replace(/\\.*$/, '\\');
        if (/.*\\.*/.test(lnk_target)) {
            target_leaf = lnk_target.replace(/^.*?\\/, '');
        }
    }

    var prefix_of_target, file_attributes;
    if (!target_is_folder) {
        prefix_of_target = PREFIX_FILE;
        file_attributes = FileAttributes_File;
    } else {
        prefix_of_target = PREFIX_FOLDER;
        file_attributes = FileAttributes_Directory;
    }

    target_root = str_to_arr(target_root);
    for (var i = 1; i <= 21; ++i) {
        target_root.push(0);
    }

    var id_list_items = gen_IDLIST(item_data);
    id_list_items = id_list_items.concat(
            gen_IDLIST(prefix_root.concat(target_root, END_OF_STRING)));
    if (target_leaf) {
        target_leaf = str_to_arr(target_leaf);
        id_list_items = id_list_items.concat(
                gen_IDLIST(prefix_of_target.concat(target_leaf, END_OF_STRING)));
    }
    var id_list = gen_IDLIST(id_list_items);

    var data = [].concat(HeaderSize,
                         LinkCLSID,
                         LinkFlags,
                         file_attributes,
                         CreationTime,
                         AccessTime,
                         WriteTime,
                         FileSize,
                         IconIndex,
                         ShowCommand,
                         Hotkey,
                         Reserved,
                         Reserved2,
                         Reserved3,
                         id_list,
                         TerminalID);
    return new Blob([new Uint8Array(data)], { type: 'application/x-ms-shortcut' });
}

var blob = create_lnk_blob('C:\\Windows\\System32\\Calc.exe');

使用方法如下:

var blob_to_file = create_lnk_blob('C:\\Windows\\System32\\Calc.exe');
var blob_to_folder = create_lnk_blob('C:\\Users\\Myself\\Desktop\\'); // with a trailing slash

Demo: http://jsfiddle.net/5cjgLyan/2/


1
这是我找到的最佳解决方案,但浏览器似乎不喜欢.lnk文件(我还尝试在服务器端生成lnk文件)。 - EKS
@kay,这太棒了!但是如何向目标添加参数? - Dee
@datdinhquoc,很抱歉,我不知道。我只是将脚本转换而已,没有理解它。 - Kijewski
2
@Dee 在版本1.2中更新了mslink.sh,添加了一些新选项和参数支持。我基于最新的mslink.sh和kay的答案进行了转换。这可以帮助您,这是我的gist链接:https://gist.github.com/zgr0629/5c2e13e07c34e867420f282941a6adf0 - Gary

-1
如果您的网站允许使用PHP,这将变得非常简单。
如果您的脚本是HTML文件的一部分,只需像编写发送静态lnk文件的javascript一样编写即可。然后,在lnk地址部分,将javascript分成两个部分,分解为HTML。然后在那个点上,插入


<?php /*PHP code set a variable *? /* PHP code to generate proper string*/ PRINT /*PHP variable*/
?>

-3

我认为让它成为纯客户端是不可能的。 即使是WebRTC协议也需要至少一个iceServer来信令其他客户端。

而我认为最简单的方法是使用http://peerjs.com/

你可以先创建房主的客户端令牌。

//room owner side
peer.on('open', function(my_peer_id) {
  console.log('My peer ID is: ' + my_peer_id);
});

并将令牌发送给任何想要的其他人(通过文本文件、网络聊天等方式)

然后其他人可以使用上面的令牌进行连接

//the other one
var conn = peer.connect(other_peer_id);

当房主检测到有人进入房间后。

与信号服务器断开连接,以使令牌无法使用。

//room owner side
peer.disconnect()

关于客户端生成和读取文件,我建议您阅读下面的文章。
  1. http://www.html5rocks.com/en/tutorials/file/dndfiles/ 从文件中读取
  2. 如何使用filesaver.js 另存为文件
我相信fileReader api和blob的兼容性并不重要。因为永远不会有一个支持webrtc但不支持fileReader api的浏览器。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接