Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Infer server url on server select screen #1357

Merged
merged 27 commits into from
Oct 31, 2023
Merged
Show file tree
Hide file tree
Changes from 24 commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
a93b175
proof of concept - infer server url proto and port
sevenrats Sep 7, 2023
0d4ed65
dont fail if server not found
sevenrats Sep 7, 2023
0b7b2d5
dont fail if server not found
sevenrats Sep 7, 2023
93d683a
address comments, update debug statements
sevenrats Sep 7, 2023
01bfb31
check for and remove trailing /
sevenrats Sep 10, 2023
481284a
infer url deterministically
sevenrats Sep 10, 2023
dfb4dcb
make we found jellyfin and not emby
sevenrats Sep 10, 2023
0f41e65
document new function
sevenrats Sep 10, 2023
f29cddf
serverinfo is a string not an object
sevenrats Sep 10, 2023
bffaf06
try to find string strip bug
sevenrats Sep 10, 2023
6b714a8
trim the input not the path
sevenrats Sep 10, 2023
c0974db
restore new function doc
sevenrats Sep 10, 2023
f75c992
don't write ports if we dont need to
sevenrats Sep 11, 2023
ec7ad49
remove untrue comment
sevenrats Sep 11, 2023
30bd09e
address comment
sevenrats Sep 11, 2023
14899f6
address comments
sevenrats Sep 11, 2023
d7c756e
Merge branch 'unstable' into server_url
cewert Sep 11, 2023
fbfe130
clean up debug prints
sevenrats Sep 11, 2023
02b4c9f
fix non descriptive argument
sevenrats Sep 13, 2023
babc668
stop snake casing
sevenrats Sep 13, 2023
6b16ed9
function definition
sevenrats Sep 14, 2023
d4483a2
Update source/utils/misc.brs
sevenrats Sep 14, 2023
8f8650f
Merge branch 'unstable' into server_url
sevenrats Oct 15, 2023
1e03533
dont crash on bad input
sevenrats Oct 15, 2023
c268a53
Merge branch 'server_url' into unstable
sevenrats Oct 31, 2023
7e680a9
Merge pull request #12 from sevenrats/unstable
sevenrats Oct 31, 2023
a3af52c
apply suggestions
sevenrats Oct 31, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 8 additions & 2 deletions source/ShowScenes.brs
Original file line number Diff line number Diff line change
Expand Up @@ -281,13 +281,18 @@ function CreateServerGroup()
dialog.title = tr("Connecting to Server")
m.scene.dialog = dialog

serverUrl = standardize_jellyfin_url(screen.serverUrl)
set_setting("server", serverUrl)
serverUrl = inferServerUrl(screen.serverUrl)
sevenrats marked this conversation as resolved.
Show resolved Hide resolved

isConnected = session.server.UpdateURL(serverUrl)
serverInfoResult = invalid
if isConnected
serverInfoResult = ServerInfo()
'If this is a different server from what we know, reset username/password setting
if m.global.session.server.url <> serverUrl
set_setting("username", "")
set_setting("password", "")
end if
set_setting("server", serverUrl)
end if
dialog.close = true

Expand All @@ -298,6 +303,7 @@ function CreateServerGroup()
screen.errorMessage = tr("Server not found, is it online?")
SignOut(false)
else

if isValid(serverInfoResult.Error) and serverInfoResult.Error
' If server redirected received, update the URL
if isValid(serverInfoResult.UpdatedUrl)
Expand Down
128 changes: 106 additions & 22 deletions source/utils/misc.brs
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import "pkg:/source/utils/config.brs"

function isNodeEvent(msg, field as string) as boolean
return type(msg) = "roSGNodeEvent" and msg.getField() = field
end function
Expand Down Expand Up @@ -162,31 +164,103 @@ function option_dialog(options, message = "", defaultSelection = 0) as integer
return show_dialog(message, options, defaultSelection)
end function

'
' Take a jellyfin hostname and ensure it's a full url.
' prepend http or https and append default ports, and remove excess slashes
'
function standardize_jellyfin_url(url as string)
'Append default ports
maxSlashes = 0
if left(url, 8) = "https://" or left(url, 7) = "http://"
maxSlashes = 2
' take an incomplete url string and use it to make educated guesses about
' the complete url. then tests these guesses to see if it can find a jf server
' returns the url of the server it found, or an empty string
function inferServerUrl(url as string) as string
' if this server is already stored, just use the value directly
' the server had to get resolved in the first place to get into the registry
saved = get_setting("saved_servers")
if isValid(saved)
savedServers = ParseJson(saved)
if isValid(savedServers.lookup(url)) then return url
end if
'Check to make sure entry has no extra slashes before adding default ports.
if Instr(0, url, "/") = maxSlashes
if url.len() > 5 and mid(url, url.len() - 4, 1) <> ":" and mid(url, url.len() - 5, 1) <> ":"
if left(url, 5) = "https"
url = url + ":8920"
else
url = url + ":8096"
end if

