Giter VIP home page Giter VIP logo

chrome.ahk's People

Contributors

g33kdude avatar raptorx avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

chrome.ahk's Issues

WaitForLoad not waiting on pages opened directly on chromeinstance

I think I have found a bug in Chrome.ahk. When instantiating Chrome with url included, WaitForLoad() Method does not work. Using the code example from the 1st page, things work as expected.

#Include Chrome.ahk

; Create an instance of the Chrome class using
; the folder ChromeProfile to store the user profile
FileCreateDir, ChromeProfile
ChromeInst := new Chrome("ChromeProfile")

; Connect to the newly opened tab and navigate to another website
; Note: If your first action is to navigate away, it may be just as
; effective to provide the target URL when instantiating the Chrome class
PageInstance := ChromeInst.GetPage()
PageInstance.Call("Page.navigate", {"url": "https://autohotkey.com/"})
PageInstance.WaitForLoad()

; Execute some JavaScript
PageInstance.Evaluate("alert('Hello World!');")

; Close the browser (note: this closes *all* pages/tabs)
PageInstance.Call("Browser.close")
PageInstance.Disconnect()

ExitApp
return

Now lets change it as suggested in the comment "...it may be just as effective to provide the target URL when instantiating the Chrome class"

#Include Chrome.ahk
FileCreateDir, ChromeProfile
ChromeInst := new Chrome("ChromeProfile", "https://autohotkey.com/") ; this line changed
PageInstance := ChromeInst.GetPage()
; PageInstance.Call("Page.navigate", {"url": "https://autohotkey.com/"}) ; this line removed
PageInstance.WaitForLoad()
PageInstance.Evaluate("alert('Hello World!');")
PageInstance.Call("Browser.close")
PageInstance.Disconnect()

ExitApp
return

The alert fires before the page is fully loaded and causes a subsequent error. Here is a workaround.

#Include Chrome.ahk
url := "https://autohotkey.com/"
ChromeInst := new Chrome("ChromeProfile", url) ; new instance
PageInstance := ChromeInst.GetPage()
PageInstance.Call("Page.navigate", {"url": url}) ; must renavigate to make WaitForLoad() work
PageInstance.WaitForLoad()
PageInstance.Evaluate("alert('Hello World!');")
PageInstance.Call("Browser.close")
PageInstance.Disconnect()
ExitApp
return

How to get JSON Response From The Web?

;Some javascript
js =
(
var xhr = new XMLHttpRequest();
xhr.withCredentials = true;

xhr.addEventListener("readystatechange", function() {
  if(this.readyState === 4) {
    alert(this.responseText);
    return this.responseText;
  }
});
xhr.open("POST", "https://192.168.2.1/api/commissioning");
xhr.send();
)

; Execute some JavaScript
PageInstance.Evaluate(js)

My question is how do you get the this.responseText (as a JSON format) and parse it to ahk variable?
Thanks,

