8

我将首先解释我的目标以避免“XY 问题”的误解。我希望能够生成一个文件(在 Linux 上),当下载到 Windows 机器并双击时,将打开一个(ms office,但 IIUC 无关紧要)具有已知路径的文件(在本地 Windows 上)机)由服务器预设。

如果我在 lnk 文件中包含我要打开的文件的路径(以及关于它存储在其中的卷的其他详细信息等),似乎我正在尝试做的事情是可能的。我首先检查了 lnk 文件格式规范,试图在 python 脚本中生成文件,但这不是我的小菜一碟(我以前从未编写过代码来处理二进制文件)。此外,我想放弃该格式的许多功能,但我仍然需要了解它们的标志/字段/长度/终止等。

然后我偶然发现了 winedump 实用程序和 WINE 在我的 ~/.wine 文件夹中创建的一些 .lnk 文件。WINE 代码库中必须有一些函数能够生成 lnk 文件,但我找不到它。它在哪里?是不是像win的mklink这样的命令形式?如果不是(如果不是,我猜它是某个地方的函数)你能把它指向我以便我可以编写一个包装器命令吗?

或者有没有可以编写 lnk 文件并在 GNU/Linux 上运行的库?

4

1 回答 1

8

这些未经我测试,但我确实在此站点上找到了 2 个工具,它们确实描述了这 2 个工具,如下所示:

此应用程序允许您创建 Windows 快捷方式文件(扩展名 .LNK),而无需 Windows 操作系统。

第一个应用程序是C 应用程序,第二个是 Bash shell 脚本。这是Shell脚本,mslink.sh

#!/bin/bash

#############################################################################################
# mslink.sh v1.0
#############################################################################################
# Ce script permet de créer un Raccourci Windows (Fichier .LNK)
# Script créé en se basant sur la doc 
#   http://msdn.microsoft.com/en-us/library/dd871305.aspx
#############################################################################################

OPTIONS=$(getopt -q -n ${0} -o hl:o:p -l help,lnk-target:,output-file:,printer-link -- "$@")

eval set -- ${OPTIONS}

IS_PRINTER_LNK=0
while true; do
  case "$1" in
    -h|--help) HELP=1 ;;
    -l|--lnk-target) LNK_TARGET="$2" ; shift ;;
    -o|--output-file) OUTPUT_FILE="$2" ; shift ;;
    -p|--printer-link) IS_PRINTER_LNK=1 ;;
    --)        shift ; break ;;
    *)         echo "Option inconnue : $1" ; exit 1 ;;
  esac
  shift
done