port = CreateObject("roMessagePort")
hosts = CreateObject("roAssociativeArray")
reqs = []
candidates = urlCandidates(url)
for each endpoint in candidates
req = CreateObject("roUrlTransfer")
reqs.push(req) ' keep in scope outside of loop, else -10001
req.seturl(endpoint + "/system/info/public")
req.setMessagePort(port)
hosts.addreplace(req.getidentity().ToStr(), endpoint)
if endpoint.Left(8) = "https://"
req.setCertificatesFile("common:/certs/ca-bundle.crt")
end if
req.AsyncGetToString()
end for
handled = 0
timeout = CreateObject("roTimespan")
if hosts.count() > 0
while timeout.totalseconds() < 15
resp = wait(0, port)
if type(resp) = "roUrlEvent"
' TODO
' if response code is a 300 redirect then we should return the redirect url
' Make sure this happens or make it happen
if resp.GetResponseCode() = 200 and isJellyfinServer(resp.GetString())
selectedUrl = hosts.lookup(resp.GetSourceIdentity().ToStr())
print "Successfully inferred server URL: " selectedUrl
return selectedUrl
end if
end if
handled += 1
if handled = reqs.count()
print "inferServerUrl in utils/misc.brs failed to find a server from the string " url " but did not timeout."
return ""
end if
end while
print "inferServerUrl in utils/misc.brs failed to find a server from the string " url " because it timed out."
end if
'Append http:// to server
if left(url, 4) <> "http"
url = "http://" + url
return ""
end function

' this is the "educated guess" logic for inferServerUrl that generates a list of complete url's as candidates
' for the tests in inferServerUrl. takes an incomplete url as an arg and returns a list of extrapolated
' full urls.
function urlCandidates(input as string)
sevenrats marked this conversation as resolved.
Show resolved Hide resolved
if input.endswith("/") then input = input.Left(len(input) - 1)
url = parseUrl(input)
if url[1] = invalid
' a proto wasn't declared
url = parseUrl("none://" + input)
end if
return url
' if the proto is still invalid then the string is not valid
if url[1] = invalid then return []
proto = url[1]
host = url[2]
port = url[3]
path = url[4]
protoCandidates = []
supportedProtos = ["http:", "https:"] ' appending colons because the regex does
if proto = "none:" ' the user did not declare a protocol
' try every supported proto
for each supportedProto in supportedProtos
protoCandidates.push(supportedProto + "//" + host)
sevenrats marked this conversation as resolved.
Show resolved Hide resolved
end for
else
protoCandidates.push(proto + "//" + host) ' but still allow arbitrary protocols if they are declared
end if
finalCandidates = []
if isValid(port) and port <> "" ' if the port is defined just use that
for each candidate in protoCandidates
finalCandidates.push(candidate + port + path)
end for
else ' the port wasnt declared so use default jellyfin and proto ports
for each candidate in protoCandidates:
' proto default
finalCandidates.push(candidate + path)
' jellyfin defaults
if candidate.startswith("https")
finalCandidates.push(candidate + ":8920" + path)
else if candidate.startswith("http")
finalCandidates.push(candidate + ":8096" + path)
end if
end for
end if
return finalCandidates
end function

sub setFieldTextValue(field, value)
Expand Down Expand Up @@ -228,7 +302,7 @@ function isValidAndNotEmpty(input as dynamic) as boolean
end if
end function

' Returns an array from a url - [ url, proto, host, port, subdir/params ]
' Returns an array from a url = [ url, proto, host, port, subdir+params ]
' If port or subdir are not found, an empty string will be added to the array
' Proto must be declared or array will be empty
function parseUrl(url as string) as object
Expand Down Expand Up @@ -383,6 +457,16 @@ sub stopLoadingSpinner()
end if
end sub

' accepts the raw json string of /system/info/public and returns
' a boolean indicating if ProductName is "Jellyfin Server"
function isJellyfinServer(systemInfo as object) as boolean
d = ParseJson(systemInfo)
sevenrats marked this conversation as resolved.
Show resolved Hide resolved
if isValid(d) and isValid(d.ProductName)
return d.ProductName = "Jellyfin Server"
sevenrats marked this conversation as resolved.
Show resolved Hide resolved
end if
return False
sevenrats marked this conversation as resolved.
Show resolved Hide resolved
end function

' Check if a specific value is inside of an array
function arrayHasValue(arr as object, value as dynamic) as boolean
for each entry in arr
Expand Down