TIMEOUT on PageInstance.Call("Runtime.evaluate"

First of all this function and probably others should have a timeout to allow for exception handling, as it is this will fail in a loop after a variable number of loops, sometimes the first sometimes after 100 and sometimes not before 300 but it will fail (I'm searching for a selector that never exists for this test):

document.querySelector('#XbirthLabel') != null ? 'Exists' : 'Not Found';

The following code is basically from chrome.ahk with some debugging and the "timeout parameter":

;============================================================================
TEST_MyDbgEvaluate(PageInstance, JS)
;     https://chromedevtools.github.io/devtools-protocol/tot/Runtime/#method-evaluate
; RC: https://chromedevtools.github.io/devtools-protocol/tot/Runtime/#type-RemoteObject
;============================================================================
{
    MyDebug("TEST_DEBUG: PageInstance - Runtime.evaluate : BEFORE - timeout 5000ms (5 seconds)")
    response := PageInstance.Call("Runtime.evaluate",
                                        ( LTrim Join
                                            {
                                                "expression": JS,
                                                "objectGroup": "console",
                                                "includeCommandLineAPI": Chrome.Jxon_True(),
                                                "silent": Chrome.Jxon_False(),
                                                "returnByValue": Chrome.Jxon_False(),
                                                "userGesture": Chrome.Jxon_True(),
                                                "timeout":     2000,
                                                "awaitPromise": Chrome.Jxon_False()
                                            }
                                        ))
    MyDebug("TEST_DEBUG: PageInstance - Runtime.evaluate : AFTER")
    if  (response.exceptionDetails)
    {
        MyDebug("TEST_DEBUG: ######################### TIMEOUT ???????????? ##########################")
        throw Exception(response.result.description,, Chrome.Jxon_Dump(response.exceptionDetails))
    }

    MyDebug("TEST_DEBUG: PageInstance - Runtime.evaluate : RETURNING TO CALLER")
    return response.result

}

;===========================================================================================
TEST_MyBrowserElementExists(PageInstance, JsSelector, Dbg:=true)
;
; AtMemorialPage := MyBrowserElementExists(CreatePage, SaveToVirtCem, true)
;
; https://www.autohotkey.com/boards/viewtopic.php?t=88258
; SearchPage.WaitForLoad()
; ### NOTE ###
; "JsRc := PageInstance.Evaluate(Js).value" sometimes hangs
;===========================================================================================
{
    FoundIt    := "Exists"
    JavaScript := "document.querySelector('" . JsSelector . "') != null ? '" . FoundIt . "' : 'Not Found';"
    MyDebug("TEST_MyBrowserElementExists(): JS: " . JavaScript)
    MyDebugIndent()
        SetTimer, TimedOut_TEST_MyBrowserElementExists, 5000                 ;1 second should be way more than enough, use 5
            ;RemoteObject := PageInstance.Evaluate(JavaScript)
             RemoteObject := TEST_MyDbgEvaluate(PageInstance, JavaScript)
        SetTimer, TimedOut_TEST_MyBrowserElementExists, off
        JsRc         := RemoteObject.value
        if  (JsRc != FoundIt)
            JsRcTF := false       ;;Doesn't exist
        else
            JsRcTF := true
        MyDebug( "TEST_MyBrowserElementExists(), RC=" . JsRcTF . "(" . JsRc . "), JsSelector=" . JsSelector )
    MyDebugIndent(-1)
    return JsRcTF

The timeout parameter doesn't work. I can of course set a AHK timer and die (as the above code does) but I want the query to return false (or simply retry the call) on hang and the caller can then retry. Do I need some magic like "Chrome.Jxon_False()" to pass numbers?

I can't see how true or false is being passed in the following "chrome.ahk" code (but then I'm no AHK expert):

	Jxon_True()
	{
		static obj := {}
		return obj
	}
	
	Jxon_False()
	{
		static obj := {}
		return obj
	}

Adding pageData to pag Class

For now there is no way to access to the ID of a "Page" instance, and if you are using a lot of pages and you want to change instances, it becomes anoying.

So i made a little change to how the page instance are generated, pasing all the pageData object instead of only the WebSocket, here you have the code.

`; Chrome.ahk v1.2
; Copyright GeekDude 2018
; https://github.com/G33kDude/Chrome.ahk

class Chrome
{
static DebugPort := 9222

/*
	Escape a string in a manner suitable for command line parameters
*/
CliEscape(Param)
{
	return """" RegExReplace(Param, "(\\*)""", "$1$1\""") """"
}

/*
	Finds instances of chrome in debug mode and the ports they're running
	on. If no instances are found, returns a false value. If one or more
	instances are found, returns an associative array where the keys are
	the ports, and the values are the full command line texts used to start
	the processes.

	One example of how this may be used would be to open chrome on a
	different port if an instance of chrome is already open on the port
	you wanted to used.

	```
	; If the wanted port is taken, use the largest taken port plus one
	DebugPort := 9222
	if (Chromes := Chrome.FindInstances()).HasKey(DebugPort)
		DebugPort := Chromes.MaxIndex() + 1
	ChromeInst := new Chrome(ProfilePath,,,, DebugPort)
	```

	Another use would be to scan for running instances and attach to one
	instead of starting a new instance.

	```
	if (Chromes := Chrome.FindInstances())
		ChromeInst := {"base": Chrome, "DebugPort": Chromes.MinIndex()}
	else
		ChromeInst := new Chrome(ProfilePath)
	```
*/
FindInstances()
{
	static Needle := "--remote-debugging-port=(\d+)"
	Out := {}
	for Item in ComObjGet("winmgmts:")
		.ExecQuery("SELECT CommandLine FROM Win32_Process"
		. " WHERE Name = 'chrome.exe'")
		if RegExMatch(Item.CommandLine, Needle, Match)
			Out[Match1] := Item.CommandLine
	return Out.MaxIndex() ? Out : False
}

/*
	ProfilePath - Path to the user profile directory to use. Will use the standard if left blank.
	URLs        - The page or array of pages for Chrome to load when it opens
	Flags       - Additional flags for chrome when launching
	ChromePath  - Path to chrome.exe, will detect from start menu when left blank
	DebugPort   - What port should Chrome's remote debugging server run on
*/
__New(ProfilePath:="", URLs:="about:blank", Flags:="", ChromePath:="", DebugPort:="")
{
	; Verify ProfilePath
	if (ProfilePath != "" && !InStr(FileExist(ProfilePath), "D"))
		throw Exception("The given ProfilePath does not exist")
	this.ProfilePath := ProfilePath

	; Verify ChromePath
	if (ChromePath == "")
		FileGetShortcut, %A_StartMenuCommon%\Programs\Google Chrome.lnk, ChromePath
	if (ChromePath == "")
		RegRead, ChromePath, HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\App Pahs\chrome.exe
	if !FileExist(ChromePath)
		throw Exception("Chrome could not be found")
	this.ChromePath := ChromePath

	; Verify DebugPort
	if (DebugPort != "")
	{
		if DebugPort is not integer
			throw Exception("DebugPort must be a positive integer")
		else if (DebugPort <= 0)
			throw Exception("DebugPort must be a positive integer")
		this.DebugPort := DebugPort
	}

	; Escape the URL(s)
	for Index, URL in IsObject(URLs) ? URLs : [URLs]
		URLString .= " " this.CliEscape(URL)

	Run, % this.CliEscape(ChromePath)
	. " --remote-debugging-port=" this.DebugPort
	. (ProfilePath ? " --user-data-dir=" this.CliEscape(ProfilePath) : "")
	. (Flags ? " " Flags : "")
	. URLString
	,,, OutputVarPID
	this.PID := OutputVarPID
}

/*
	End Chrome by terminating the process.
*/
Kill()
{
	Process, Close, % this.PID
}

/*
	Queries chrome for a list of pages that expose a debug interface.
	In addition to standard tabs, these include pages such as extension
	configuration pages.
*/
GetPageList()
{
	http := ComObjCreate("WinHttp.WinHttpRequest.5.1")
	http.open("GET", "http://127.0.0.1:" this.DebugPort "/json")
	http.send()
	return this.Jxon_Load(http.responseText)
}

/*
	Returns a connection to the debug interface of a page that matches the
	provided criteria. When multiple pages match the criteria, they appear
	ordered by how recently the pages were opened.

	Key        - The key from the page list to search for, such as "url" or "title"
	Value      - The value to search for in the provided key
	MatchMode  - What kind of search to use, such as "exact", "contains", "startswith", or "regex"
	Index      - If multiple pages match the given criteria, which one of them to return
	fnCallback - A function to be called whenever message is received from the page
*/
GetPageBy(Key, Value, MatchMode:="exact", Index:=1, fnCallback:="")
{
	Count := 0
	for n, PageData in this.GetPageList()
	{
		if (((MatchMode = "exact" && PageData[Key] = Value) ; Case insensitive
			|| (MatchMode = "contains" && InStr(PageData[Key], Value))
			|| (MatchMode = "startswith" && InStr(PageData[Key], Value) == 1)
			|| (MatchMode = "regex" && PageData[Key] ~= Value))
			&& ++Count == Index)
			return new this.Page(PageData, fnCallback)
	}
}

/*
	Shorthand for GetPageBy("url", Value, "startswith")
*/
GetPageByURL(Value, MatchMode:="startswith", Index:=1, fnCallback:="")
{
	return this.GetPageBy("url", Value, MatchMode, Index, fnCallback)
}

/*
	Shorthand for GetPageBy("title", Value, "startswith")
*/
GetPageByTitle(Value, MatchMode:="startswith", Index:=1, fnCallback:="")
{
	return this.GetPageBy("title", Value, MatchMode, Index, fnCallback)
}

/*
	Shorthand for GetPageBy("type", Type, "exact")

	The default type to search for is "page", which is the visible area of
	a normal Chrome tab.
*/
GetPage(Index:=1, Type:="page", fnCallback:="")
{
	return this.GetPageBy("type", Type, "exact", Index, fnCallback)
}

/*
	Connects to the debug interface of a page given its WebSocket URL.
*/
class Page
{
	Connected := False
	ID := 0
	Responses := []
	pageData := {}

	/*
	  pageData   - object with the information about the page.
		fnCallback - A function to be called whenever message is received
	*/
	__New(pageData, fnCallback:="")
	{
		this.pageData := pageData
		; wsurl      - The desired page's WebSocket URL
		wsurl := pageData.webSocketDebuggerUrl
		this.fnCallback := fnCallback
		this.BoundKeepAlive := this.Call.Bind(this, "Browser.getVersion",, False)

		; TODO: Throw exception on invalid objects
		if IsObject(wsurl)
			wsurl := wsurl.webSocketDebuggerUrl

		wsurl := StrReplace(wsurl, "localhost", "127.0.0.1")
		this.ws := {"base": this.WebSocket, "_Event": this.Event, "Parent": this}
		this.ws.__New(wsurl)

		while !this.Connected
			Sleep, 50
	}

	/*
		Calls the specified endpoint and provides it with the given
		parameters.

		DomainAndMethod - The endpoint domain and method name for the
		endpoint you would like to call. For example:
		PageInst.Call("Browser.close")
		PageInst.Call("Schema.getDomains")

		Params - An associative array of parameters to be provided to the
		endpoint. For example:
		PageInst.Call("Page.printToPDF", {"scale": 0.5 ; Numeric Value
		, "landscape": Chrome.Jxon_True() ; Boolean Value
		, "pageRanges: "1-5, 8, 11-13"}) ; String value
		PageInst.Call("Page.navigate", {"url": "https://autohotkey.com/"})

		WaitForResponse - Whether to block until a response is received from
		Chrome, which is necessary to receive a return value, or whether
		to continue on with the script without waiting for a response.
	*/
	Call(DomainAndMethod, Params:="", WaitForResponse:=True)
	{
		if !this.Connected
			throw Exception("Not connected to tab")

		; Use a temporary variable for ID in case more calls are made
		; before we receive a response.
		ID := this.ID += 1
		this.ws.Send(Chrome.Jxon_Dump({"id": ID
		, "params": Params ? Params : {}
		, "method": DomainAndMethod}))

		if !WaitForResponse
			return

		; Wait for the response
		this.responses[ID] := False
		while !this.responses[ID]
			Sleep, 50

		; Get the response, check if it's an error
		response := this.responses.Delete(ID)
		if (response.error)
			throw Exception("Chrome indicated error in response",, Chrome.Jxon_Dump(response.error))

		return response.result
	}

	/*
		Run some JavaScript on the page. For example:

		PageInst.Evaluate("alert(""I can't believe it's not IE!"");")
		PageInst.Evaluate("document.getElementsByTagName('button')[0].click();")
	*/
	Evaluate(JS)
	{
		response := this.Call("Runtime.evaluate",
		( LTrim Join
		{
			"expression": JS,
			"objectGroup": "console",
			"includeCommandLineAPI": Chrome.Jxon_True(),
			"silent": Chrome.Jxon_False(),
			"returnByValue": Chrome.Jxon_False(),
			"userGesture": Chrome.Jxon_True(),
			"awaitPromise": Chrome.Jxon_False()
		}
		))

		if (response.exceptionDetails)
			throw Exception(response.result.description,, Chrome.Jxon_Dump(response.exceptionDetails))

		return response.result
	}

	/*
		Waits for the page's readyState to match the DesiredState.

		DesiredState - The state to wait for the page's ReadyState to match
		Interval     - How often it should check whether the state matches
	*/
	WaitForLoad(DesiredState:="complete", Interval:=100)
	{
		while this.Evaluate("document.readyState").value != DesiredState
			Sleep, Interval
	}

	/*
		Internal function triggered when the script receives a message on
		the WebSocket connected to the page.
	*/
	Event(EventName, Event)
	{
		; If it was called from the WebSocket adjust the class context
		if this.Parent
			this := this.Parent

		; TODO: Handle Error events
		if (EventName == "Open")
		{
			this.Connected := True
			BoundKeepAlive := this.BoundKeepAlive
			SetTimer, %BoundKeepAlive%, 15000
		}
		else if (EventName == "Message")
		{
			data := Chrome.Jxon_Load(Event.data)

			; Run the callback routine
			fnCallback := this.fnCallback
			if (newData := %fnCallback%(data))
				data := newData

			if this.responses.HasKey(data.ID)
				this.responses[data.ID] := data
		}
		else if (EventName == "Close")
		{
			this.Disconnect()
		}
		else if (EventName == "Error")
		{
			throw Exception("Websocket Error!")
		}
	}

	/*
		Disconnect from the page's debug interface, allowing the instance
		to be garbage collected.

		This method should always be called when you are finished with a
		page or else your script will leak memory.
	*/
	Disconnect()
	{
		if !this.Connected
			return

		this.Connected := False
		this.ws.Delete("Parent")
		this.ws.Disconnect()

		BoundKeepAlive := this.BoundKeepAlive
		SetTimer, %BoundKeepAlive%, Delete
		this.Delete("BoundKeepAlive")
	}

	class WebSocket
	{
		__New(WS_URL)
		{
			static wb

			; Create an IE instance
			Gui, +hWndhOld
			Gui, New, +hWndhWnd
			this.hWnd := hWnd
			Gui, Add, ActiveX, vWB, Shell.Explorer
			Gui, %hOld%: Default

			; Write an appropriate document
			WB.Navigate("about:<!DOCTYPE html><meta http-equiv='X-UA-Compatible'"
			. "content='IE=edge'><body></body>")
			while (WB.ReadyState < 4)
				sleep, 50
			this.document := WB.document

			; Add our handlers to the JavaScript namespace
			this.document.parentWindow.ahk_savews := this._SaveWS.Bind(this)
			this.document.parentWindow.ahk_event := this._Event.Bind(this)
			this.document.parentWindow.ahk_ws_url := WS_URL

			; Add some JavaScript to the page to open a socket
			Script := this.document.createElement("script")
			Script.text := "ws = new WebSocket(ahk_ws_url);`n"
			. "ws.onopen = function(event){ ahk_event('Open', event); };`n"
			. "ws.onclose = function(event){ ahk_event('Close', event); };`n"
			. "ws.onerror = function(event){ ahk_event('Error', event); };`n"
			. "ws.onmessage = function(event){ ahk_event('Message', event); };"
			this.document.body.appendChild(Script)
		}

		; Called by the JS in response to WS events
		_Event(EventName, Event)
		{
			this["On" EventName](Event)
		}

		; Sends data through the WebSocket
		Send(Data)
		{
			this.document.parentWindow.ws.send(Data)
		}

		; Closes the WebSocket connection
		Close(Code:=1000, Reason:="")
		{
			this.document.parentWindow.ws.close(Code, Reason)
		}

		; Closes and deletes the WebSocket, removing
		; references so the class can be garbage collected
		Disconnect()
		{
			if this.hWnd
			{
				this.Close()
				Gui, % this.hWnd ": Destroy"
				this.hWnd := False
			}
		}
	}
}

