Jump to content

Recommended Posts

Posted (edited)

I tried to implement the code in this topic:

Firstly, i have no idea how these lines of code work but meanwhile i noticed that:

; Everytime autoit wants to call a method, get or set a property in a object it needs to go to
; IDispatch::GetIDsFromNames. This is our version of that function, note that by defining this ourselves
; we can fool autoit to believe that the object supports a lot of different properties/methods.
Func __IDispatch_GetIDsFromNames($pSelf, $riid, $rgszNames, $cNames, $lcid, $rgDispId)
...
EndFunc

The problem is i ran into is that some object calls didn't go through IDispatch::GetIDsFromNames.
Here is the code to replicate what i'm mentioning:

  Reveal hidden contents

I followed the example in the topic and tried to do the same thing with method .Documents (line 193) and .Open (line 194) but didn't get the same result because .Documents was being passed through __IDispatch_GetIDsFromNames while .Open didn't.

$Au3_CallByName = 'Documents'
Local $oDoc = $oAppl.Au3_CallByName
$Au3_CallByName = 'Open'
$oDoc = $oDoc.Au3_CallByName($sFilePath, $bConfirmConversions, $bReadOnly, $bAddToRecentFiles, $sOpenPassword, "", $bRevert, $sWritePassword, "", $iFormat)

Console outputs:

==> The requested action with this object has failed.:
$oDoc = $oDoc.Au3_CallByName($sFilePath, $bConfirmConversions, $bReadOnly, $bAddToRecentFiles, $sOpenPassword, "", $bRevert, $sWritePassword, "", $iFormat)
$oDoc = $oDoc^ ERROR

Is there any workarounds to solve this?

Thank you!

Edited by sylremo
Posted (edited)

Have you tried hooking the word object instead?

Au3_CallByName_Init(True, "Word.Application")

you might need it before or after Word_Create()

 

@ad777 the udf takes care of that

Edited by Bilgus
Posted (edited)

Hi @Bilgus, thank you for your quick response

  On 4/15/2022 at 11:34 PM, Bilgus said:

Have you tried hooking the word object instead?

Expand  

Yes, i tried but it made no differences.

I came up with another example

  Reveal hidden contents


This JSON object after calling parse method will accept every method and property names, so obviously it can not be passed through __IDispatch_GetIDsFromNames, so i think maybe i will try to intercept every dot statement, and this is the question i have for this post.

I tried to hook the HTMLFILE object, place it both before and after the ObjCreate statement.

Au3_CallByName_Init(True, 'HTMLFILE')

This time i commented out the "If" statement that indicate that AutoIt didn't find the method name (now __IDispatch_GetIDsFromNames should be triggered on every dot statement), slightly modified CBN_CB_Au3_CallByName to use its own param $sName as method name instead of $Au3_CallByName, and added a trace line in order to track those names.

Func __IDispatch_GetIDsFromNames($pSelf, $riid, $rgszNames, $cNames, $lcid, $rgDispId)
    Local Const $DISP_E_UNKNOWNNAME = 0x80020006, $DISPID_UNKNOWN = -1
    Local Static $pGIFN = __Pointer_GetIDsFromNames()
    Local $hRes

    ;Call the original GetIDsFromNames
    $hRes = DllCallAddress("LRESULT", $pGIFN, "ptr", $pSelf, "ptr", $riid, _
            "struct*", $rgszNames, "dword", $cNames, "dword", $lcid, "ptr", $rgDispId)
    If @error Then
        ConsoleWriteError("Error: GetIDsFromNames: " & @error & @CRLF)
        Return $DISP_E_UNKNOWNNAME
    EndIf

    ; Autoit didnt find the name now its our turn
    ;~ If $DISPID_UNKNOWN = DllStructGetData(DllStructCreate("long[" & $cNames & "]", $rgDispId), 1, 1) Then
        ;$rgszNames is a pointer to an array[$cNames] of names -- Autoit only asks for one member $cNames = 1
        Local $tName = DllStructCreate("wchar[64]", DllStructGetData(DllStructCreate("ptr[" & $cNames & "]", $rgszNames), 1, 1))
        Local $sName = DllStructGetData($tName, 1)
        ConsoleWrite('__IDispatch_GetIDsFromNames triggered: ' & $sName & @CRLF)

        ;We just prepend CBN_CB_ to the function name and try to call it
        $hRes = Call("CBN_CB_" & $sName, $sName, $pGIFN, $pSelf, $riid, $rgszNames, $cNames, $lcid, $rgDispId)
        If Not @error And $hRes <> Default Then Return $hRes         ; User handled the function

        ;Call the original GetIDsFromNames
        $hRes = DllCallAddress("LRESULT", $pGIFN, "ptr", $pSelf, "ptr", $riid, _
                "struct*", $rgszNames, "dword", $cNames, "dword", $lcid, "ptr", $rgDispId)
        If @error Then
            ConsoleWrite("Error: GetIDsFromNames: " & @error & @CRLF)
            Return $DISP_E_UNKNOWNNAME
        EndIf
    ;~ EndIf
    Return $hRes[0]
EndFunc   ;==>__IDispatch_GetIDsFromNames
Func CBN_CB_Au3_CallByName($sName, $pGIFN, $pSelf, $riid, $rgszNames, $cNames, $lcid, $rgDispId)
    Local Const $DISP_E_UNKNOWNNAME = 0x80020006
    ;~ ConsoleWrite(">Call By Name: " & $sName & " -> " & $Au3_CallByName & @CRLF)
    Local Static $tpMember = DllStructCreate("ptr")
    ;~ If $Au3_CallByName Then
        Local $hRes, $tMember

        ;ConsoleWrite("CallByName: " & $Au3_CallByName & @CRLF)
        $tMember = DllStructCreate("wchar[" & StringLen($sName) + 1 & "]")
        DllStructSetData($tMember, 1, $sName)
        DllStructSetData($tpMember, 1, DllStructGetPtr($tMember))
        $rgszNames = $tpMember
        ;~ $Au3_CallByName = 0

        ;Call the original GetIDsFromNames
        $hRes = DllCallAddress("LRESULT", $pGIFN, "ptr", $pSelf, "ptr", $riid, _
                "struct*", $rgszNames, "dword", $cNames, "dword", $lcid, "ptr", $rgDispId)
        If @error Then
            ConsoleWrite("Error: Call By Name: " & @error & @CRLF)
            Return $DISP_E_UNKNOWNNAME
        EndIf
        Return $hRes[0]
    ;~ EndIf
    Return Default     ;Default handler
EndFunc   ;==>CBN_CB_Au3_CallByName


Here are the console outputs:

__IDispatch_GetIDsFromNames triggered: parentwindow
__IDispatch_GetIDsFromNames triggered: parentwindow

Of course i did manually check with IsObj to make sure that every dot statement is being used on a valid object. And according to console, only parentwindow was caught, while execScript, eval, parse, title didn't.

Edited by sylremo
provide more infos, correct grammars
Posted

Sorry you haven't gotten any satisfactory answers so far, I don't have a windows machine to test on ATM so I'm not of much help

Likely whats going on is the object is in a different process 'multithreaded apartment' you could look at the

TLB in a decent object browser Com View https://www.softpedia.com/get/System/System-Info/COMView.shtml

 

 

 

Create an account or sign in to comment

You need to be a member in order to leave a comment

Create an account

Sign up for a new account in our community. It's easy!

Register a new account

Sign in

Already have an account? Sign in here.

Sign In Now
×
×
  • Create New...