if [ $# -ne 0 ]; then
  echo "Option(s) inconnue(s) : $@"
  exit 1
fi

[ ${#LNK_TARGET} -eq 0 ] || [ ${#OUTPUT_FILE} -eq 0 ] && echo "
Usage :
 ${0} -l cible_du_fichier_lnk -o mon_fichier.lnk [-p]

Options :
 -l, --lnk-target               Précise la cible du raccourci
 -o, --output-file              Enregistre le raccourci dans un fichier
 -p, --printer-link             Génère un raccourci de type imprimante réseau
" && exit 1

#############################################################################################
# Fonctions
#############################################################################################

function ascii2hex() {
    echo $(echo -n ${1} | hexdump -v -e '/1 " x%02x"'|sed s/\ /\\\\/g)
}

function gen_IDLIST() {
        ITEM_SIZE=$(printf '%04x' $((${#1}/4+2)))
        echo '\x'${ITEM_SIZE:2:2}'\x'${ITEM_SIZE:0:2}${1}
}

function convert_CLSID_to_DATA() {
    echo -n ${1:6:2}${1:4:2}${1:2:2}${1:0:2}${1:11:2}${1:9:2}${1:16:2}${1:14:2}${1:19:4}${1:24:12}|sed s/"\([A-Fa-f0-9][A-Fa-f0-9]\)"/\\\\x\\1/g
}

#############################################################################################
# Variables issues de la documentation officielle de Microsoft
#############################################################################################

HeaderSize='\x4c\x00\x00\x00'                           # HeaderSize
LinkCLSID=$(convert_CLSID_to_DATA "00021401-0000-0000-c000-000000000046")   # LinkCLSID
LinkFlags='\x01\x01\x00\x00'                            # HasLinkTargetIDList ForceNoLinkInfo 

FileAttributes_Directory='\x10\x00\x00\x00'                 # FILE_ATTRIBUTE_DIRECTORY
FileAttributes_File='\x20\x00\x00\x00'                      # FILE_ATTRIBUTE_ARCHIVE

CreationTime='\x00\x00\x00\x00\x00\x00\x00\x00'
AccessTime='\x00\x00\x00\x00\x00\x00\x00\x00'
WriteTime='\x00\x00\x00\x00\x00\x00\x00\x00'

FileSize='\x00\x00\x00\x00'
IconIndex='\x00\x00\x00\x00'
ShowCommand='\x01\x00\x00\x00'                          # SW_SHOWNORMAL
Hotkey='\x00\x00'                               # No Hotkey
Reserved='\x00\x00'                             # Valeur non modifiable
Reserved2='\x00\x00\x00\x00'                            # Valeur non modifiable
Reserved3='\x00\x00\x00\x00'                            # Valeur non modifiable
TerminalID='\x00\x00'                               # Valeur non modifiable

CLSID_Computer="20d04fe0-3aea-1069-a2d8-08002b30309d"               # Poste de travail
CLSID_Network="208d2c60-3aea-1069-a2d7-08002b30309d"                # Favoris réseau

#############################################################################################
# Constantes trouvées à partir de l'analyse de fichiers lnk
#############################################################################################

PREFIX_LOCAL_ROOT='\x2f'                            # Disque local
PREFIX_FOLDER='\x31\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'        # Dossier de fichiers
PREFIX_FILE='\x32\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'          # Fichier
PREFIX_NETWORK_ROOT='\xc3\x01\x81'                      # Racine de serveur de fichiers réseau
PREFIX_NETWORK_PRINTER='\xc3\x02\xc1'                       # Imprimante réseau

END_OF_STRING='\x00'

#############################################################################################

# On retire l'anti-slash final s'il y en a un
LNK_TARGET=${LNK_TARGET%\\}

# On sépare le chemin racine du lien de la cible finale
# On distingue aussi si le lien est de type local ou réseau
# On définie la valeur Item_Data suivant le cas d'un lien réseau ou local

IS_ROOT_LNK=0
IS_NETWORK_LNK=0

if [[ ${LNK_TARGET} == \\\\* ]]; then
    IS_NETWORK_LNK=1
    PREFIX_ROOT=${PREFIX_NETWORK_ROOT}
    Item_Data='\x1f\x58'$(convert_CLSID_to_DATA ${CLSID_Network})

        TARGET_ROOT=${LNK_TARGET%\\*}
        if [[ ${LNK_TARGET} == \\\\*\\* ]]; then
                TARGET_LEAF=${LNK_TARGET##*\\}
        fi
        if [ ${TARGET_ROOT} == \\ ]; then
                TARGET_ROOT=${LNK_TARGET}
        fi
else
    PREFIX_ROOT=${PREFIX_LOCAL_ROOT}
    Item_Data='\x1f\x50'$(convert_CLSID_to_DATA ${CLSID_Computer})

    TARGET_ROOT=${LNK_TARGET%%\\*}
        if [[ ${LNK_TARGET} == *\\* ]]; then
        TARGET_LEAF=${LNK_TARGET#*\\}
        fi
    [[ ! ${TARGET_ROOT} == *\\ ]] && TARGET_ROOT=${TARGET_ROOT}'\'
fi

if [ ${IS_PRINTER_LNK} -eq 1 ]; then
    PREFIX_ROOT=${PREFIX_NETWORK_PRINTER}
    TARGET_ROOT=${LNK_TARGET}
    IS_ROOT_LNK=1
fi

[ ${#TARGET_LEAF} -eq 0 ] && IS_ROOT_LNK=1

#############################################################################################

# On sélectionne le préfixe qui sera utilisé pour afficher l'icône du raccourci

if [[ ${TARGET_LEAF} == *.??? ]]; then
    PREFIX_OF_TARGET=${PREFIX_FILE}
    TYPE_TARGET="fichier"
    FileAttributes=${FileAttributes_File}
else
    PREFIX_OF_TARGET=${PREFIX_FOLDER}
    TYPE_TARGET="dossier"
    FileAttributes=${FileAttributes_Directory}
fi

# On convertit les valeurs des cibles en binaire
TARGET_ROOT=$(ascii2hex ${TARGET_ROOT})
TARGET_ROOT=${TARGET_ROOT}$(for i in `seq 1 21`;do echo -n '\x00';done) # Nécessaire à partir de Vista et supérieur sinon le lien est considéré comme vide (je n'ai trouvé nul part d'informations à ce sujet)

TARGET_LEAF=$(ascii2hex ${TARGET_LEAF})

# On crée l'IDLIST qui représente le cœur du fichier LNK

if [ ${IS_ROOT_LNK} -eq 1 ];then
    IDLIST_ITEMS=$(gen_IDLIST ${Item_Data})$(gen_IDLIST ${PREFIX_ROOT}${TARGET_ROOT}${END_OF_STRING})
else
    IDLIST_ITEMS=$(gen_IDLIST ${Item_Data})$(gen_IDLIST ${PREFIX_ROOT}${TARGET_ROOT}${END_OF_STRING})$(gen_IDLIST ${PREFIX_OF_TARGET}${TARGET_LEAF}${END_OF_STRING})
fi

IDLIST=$(gen_IDLIST ${IDLIST_ITEMS})

#############################################################################################

if [ ${IS_NETWORK_LNK} -eq 1 ]; then
    TYPE_LNK="réseau"
    if [ ${IS_PRINTER_LNK} -eq 1 ]; then
        TYPE_TARGET="imprimante"
    fi
else
    TYPE_LNK="local"
fi

echo "Création d'un raccourci de type \""${TYPE_TARGET}" "${TYPE_LNK}"\" avec pour cible "${LNK_TARGET} 1>&2

echo -ne ${HeaderSize}${LinkCLSID}${LinkFlags}${FileAttributes}${CreationTime}${AccessTime}${WriteTime}${FileSize}${IconIndex}${ShowCommand}${Hotkey}${Reserved}${Reserved2}${Reserved3}${IDLIST}${TerminalID} > "${OUTPUT_FILE}"
于 2014-05-25T06:23:51.287 回答