HISE Logo Forum
    • Categories
    • Register
    • Login

    HEADS UP: Server.downloadFile() issue on macOS Sequoia - and a solution

    Scheduled Pinned Locked Moved Bug Reports
    30 Posts 5 Posters 6.0k Views
    Loading More Posts
    • Oldest to Newest
    • Newest to Oldest
    • Most Votes
    Reply
    • Reply as topic
    Log in to reply
    This topic has been deleted. Only users with topic management privileges can see it.
    • T
      tomekslesicki
      last edited by tomekslesicki

      I've had three customers report that the download fails immediately on Sequoia. Apparently, GET / POST calls work fine, but there was an issue with downloading stuff from my Cloudflare server. The issue was only present on macOS Sequoia, but not on every computer - seems random in a typically Apple way.

      I call different servers throughout the installation process to get the license key validated (this works fine), and to get the files (which caused the issue). My logic starts with

      Server.setBaseURL("https://");
      

      And then the rest happens in the expected way.

      I built a Debug version and checked the log. It turns out that the downloadFile callback periodically triggers a call to http://google.com

      Now, why is it calling Google, I have no idea. But the problem is about the http part. Apparently, it triggers some safety nonsense in macOS that's blocking the entire download traffic! Https calls work fine, but pass something without 's' into the standalone app and you're up for trouble.

      All this is happening on a standalone app build, I have not tested it with plugin exports because this is not the flow I offer with this app.

      The solution is: start building the standalone app, and quit the process right after the XCode files are created. Then, open the XCode project in the Build folder, choose the Signing and capabilities tab, then click on the little + Capability text that doesn't even look like it can be clicked (awesome UX work here, Apple) and choose App Transport Security Exception. Add your main domains there, and google.com. Build and you will be fine.

      d.healeyD 2 Replies Last reply Reply Quote 0
      • d.healeyD
        d.healey @tomekslesicki
        last edited by

        @tomekslesicki isn't that triggered by Server.isOnline()? I don't recall seeing it for the download function.

        Free HISE Bootcamp Full Course for beginners.
        YouTube Channel - Public HISE tutorials
        My Patreon - HISE tutorials

        T 1 Reply Last reply Reply Quote 0
        • T
          tomekslesicki @d.healey
          last edited by

          @d-healey maybe, but I added that check to the download callback as I was troubleshooting. Downloading was not working on Sequoia before that, and only adding the exceptions in XCode solves this. Maybe the https:// base is not passed to the download callback properly, I don't know.

          1 Reply Last reply Reply Quote 0
          • Dan KorneffD
            Dan Korneff
            last edited by

            I think I've been here before:
            https://forum.hise.audio/topic/8742/how-to-debug-server-isonline

            The functions check google and amazon to see if they are reachable, but some macos systems have issues with http calls.
            I've updated the source to point to my https domain. I reasoned it wasn't important if they are online but unable to reach my server.

            Dan Korneff - Producer / Mixer / Audio Nerd

            1 Reply Last reply Reply Quote 0
            • d.healeyD
              d.healey
              last edited by

              Yeah it's the crude method HISE uses for checking if there is an active internet connection. It's also a big issue for users in China because the great firewall doesn't always allow the requests through.

              I made a feature request back in January to add proper network detection in HISE by getting the status from the operating system. Someone else has already done the work to implement it as a JUCE module, it just needs integrating into HISE.

              https://forum.hise.audio/topic/11515/feature-request-network-connectivity-checker?_=1745953247545

              Free HISE Bootcamp Full Course for beginners.
              YouTube Channel - Public HISE tutorials
              My Patreon - HISE tutorials

              T A 2 Replies Last reply Reply Quote 2
              • T
                tomekslesicki @d.healey
                last edited by

                @d-healey that would be cool, but actually just changing the check from http to https would make it more mac-friendly. @Christoph-Hart

                There’s also a question of whether to base url is passing the https part properly, but that’s too complicated for me to check. As I mentioned, the download callback wasn’t working for some Sequoia users way before I added the online check, so there’s definitely something going on.

                Anyway - changing the build settings in xcode is a way around it.

                d.healeyD 1 Reply Last reply Reply Quote 0
                • d.healeyD
                  d.healey @tomekslesicki
                  last edited by

                  @tomekslesicki This reminds me actually. I added this to my custom plists for MacOS builds ages ago which bypasses the http/https security restriction

                  <plist><dict><key>NSAppTransportSecurity</key><dict><key>NSAllowsArbitraryLoads</key><true/></dict></dict></plist>
                  

                  Free HISE Bootcamp Full Course for beginners.
                  YouTube Channel - Public HISE tutorials
                  My Patreon - HISE tutorials

                  Christoph HartC 1 Reply Last reply Reply Quote 0
                  • Christoph HartC
                    Christoph Hart @d.healey
                    last edited by

                    but actually just changing the check from http to https would make it more mac-friendly.

                    Done. I also agree, the other suggestion from Dave is overkill, plus it's a GPL licensed module so I would have to "get loosely inspired" by it to not mess up the licensing situation.

                    d.healeyD T 2 Replies Last reply Reply Quote 2
                    • d.healeyD
                      d.healey @Christoph Hart
                      last edited by

                      @Christoph-Hart said in HEADS UP: Server.downloadFile() issue on macOS Sequoia - and a solution:

                      the other suggestion from Dave is overkill

                      It would solve the issue for Chinese users getting timeouts. It also would allow us to differentiate between mobile data and wifi/ethernet so that the user can choose to only download on unmetered networks.

                      But yeah I forgot about the license thing.

                      Free HISE Bootcamp Full Course for beginners.
                      YouTube Channel - Public HISE tutorials
                      My Patreon - HISE tutorials

                      1 Reply Last reply Reply Quote 0
                      • A
                        aaronventure @d.healey
                        last edited by

                        @d-healey said in HEADS UP: Server.downloadFile() issue on macOS Sequoia - and a solution:

                        I made a feature request back in January to add proper network detection in HISE by getting the status from the operating system. Someone else has already done the work to implement it as a JUCE module, it just needs integrating into HISE.

                        Background Task

                        ping -c 1 1.1.1.1 >/dev/null
                        echo $?
                        

                        If it returns 0, you're online.

                        d.healeyD 1 Reply Last reply Reply Quote 0
                        • d.healeyD
                          d.healey @aaronventure
                          last edited by

                          @aaronventure How is that different from HISE pinging google's server?

                          Free HISE Bootcamp Full Course for beginners.
                          YouTube Channel - Public HISE tutorials
                          My Patreon - HISE tutorials

                          A 1 Reply Last reply Reply Quote 0
                          • A
                            aaronventure @d.healey
                            last edited by

                            @d-healey It's not really, but how would you check if you're online if not by connecting to another server? 1.1.1.1 is a CloudFlare IP which uses AnyCast, so it automatically connects to the nearest operational server, vastly reducing a chance that you're making a call when the server appears to be down.

                            I Imagine that's why Google was picked: the chance of downtime being so rare.

                            This is supposed to be even more robust, with the benefit of the latency being almost non-existent in most countries where people buy audio plugins.

                            d.healeyD 1 Reply Last reply Reply Quote 0
                            • d.healeyD
                              d.healey @aaronventure
                              last edited by

                              @aaronventure said in HEADS UP: Server.downloadFile() issue on macOS Sequoia - and a solution:

                              but how would you check if you're online if not by connecting to another server

                              Ask the OS if the system is currently connected to a network.

                              Free HISE Bootcamp Full Course for beginners.
                              YouTube Channel - Public HISE tutorials
                              My Patreon - HISE tutorials

                              A 1 Reply Last reply Reply Quote 0
                              • A
                                aaronventure @d.healey
                                last edited by

                                @d-healey that's true, but how do you think the OS does the check? I would bet it's something like this. Maybe macOS is calling www.apple.com and Windows is calling microsoft.com, as that also confirms DNS resolution and HTTPS working.

                                d.healeyD 1 Reply Last reply Reply Quote 0
                                • d.healeyD
                                  d.healey @aaronventure
                                  last edited by d.healey

                                  @aaronventure How it does it doesn't matter, the problem is that every time you ping from in HISE and you don't reach the server you have to wait for the timeout time. I had a user in China who kept getting hit by 20 second delays every time I was calling Server.isOnline() - which I was doing in a few places in my login/download chain.

                                  If I could just ask the OS are you online, I get an instant yes/no. The checking, however it happens has already been done - it's probably reported by the router or access point but I'm not sure.

                                  My solution at the moment is I call Server.isOnline() once when the plugin loads and store the result. However if the user goes offline during the session then this will no longer be true, but I can handle that more easily than the 20 second delay.

                                  It's also possible to reduce the server timeout time, but you can only take it so low before you start giving users on slow connections the wrong result.

                                  This method would also allow for a broadcaster that reports when the system goes online/offline - I think I said this in my feature request post too.

                                  Free HISE Bootcamp Full Course for beginners.
                                  YouTube Channel - Public HISE tutorials
                                  My Patreon - HISE tutorials

                                  A 1 Reply Last reply Reply Quote 0
                                  • A
                                    aaronventure @d.healey
                                    last edited by aaronventure

                                    @d-healey Ah I see your point. Also I figured if you're making a channel strip plugin or something and a user just happens to have 100 instances in a project, it could get interesting on load.

                                    I did some more digging. On macOS, it appears that

                                    /usr/sbin/scutil -r $? 
                                    

                                    will check your internet connection without sending any packets. It returns "Reachable" or "Not Reachable".

                                    If you call

                                    /usr/sbin/scutil -r -W $? 
                                    

                                    the task will run until terminated and will respond with Reachable or Not Reachable whenever the network status changes (I just tested it), which is pretty much like a broadcaster.

                                    @Christoph-Hart you can implement this instead of having to clean room that module. I'm sure Windows has a similar mechanism.

                                    d.healeyD 1 Reply Last reply Reply Quote 1
                                    • d.healeyD
                                      d.healey @aaronventure
                                      last edited by

                                      @aaronventure said in HEADS UP: Server.downloadFile() issue on macOS Sequoia - and a solution:

                                      I did some more digging. On macOS, it appears that

                                      Oh that's nice and simple.

                                      I bet ChatGPT knows the magic spell for Windows too, I'll ask it tomorrow if you don't get there first :)

                                      Free HISE Bootcamp Full Course for beginners.
                                      YouTube Channel - Public HISE tutorials
                                      My Patreon - HISE tutorials

                                      A 1 Reply Last reply Reply Quote 0
                                      • A
                                        aaronventure @d.healey
                                        last edited by aaronventure

                                        @d-healey said in HEADS UP: Server.downloadFile() issue on macOS Sequoia - and a solution:

                                        I bet ChatGPT knows the magic spell for Windows too, I'll ask it tomorrow if you don't get there first :)

                                        Hah, sure, but I'm too lazy to go test it right now.

                                        So here's the reply :

                                        One-shot check (returns Boolean, no packets sent)

                                        $nlm = [Activator]::CreateInstance([type]'Microsoft.Windows.NetworkList.NetworkListManager')
                                        $nlm.IsConnectedToInternet   # $true = online, $false = offline
                                        

                                        Continuous monitoring (no polling)

                                        $nlm  = [Activator]::CreateInstance([type]'Microsoft.Windows.NetworkList.NetworkListManager')
                                        
                                        Register-ObjectEvent -InputObject $nlm `
                                                             -EventName   'ConnectivityChanged' `
                                                             -SourceIdentifier NLM `
                                                             -Action { 
                                                                 if ($Event.SourceEventArgs.IsConnectedToInternet) {
                                                                     Write-Host 'online'
                                                                 } else {
                                                                     Write-Host 'offline'
                                                                 }
                                                             }
                                        
                                        # keep the session alive
                                        while ($true) { Start-Sleep -Seconds 3600 }
                                        

                                        HISE-native integration of this might be better than manually launching Background Task as then it can just start a process and every instance of the plugin can read from the process instead of having every plugin instance start its own monitoring process through BackgroundTask (as I don't think BackgroundTask can connect to an existing one?)

                                        1 Reply Last reply Reply Quote 1
                                        • T
                                          tomekslesicki @Christoph Hart
                                          last edited by

                                          @Christoph-Hart is this a part of the latest develop branch commint?

                                          T 1 Reply Last reply Reply Quote 0
                                          • T
                                            tomekslesicki @tomekslesicki
                                            last edited by tomekslesicki

                                            Ok, a quick update on the subject - the XCode transport solution only works for some users. For others, the download callback returns this.data.finished = true && this.data.sucess = false even before anything can be downloaded. This is happening both on Windows now, and macOS, for about 0.5-1% of users, for everyone else things are working fine.

                                            Here's a very simplified overview of my download logic:

                                            At the beginning of the scrip, I call:

                                            Server.setBaseURL("https://");
                                            

                                            which is used for POST calls, and then this download callback, with the url aprt being the rest of the url, after the https part.

                                            inline function startDownload()
                                            {
                                            	// Download samples and content
                                            	for (i = 0; i < plugincontent.length; i++)
                                            	{
                                            		local files = workingFolder.getChildFile(plugincontent[i]);
                                            		downloads[i] = Server.downloadFile(url + plugincontent[i], p, files, downloadCallback);
                                            	}	
                                            
                                            	// There's just one zip file and it's always the first in the plugincontent array
                                            	zips[0] = workingFolder.getChildFile(plugincontent[0].replace(subfoldernamefull).replace(subfoldernamedemo));
                                            	
                                            	// Show update progress panel
                                            }
                                            
                                            
                                            function downloadCallback(obj)
                                            {
                                            	if (this.data.finished != true)
                                            	{
                                            		// Show and update download progress
                                            	}
                                            
                                            	// Connection lost
                                            	if (this.data.finished && this.data.success == false)
                                            	{
                                            		// Show error message
                                            	}
                                            
                                            	if (this.data.finished && this.data.success == true)
                                            	{
                                            		// Update downloaded files count and extract zips
                                            	}
                                            }
                                            

                                            As said, this works 100% for POST calls for all users. The problem is for 1% of users with the download callback. I remember having a similar issue with GET when I started this project.

                                            Save me, please!

                                            T 1 Reply Last reply Reply Quote 0
                                            • First post
                                              Last post

                                            19

                                            Online

                                            2.0k

                                            Users

                                            12.7k

                                            Topics

                                            110.5k

                                            Posts