This is kind of a silly question, but all of the examples I've seen of deploying a Go project use App Engine and I just want to deploy on an Ubuntu server using Digital Ocean. How do I go from running 127.0.0.1:8080/index
locally to running example-site.com/index
on a remote server?
评论:
Battleroid:
manwith4names:I would imagine like you do so with any other Web application. Probably use Nginx to proxy requests to it.
Battleroid:Any examples of using Nginx to proxy requests? I've never deployed a web application before
weberc2:I'm on my phone but in Nginx it's usually as simple as just rerouting requests for / to something like proxy_pass 127.0.0.1:8000.
If you search say, how to host a Go or even a Python Web application with Nginx I'm sure you'll find a bunch of info.
After I get some sleep I'll try to post something a bit more constructive.
flexd:You don't need Nginx for this AFAIK; you can just make your app listen on port 80 and run it as sudo.
EDIT: never mind. Don't do this. It's a bad security practice. Thanks to those who corrected me.
weberc2:Please do not encourage people to run things as root.
You could do
setcap 'cap_net_bind_service=+ep' /path/to/program
to give it theCAP_NET_BIND_SERVICE
capability to be able to bind to ports under 1024 without root.But the recommended way is to use nginx/apache/Caddy https://caddyserver.com/docs/proxy to proxy requests to the backend.
Battleroid:I didn't know that. Thanks for the heads up.
weberc2:Yeah that too, but I would just use Nginx anyway. More control over how you proxy requests and junk like that. That and you shouldn't host user applications below port 1023 (?, might be 1024 can't remember).
weberc2:Yeah, another user corrected me as well. Evidently you shouldn't run your application as root, but instead use a reverse proxy.
manwith4names:Not a silly question. You need to have a remote server and upload your program to that server (FTP, rsync, scp, etc). You'll also need some way to connect port 80 to whatever port your application is listening on (you can even make your app listen on 80, though you'll likely need to run it as a privileged user). Finally, you'll need your Ubuntu server to have a static IP address and you'll need to purchase a the domain name to register your IP address to (this last point is a bit complex, and I don't know what you already know about DNS, so feel free to ask further questions).
flexd:I'm planning to use digital ocean for my remote server and I believe I need to use Nginx?
weberc2:You can use nginx, or apache, or caddy (written in go). You have lots of options!
neoasterisk:Evidently you should use nginx or caddy; my advice about running your application as root and listening on 80 is insecure. In the past I've used rsync to deploy my payload to DO from my build server; this worked pretty well. You'll also want to configure your server to use upstart or whatever Ubuntu is using these days; this will make sure your app runs on server reboot and it will restart it if it crashes, etc.
First just transfer your Go app to the remote server. You can do that either by uploading the binary and any static files via
ssh
, particularly with thescp
command or alternatively:
- install Go to the remote server
go get
your app- deploy with
go install
.
The remote server has an IP say 104.1.2.3
so after you deploy the Go app to the remote server you can access it by visiting http://104.1.2.3:8080/index
. Of course it's more human friendly to use domain names that lead to IPs. So if in your DNS you have that example-site.com -> 104.1.2.3
then you can access the app via http://example-site.com/index
. You need to check the settings of your DNS provider for that.
Another thing you need to be aware of is port forwarding which practically means that you need to have port 8080 open in the firewall. It's more common for security reasons to have 8080 disabled and access every app from port 80. If the only thing you need to run is your Go app then just configure it to listen on 80 and you are done. The flag package is ideal for that so that when you start your app in the remote server you can do something like goapp -http=localhost:80
. That gives you the flexibility to change the ports that your apps are using without recompiling.
If you want to run more than one apps then you need to configure another server to act as a reverse proxy like nginx
or apache2
. That way you could for example have http://example-site.com/index
served from Go app1 and http://example-site.com/service
served from Go app2.
In this case, you need to configure the reverse proxy in such a way so that /index
goes to 0.0.0.0:8080
(Go app 1) and /service
goes to 0.0.0.0:8081
(Go app2).
Now after you are done with your tests, you can turn your Go app into a service so that it automatically starts when the Ubuntu server starts. That can be done with Upstart or Systemd depending on your version. Check out a sample Upstart configuration file:
# goapp upstart conf script
description "goapp upstart conf script"
start on (net-device-up and local-filesystems and runlevel [2345])
stop on runlevel [!2345]
respawn
respawn limit 10 5
console log
exec /home/yourname/go/bin/goapp -http="localhost:8080"
Modify and write to a file /etc/init/goapp.conf
and then start the service by sudo service goapp start
.
Lastly, if you log stuff in your application like with the standard log package log.Println("An important event occurred!")
then you can find those messages under /var/log/upstart/goapp.log
.
Happy hacking!