Jxon_Load(ByRef src, args*)
{
	static q := Chr(34)

	key := "", is_key := false
	stack := [ tree := [] ]
	is_arr := { (tree): 1 }
	next := q . "{[01234567890-tfn"
	pos := 0
	while ( (ch := SubStr(src, ++pos, 1)) != "" )
	{
		if InStr(" `t`n`r", ch)
			continue
		if !InStr(next, ch, true)
		{
			ln := ObjLength(StrSplit(SubStr(src, 1, pos), "`n"))
			col := pos - InStr(src, "`n",, -(StrLen(src)-pos+1))

			msg := Format("{}: line {} col {} (char {})"
			,   (next == "")      ? ["Extra data", ch := SubStr(src, pos)][1]
			: (next == "'")     ? "Unterminated string starting at"
			: (next == "\")     ? "Invalid \escape"
			: (next == ":")     ? "Expecting ':' delimiter"
			: (next == q)       ? "Expecting object key enclosed in double quotes"
			: (next == q . "}") ? "Expecting object key enclosed in double quotes or object closing '}'"
			: (next == ",}")    ? "Expecting ',' delimiter or object closing '}'"
			: (next == ",]")    ? "Expecting ',' delimiter or array closing ']'"
			: [ "Expecting JSON value(string, number, [true, false, null], object or array)"
			, ch := SubStr(src, pos, (SubStr(src, pos)~="[\]\},\s]|$")-1) ][1]
			, ln, col, pos)

			throw Exception(msg, -1, ch)
		}

		is_array := is_arr[obj := stack[1]]

		if i := InStr("{[", ch)
		{
			val := (proto := args[i]) ? new proto : {}
			is_array? ObjPush(obj, val) : obj[key] := val
			ObjInsertAt(stack, 1, val)

			is_arr[val] := !(is_key := ch == "{")
			next := q . (is_key ? "}" : "{[]0123456789-tfn")
		}

		else if InStr("}]", ch)
		{
			ObjRemoveAt(stack, 1)
			next := stack[1]==tree ? "" : is_arr[stack[1]] ? ",]" : ",}"
		}

		else if InStr(",:", ch)
		{
			is_key := (!is_array && ch == ",")
			next := is_key ? q : q . "{[0123456789-tfn"
		}

		else ; string | number | true | false | null
		{
			if (ch == q) ; string
			{
				i := pos
				while i := InStr(src, q,, i+1)
				{
					val := StrReplace(SubStr(src, pos+1, i-pos-1), "\\", "\u005C")
					static end := A_AhkVersion<"2" ? 0 : -1
					if (SubStr(val, end) != "\")
						break
				}
				if !i ? (pos--, next := "'") : 0
					continue

				pos := i ; update pos

				val := StrReplace(val,    "\/",  "/")
				, val := StrReplace(val, "\" . q,    q)
				, val := StrReplace(val,    "\b", "`b")
				, val := StrReplace(val,    "\f", "`f")
				, val := StrReplace(val,    "\n", "`n")
				, val := StrReplace(val,    "\r", "`r")
				, val := StrReplace(val,    "\t", "`t")

				i := 0
				while i := InStr(val, "\",, i+1)
				{
					if (SubStr(val, i+1, 1) != "u") ? (pos -= StrLen(SubStr(val, i)), next := "\") : 0
						continue 2

					; \uXXXX - JSON unicode escape sequence
					xxxx := Abs("0x" . SubStr(val, i+2, 4))
					if (A_IsUnicode || xxxx < 0x100)
						val := SubStr(val, 1, i-1) . Chr(xxxx) . SubStr(val, i+6)
				}

				if is_key
				{
					key := val, next := ":"
					continue
				}
			}

			else ; number | true | false | null
			{
				val := SubStr(src, pos, i := RegExMatch(src, "[\]\},\s]|$",, pos)-pos)

				; For numerical values, numerify integers and keep floats as is.
				; I'm not yet sure if I should numerify floats in v2.0-a ...
				static number := "number", integer := "integer"
				if val is %number%
				{
					if val is %integer%
						val += 0
				}
				; in v1.1, true,false,A_PtrSize,A_IsUnicode,A_Index,A_EventInfo,
				; SOMETIMES return strings due to certain optimizations. Since it
				; is just 'SOMETIMES', numerify to be consistent w/ v2.0-a
				else if (val == "true" || val == "false")
					val := %value% + 0
				; AHK_H has built-in null, can't do 'val := %value%' where value == "null"
				; as it would raise an exception in AHK_H(overriding built-in var)
				else if (val == "null")
					val := ""
				; any other values are invalid, continue to trigger error
				else if (pos--, next := "#")
					continue

				pos += i-1
			}

			is_array? ObjPush(obj, val) : obj[key] := val
			next := obj==tree ? "" : is_array ? ",]" : ",}"
		}
	}

	return tree[1]
}

Jxon_Dump(obj, indent:="", lvl:=1)
{
	static q := Chr(34)

	if IsObject(obj)
	{
		static Type := Func("Type")
		if Type ? (Type.Call(obj) != "Object") : (ObjGetCapacity(obj) == "")
			throw Exception("Object type not supported.", -1, Format("<Object at 0x{:p}>", &obj))

		prefix := SubStr(A_ThisFunc, 1, InStr(A_ThisFunc, ".",, 0))
		fn_t := prefix "Jxon_True",  obj_t := this ? %fn_t%(this) : %fn_t%()
		fn_f := prefix "Jxon_False", obj_f := this ? %fn_f%(this) : %fn_f%()

		if (&obj == &obj_t)
			return "true"
		else if (&obj == &obj_f)
			return "false"

		is_array := 0
		for k in obj
			is_array := k == A_Index
		until !is_array

		static integer := "integer"
		if indent is %integer%
		{
			if (indent < 0)
				throw Exception("Indent parameter must be a postive integer.", -1, indent)
			spaces := indent, indent := ""
			Loop % spaces
				indent .= " "
		}
		indt := ""
		Loop, % indent ? lvl : 0
			indt .= indent

		this_fn := this ? Func(A_ThisFunc).Bind(this) : A_ThisFunc
		lvl += 1, out := "" ; Make #Warn happy
		for k, v in obj
		{
			if IsObject(k) || (k == "")
				throw Exception("Invalid object key.", -1, k ? Format("<Object at 0x{:p}>", &obj) : "<blank>")

			if !is_array
				out .= ( ObjGetCapacity([k], 1) ? %this_fn%(k) : q . k . q ) ;// key
			.  ( indent ? ": " : ":" ) ; token + padding
			out .= %this_fn%(v, indent, lvl) ; value
			.  ( indent ? ",`n" . indt : "," ) ; token + indent
		}

		if (out != "")
		{
			out := Trim(out, ",`n" . indent)
			if (indent != "")
				out := "`n" . indt . out . "`n" . SubStr(indt, StrLen(indent)+1)
		}

		return is_array ? "[" . out . "]" : "{" . out . "}"
	}

	; Number
	else if (ObjGetCapacity([obj], 1) == "")
		return obj

	; String (null -> not supported by AHK)
	if (obj != "")
	{
		obj := StrReplace(obj,  "\",    "\\")
		, obj := StrReplace(obj,  "/",    "\/")
		, obj := StrReplace(obj,    q, "\" . q)
		, obj := StrReplace(obj, "`b",    "\b")
		, obj := StrReplace(obj, "`f",    "\f")
		, obj := StrReplace(obj, "`n",    "\n")
		, obj := StrReplace(obj, "`r",    "\r")
		, obj := StrReplace(obj, "`t",    "\t")

		static needle := (A_AhkVersion<"2" ? "O)" : "") . "[^\x20-\x7e]"
		while RegExMatch(obj, needle, m)
			obj := StrReplace(obj, m[0], Format("\u{:04X}", Ord(m[0])))
	}

	return q . obj . q
}

Jxon_True()
{
	static obj := {}
	return obj
}

Jxon_False()
{
	static obj := {}
	return obj
}

}
`

Suddenly stop running script

Hi good day, I have used Chrome.ahk file to make a page refresh in google chrome

The objective is to click a button when there is changes
Example when Box Orange turns to White press it
if not loop in a refresh

But the codes seems to stop after running 5 to 10 minutes in the 244 line and 245 line of Chome.ahk

Do help me thanks

New Chrome User Profile issue.

The error occurs when make new chrome user profile.

I think, some patch makes "--user-data-dir=" argument get only abstract folder path.

so, write correct path or attach previous script path. may be the solve.

Chrome.ahk Line 91
Like this.
Run, % this.CliEscape(ChromePath) . " --remote-debugging-port=" this.DebugPort . (ProfilePath ? " --user-data-dir=" . this.CliEscape(**A_ScriptDir . "\" .** ProfilePath) : "") . (Flags ? " " Flags : "") . URLString ,,, OutputVarPID

if do that, it can solve some other issues I think.

WebSocket Error

Hello,

I have been using the latest Chrome.ahk for some personal purposes for a long time now, suddenly when I wanted to run a script yesterday, once it tried to execute the GetPage() function, I got a Script error with "Websocket Error!"
I did not change anything in my script, it opens a webpage in Chrome started in debug mode (port 9222 is used), and I have the same issue in my other scripts just with different URLs opened.
Could someone please help me, or advise how to solve this problem?

image

Thank you,
Gabor

The given ProfilePath does not exist

image

I have added to the top of AutoHotkey.ahk:
#Include C:\Users\Larry\Dropbox_bu\ahk\Chrome.ahk-master/Chrome.ahk

I have Chrome running in debug mode.

I have extensive experience with automation using Selenium. Looking to automate a web-trading platform.

Thanks for your help!!

WaitForLoad for a limited period of time?

When the server is under heavy load its response took about two minutes. WaitForLoad function takes too long for application. How can wait for about 20 seconds and throw exception? I tried this;
WaitForLoadForXSeconds(DesiredState:="complete", Interval:=100, forXSeconds:= 10) { forXSeconds := forXSeconds * 1000 donguBaslamaAni := A_TickCount while (this.Evaluate("document.readyState").value != DesiredState) { Sleep, Interval if ((A_TickCount - donguBaslamaAni) > 300) { Throw Exception ("Sayfa yanıt vermiyor.") Return } } }

But didnt worked.

EventCallbacks.ahk > Could not retrieve page 1!

The event callbacks demo has a problem for me. it exits with "Could not retrieve page 1"

chrome.ahk 1.2
autohotkey Version 1.1.30.03
chrome Version 74.0.3729.131 (Official Build) (64-bit)

Stuck at sleep loop

Sometimes, the return of this.ws.Send on line 233 is so fast that that execution gets line 320 before line 241.
ID not exists on responses list, so it get stucked on 242 loop.

232 ID := this.ID += 1
233 this.ws.Send(Chrome.Jxon_Dump({"id": ID
234 , "params": Params ? Params : {}
235 , "method": DomainAndMethod}))
236
237 if !WaitForResponse
238 return
239
240 ; Wait for the response
241 this.responses[ID] := False
242 while !this.responses[ID]
243 Sleep, 50

315 ; Run the callback routine
316 fnCallback := this.fnCallback
317 if (newData := %fnCallback%(data))
318 data := newData
319
320 if this.responses.HasKey(data.ID)
321 this.responses[data.ID] := data

The solution is to transfer "this.responses[ID] := False" to before "this.ws.Send"

232 ID := this.ID += 1
233 if WaitForResponse
234 this.responses[ID] := False
235 this.ws.Send(Chrome.Jxon_Dump({"id": ID
236 , "params": Params ? Params : {}
237 , "method": DomainAndMethod}))
238
239 if !WaitForResponse
240 return
241 ; Wait for the response
242 while !this.responses[ID]
243 Sleep, 50

Chrome Automation breaks after Chrome v109.0.5414.75

I upgraded some computers from Google Chrome 109.0.5414.75 to 111.0.5563.147 and 112.0.5615.138 and Chrome Automation breaks and just sits at the About Blank page. I also tested v113 and that no longer works.

There are some posts that might be related:

https://stackoverflow.com/questions/76058764/data-in-the-address-bar-while-using-chromedriver-112-0-5615-49
The problem is being investigated: https://bugs.chromium.org/p/chromedriver/issues/detail?id=4357
For those who need an urgent workaround please refer to https://bugs.chromium.org/p/chromedriver/issues/detail?id=4357#c1

Hangs on while !this.responses[ID]

Have a script that runs every 15 minutes. Stops working after a while and saw that it's hung on
; Wait for the response
this.responses[ID] := False
while !this.responses[ID]
Sleep, 50

Publishing Chrome.ahk as an ahkpm package?

Hi @G33kDude. I'm wondering if you'd be interested in publishing Chrome.ahk as an ahkpm package?

ahkpm is a package manager built specifically for AutoHotkey libraries.

I've written a guide on how to publish ahkpm packages and it likely wouldn't take much time at all.

The main thing I noticed that you might need to change is to start tagging new versions with the appropriate semantic version number. e.g. git tag 1.3.18.

I'd be happy to open up a PR adding the ahkpm.json file if you're interested.

include file errors

Downloading the zip file, it does not include the libs; so, I downloaded the libs manually.
It is necessary to include some lines like #Include %A_MyDocuments%/autohotkey/chrome.ahk for some users, but that's a separate matter.
If this is done properly, and chrome.ahk is included, I am assuming that json.ahk and jxon.ahk come from your other repo because clicking on the folder in the code links to your autohotkeyjson repo; so, I downloaded those and put them in the %A_MyDocuments%/autohotkey/lib directory.
Resulting error: call to nonexistent function in include file jxon.ahk specifically: jxon_dump(k) line 165-184 (see image).
image

'Failed to parse JSON' when running script

I'm trying to upgrade from v1.2 to v1.3. When I include Chrome.ahk in a script I get this error:

Failed to parse JSON (-1,0)
Specifically: Unexpected character at position 0: "

v1.2 was working as expected 🤷
Any help is much appreciated!

How do I send clicks to all my tabs?

#NoEnv
#Persistent
#SingleInstance
SetBatchLines, -1
SetTitleMatchMode 2
#Include Chrome.ahk

url:= "https://gota.io/web/"

FileCreateDir, ChromeProfile
ChromeInst := new Chrome
winwait, Chrome

if !(Page1 := ChromeInst.GetPage())
{
MsgBox, Could not retrieve page!
ChromeInst.Kill()
}
else
Page1.WaitForLoad()

Page1.Call("Page.navigate", {"url": url})
Page1.WaitForLoad()

~$Shift::
Page1.call("Input.dispatchKeyEvent" , {type: "keyDown", windowsVirtualKeyCode: 16})
Page1.call("Input.dispatchKeyEvent" , {type: "keyUp", windowsVirtualKeyCode: 16})
return

This is the code I used and I'm wondering how do I send multiple clicks through specific page like this. When you press shift it sends it to that specific tab . Even better if the page is active even if its not seened or clicked on and it follows the mouse pointer and reacts to it

Wrong path chrome.lnk in Chrome.ahk_v1.2(Release)

script.ahk line 77

Wrong path
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\App Pahs\chrome.exe

correct path
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\App Paths\chrome.exe

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.