A free and open source relay server for NVDA Remote
The NVDA Remote Relay server is a multiplatform, free and open source server for NVDA Remote. It has the same functionality as the official NVDA Remote server (nvdaremote.com, allinaccess.com), but you can install and use it anywhere!
Do you want to control several computers located inside your home from outside? You can't forward the tcp port 6837 for all of them, so this server is for you.
This server is multiplatform, so you can install it under Windows, Mac and many Linux distributions, including Arch, Debian and Centos. If you have installed python 2.x or 3.x on your system, probably you will be able to run this server.
Building NVDA Remote Relay server is very easy. On most platforms, you only need to run the build.sh script corresponding to your platform, and install the generated package after that. Unless otherwise specified, you should clone this repository and run all commands as root.
- Navigate to the debian directory inside this repo (choose between standard Debian-based distributions, Debian without systemd (ideal for OpenVZ, Docker and other containers) and Debian legacy version for Python 2). If you run Debian versions later than Debian 9, or Ubuntu 15.04 and later, the Debian package is what you need.
- Ensure that the build.sh script can be executed:
chmod +x build.sh
- Run the script:
./build.sh
- Install the package:
dpkg -i NVDARemoteServer.deb
To uninstall it, run dpkg --purge nvda-remote-server
To uninstall it keeping the configuration files, run dpkg --remove nvda-remote-server
To build the rpm version, the rpmdevtools package is required. Install it using your package manager.
- Navigate to the RHEL9 or rpm-generic directory inside this repo. Use Generic package for all RPM-based distros. Use RHEL9 package for Alma Linux 9, Rocky Linux 9 or similar. It will require specifically Python 3.12, which is more recent than the provided python3 package.
- Ensure that the build.sh script can be executed:
chmod +x build.sh
- Run the script:
./build.sh
. The final rpm will be located at~/rpmbuild/RPMS/noarch
directory. - Install the package using rpm:
rpm -U NVDARemoteServer.rpm
To uninstall it, run rpm -e NVDARemoteServer
- Navigate to the Mac Os directory inside this repo.
- Ensure that the build.sh script can be executed:
chmod +x build.sh
- Run the script:
sudo ./build.sh
- Install the generated package using Finder or the terminal. Remember to allow untrusted software installation in System preferences > Security and privacy. To install from the terminal, run the following command:
installer -pkg NVDARemoteServer.pkg -target /
To uninstall it, run NVDARemoteUninstall
Note: you must clone the repository and perform the following actions in a standard, non-privileged user account.
- Navigate to the Arch directory inside this repo.
- Ensure that the build.sh script can be executed:
chmod +x build.sh
- Run the script:
./build.sh
- Install the package with pacman:
sudo pacman -U NVDARemoteServer.pkg.tar.zst
To uninstall it, run: sudo pacman --remove NVDARemoteServer
- Navigate to the MSYS directory inside this repo.
- Repeat steps 3 and 4 from arch section.
- Navigate to the Cygwin directory from Cygwin Bash shell.
- Run install.sh:
./install.sh
To uninstall it, run NVDARemoteUninstall
You need one or multiple Python installations, depending on what you want to build. Install the x86 and x64 versions if you want to build the server for both architectures. Go to the Python downloads page and choose:
- Python 2.7.18 if you want to maximize compatibility with older Windows versions, including Windows xp. The server will also work on Windows 10.
- Python 3.x (3.13.1 or later) if you want to take advantage of all the Python performance and security improvements.
You need also Python for Windows Extensions, build 308 or later. Install this package running pip install pywin32
command.
Finally, you must install a packager in order to build the binary version. If you are building with Python 2.7, you can use Pyinstaller (install with pip install pyinstaller
), cx-freeze (install with pip install cx-freeze
), or py2exe 0.6.9. On Python 3, only pyinstaller and cx-freeze are supported.
Once the build environment is ready, open a command prompt and navigate to the root folder of this repository, then:
- If you want to use py2exe, run:
python setup_windows.py py2exe
- If you prefer pyinstaller, run:
pyinstaller pyinstaller.spec
. - If you prefer cx-freeze, run:
python setup_windows_cxfreeze.py build
.
Note: if you build with several Python versions on the same machine, don't add python to the path environment variable during installation. Instead, use the Python launcher included with Python 3, or specify the full path to your python executable. For example: C:\\python27x86\\python setup_windows.py py2exe
.
The binaries will be placed in the dist folder. For cx-freeze builds, the binaries will be placed in the build folder. If you build for multiple architectures, using multiple Python installations and packagers, remember saving the dist folder contents (in case of cx-freeze, the build folder) to another location before building again. To avoid conflicts, remove the build and dist directories after saving their contents.
The server is almost portable, there is no installation required. If you install the Windows service, remember uninstalling it before moving the server to another location or removing it.
You need Docker installed on a Linux system. Navigate to the docker directory and run:
docker build -t nvda-remote-server .
Change or add more tags if you plan to push the image to a Docker registry. For example: docker build -t jmdaweb/nvda-remote-server:latest -t jmdaweb/nvda-remote-server:2.4 .
If you want to build a multiplatform image, provided that you meet the prerrequisites described here, you can use a command like the following: docker buildx build -t jmdaweb/nvda-remote-server:latest -t jmdaweb/nvda-remote-server:2.4 --platform linux/amd64 --platform linux/386 --platform linux/arm64/v8 --platform linux/arm/v7 --platform linux/arm/v5 --platform linux/mips64le --platform linux/ppc64le --platform linux/s390x --push .
Before you begin, check that the tcp port you have chosen for the server (by default 6837) is allowed through your firewall.
On Unix platforms, including Mac Os, there is a script located in /usr/bin called NVDARemoteServer. You can run this script without parameters to get a short help message.
If you want to start the server in debugging mode (useful to see activity and errors) run:
sudo NVDARemoteServer debug
On most platforms, you can stop the server by pressing ctrl+c when it is running in this mode.
To start the server, run:
sudo NVDARemoteServer start
To stop it, run:
sudo NVDARemoteServer stop
Run sudo NVDARemoteServer restart
to restart the server.
If the server freezes, run sudo NVDARemoteServer kill
to kill the process.
On Arch, RHEL and Debian based distributions, the NVDA Remote Relay server is also installed as a service, so you can configure it to run at system startup, and manage it with the systemctl utility. You can run NVDARemoteServer status
to see if the service is running.
If you want to configure the service to run at system startup, run sudo NVDARemoteServer enable
. Run sudo NVDARemoteServer disable
to configure the service to start manually.
If you want to install the server as a system service on Windows, run service_manager.cmd and choose the right options on the displayed menu.
Run debug.cmd to start the server in debugging mode.
If you want to run the server inside a Docker container, firstly create a data directory. This will store the configuration file and extra certificates if needed. Change ownership for this directory to uid and gid 991 with the chown command. Then, use a command similar to the following:
docker run -d -v ./data:/data -p 6837:6837 --rm jmdaweb/nvda-remote-server:latest
The command above destroys the container once it is stopped. If you want a permanent container which you can start and stop without creation and removal, run:
docker run -d -v ./data:/data -p 6837:6837 --name my_nvda_remote jmdaweb/nvda-remote-server:latest
Or, if you want to create it and run later:
docker create -v ./data:/data -p 6837:6837 --name my_nvda_remote jmdaweb/nvda-remote-server:latest
You can change jmdaweb/nvda-remote-server:latest to your custom image name if you have built it. The --name argument specifies the container name. If you want to use a diferent port, change the first part after -p. The second must be always 6837.
Once created, you can start your container with the following command:
docker start my_nvda_remote
To stop it, run this command:
docker stop my_nvda_remote
Run this command to remove the container. It must be stopped first:
docker rm my_nvda_remote
A docker-compose.yml file is also available. Navigate to the root directory of this repository, and run docker compose up -d
to start the service and docker compose down
to stop it. Remember creating the data directory and assigning the required ownership!
Do you run the server in production mode and it suddenly is stopped? Now you can look at the following files:
On most Linux distributions and Mac Os x: go to /var/log/NVDARemoteServer.log
On Arch Linux: go to /run/NVDARemoteServer/NVDARemoteServer.log
On Windows: NVDARemoteServer.log is located inside the program folder.
On a Docker container: run docker logs containername
. For example: docker logs my_nvda_remote
Read the Docker documentation to see all available commands.
The server includes a default self-signed certificate to encrypt connections. This is a huge security risk.
It is strongly recommended that you create your own certificate before starting the server for the first time. You can do it by running the NVDARemoteCertificate script on almost all platforms:
sudo NVDARemoteCertificate
The script takes no arguments. Follow the on-screen instructions to complete the process.
The script will create a secp384r1 ecdsa private key and a certificate, and combine them in a single server.pem file. Once finished, it is recomended restarting the server if it's running.
On Windows, a pre-built OpenSSL version is included in the server directory. You can run NVDARemoteCertificate.cmd to create a certificate.
If your server is running inside a Docker container, you can run a command similar to the following:
docker exec -t -i my_nvda_remote NVDARemoteCertificate
Starting with version 1.5, NVDARemoteServer includes a configuration file that you can modify to change some server settings. You must restart the server after modifying this file. The comments in the file will guide you when changing settings.
By default, the configuration file is located inside the program folder (Windows) or in /etc (all other distributions), and it is named NVDARemoteServer.conf
.
You can test your changes in debugging mode before modifying the configuration file. Although the changes in the configuration also are applied when you run the server in debug mode, you can pass some commandline parameters to perform tests. The following options are available:
--interface=ip
: listen only on the specified ip address. This setting doesn't affect IPV6 interfaces. In some platforms, this setting will not work if IPV6 socket binds to all interfaces.--interface6=ip
: listen only on the specified IPV6 address. This setting doesn't affect IPV4 interfaces.--port=port
: listen only on the specified tcp port.--port6=port
: listen on the specified port, but only for IPV6. By default, use the value specified in --port. Use this value if you want different ports for IPV4 and IPV6 sockets.--logfile=path
,--pidfile=path
: these parameters are available, but unuseful in debug mode. You can use them on init.d and systemd units, but it's not recommended. Use --configfile instead. If you change pidfile in the configuration file and use the server as a system daemon, update the pidfile variable in the service units for the status command to work properly.--loglevel=n
, where n is a number between 0 (almost quiet) and 4 (very verbose).--keyfile=path
: path to the private key used for ssl connections. Use only if the cert file explained below does not contain a private key.--certfile=path
: path to the private key and certificate used for ssl connections. They must be in the same file. If the pem file only contains certificates, use the keyfile option explained above.--motd=string
: specify the message of the day displayed to all clients when they join a channel. Enclose the message between quotes. A warning message will be appended to the provided string if loglevel is set to 4, or displayed alone if no message is given.--motd_force_display=integer
: display the message of the day even if it has not changed since last time the client joined a channel. 0 means do not force display, 1 means force display. This option is ignored when loglevel is set to 4 or above. In this case, the message is always displayed.--includeTracebacks=integer
: display Python tracebacks when exceptions are raised. 0 means do not display tracebacks, 1 means display tracebacks.--allowedMessageLength=integer
: defines maximum allowed length, in characters, for incoming client messages. 0 means no limit. Note that characters may have different lengths depending on the Python version and encoding used.--configfile=path
: read config file from path. All the previous options can be edited in the configuration file.
Note: the command line arguments override the supplied ones in the configuration file.
You can use your Let's Encrypt certificate with this server. However, the server runs by default with a non-privileged user, so it won't be able to read the required files from the default location. You can proceed in two ways:
- Copy privkey.pem and fullchain.pem to a readable location. Then, edit the configuration file and update the certfile and keyfile settings. You can follow this procedure also in Docker containers by copying certificate and private key to the data volume.
- Use the provided Let's Encrypt sample hook. Edit the
NVDARemoteCertificate-letsencrypt
file, update the domain name, make it executable, and place it in/etc/letsencrypt/renewal-hooks/post
. No extra steps are required. The certificate will be available for the server immediately after each renewal.
Mac os x El Capitan adds a new feature called system integrity protection to prevent malware from modifying system files. If this feature is enabled, you won't be able to install NVDA Remote Server on your Mac.
If you want to disable it, follow these steps:
- Reboot your system. While rebooting, hold the command+r key to enter in recovery mode.
- Go to the utilities menu, and choose terminal.
- Type the following command:
csrutil disable
- Reboot your computer and go back to the main operating system.
- Install NVDA Remote Server.
Caution! Disabling system integrity protection is a security risk. To enable it back, run csrutil enable
in recovery mode.
If you run a Debian 8, RHEL 7 or RHEL 8 based OpenVZ or Docker container, don't install the Debian 8 or RHEL packages. They will fail and leave the installation in a bad state. In these containers, Systemd produces errors because it can't connect directly to the kernel.
Solution: install Debian 7 or RHL 6 package instead, or run systemd in a privileged container.
Sometimes, calling systemctl causes an error because systemd is not running (for example, after upgrading from Debian 7 to Debian 8). As a consequence, the server package is not installed.
Solution: systemd-sysv is required to install this package. If you tried installing it, run apt -f install or apt-get -f install. Then, reboot your computer. Finally, run dpkg --configure nvda-remote-server.
Mac Os Sierra doesn't allow untrusted software to be installed or used by default. In addition, this option has been removed from System Preferences application. To allow untrusted software:
- Go to the utilities folder, and open terminal.
- Type the following command:
sudo spctl --master-disable
- Enter your password. Now, if system integrity protection has been disabled, you can install the package as described above.
Caution! Allowing untrusted applications is a security risk. To revert to the recommended settings, run sudo spctl --master-enable