├── .dockerignore
├── .gitignore
├── Dockerfile
├── Makefile
├── README.md
├── app.js
├── bin
└── www
├── kube
├── api-deployment.yaml
└── api-svc.yaml
├── package.json
├── public
└── stylesheets
│ └── style.css
├── routes
├── healthcheck.js
├── index.js
└── users.js
└── views
├── error.pug
├── index.pug
├── layout.pug
├── opensearch.pug
└── welcome.pug
/.dockerignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | npm-debug.log
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # Logs
2 | logs
3 | *.log
4 |
5 | # Runtime data
6 | pids
7 | *.pid
8 | *.seed
9 |
10 | # Directory for instrumented libs generated by jscoverage/JSCover
11 | lib-cov
12 |
13 | # Coverage directory used by tools like istanbul
14 | coverage
15 |
16 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
17 | .grunt
18 |
19 | # node-waf configuration
20 | .lock-wscript
21 |
22 | # Compiled binary addons (http://nodejs.org/api/addons.html)
23 | build/Release
24 |
25 | # Dependency directory
26 | # https://docs.npmjs.com/cli/shrinkwrap#caveats
27 | node_modules
28 |
29 | # Debug log from npm
30 | npm-debug.log
31 |
--------------------------------------------------------------------------------
/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM node:alpine
2 |
3 | # Create app directory
4 | RUN mkdir -p /usr/src/app
5 | WORKDIR /usr/src/app
6 |
7 | # Install app dependencies
8 | COPY package.json /usr/src/app/
9 | RUN npm install
10 |
11 | # Bundle app source
12 | COPY . /usr/src/app
13 |
14 | EXPOSE 3000
15 | CMD [ "npm", "start" ]
--------------------------------------------------------------------------------
/Makefile:
--------------------------------------------------------------------------------
1 | build:
2 | docker build -t quay.io/openai/gonode .
3 |
4 | push: build
5 | docker push quay.io/openai/gonode
6 |
7 | run: build
8 | docker run -it -p 3000:5000 -e AWS_DEFAULT_REGION=us-west-2 quay.io/openai/gonode
9 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | **Status:** Archive (code is provided as-is, no updates expected)
2 |
3 | Use this service to quickly alias and redirect to long URLs
4 |
5 | ## Setup
6 | ### Server setup
7 | Recommended to use Heroku to host the service if you don't care about security. Should take about 5 minutes to stand up. Otherwise you can put it on a Kubernetes cluster.
8 | 1. Create a Firebase project
9 | 1. Export Firebase creds on your [Firebase service account](https://console.developers.google.com/iam-admin/serviceaccounts/) as a json file
10 | 1. Create a new Heroku app and connect it to this git repo or your fork of it
11 | 1. Make json file's creds available as environment variables in your deployment environment (eg. through Heroku dashboard)
12 | * `FIREBASE_PROJECT_ID`
13 | * `FIREBASE_CLIENT_EMAIL`
14 | * `FIREBASE_DATABASE_NAME`
15 | * `FIREBASE_PRIVATE_KEY` which can have literal `\n` instead of actual newlines
16 | 1. Deploy!
17 |
18 | ### Client setup
19 | Add this site to your Chrome search engines by scrolling down until you see three empty boxes. Enter values go, go, and go.corp.mycompany.com/%s, respectively.
20 |
21 | ### Advanced setup
22 | To get the most benefit from the service, you should setup a DNS entry
23 | on your local network, `go.corp.mycompany.com`. Make sure that
24 | `corp.mycompany.com` is in the search domains for each user on the
25 | network. This is usually easily accomplished by configuring your DHCP
26 | server. Now, simply typing `go` into your browser should take you to
27 | the service, where you can register shortcuts. Those
28 | shortcuts will also be available by typing `go/shortcut`.
29 |
30 | On a mac, you can add the domain to your DNS search paths
31 | ```
32 | networksetup -setsearchdomains Wi-Fi corp.example.com
33 | ```
34 | However, this will require all bare names to hang instead of failing instantly, so you may wish to stick with the Chrome-only solution.
35 |
36 | ## Use
37 | When you type in the Chrome URL bar go<TAB> anything you type afterwards will bring you to that alias here.
38 | You can then paste your link in the box and the next time you visit this alias, it will bring you straight to the site you pasted before.
39 |
40 | You can override any alias by visiting eg., #{path}/myalias/mynewsite.com
41 |
42 | ## Develop
43 | Export your firebase credentials as appropriate environment variables. You'll need:
44 | * `FIREBASE_PROJECT_ID`
45 | * `FIREBASE_CLIENT_EMAIL`
46 | * `FIREBASE_DATABASE_NAME`
47 | * `FIREBASE_PRIVATE_KEY` which can have literal `\n` instead of actual newlines
48 |
49 | ```
50 | npm install
51 | npm install -g devtool
52 | npm run dev
53 | ```
54 |
55 | Visit http://localhost:3000/myalias
56 |
57 | ## Test in docker
58 | ```
59 | make run
60 | ```
61 |
62 | ## Deploy
63 | ```
64 | kubectl apply -f kube/api-deployment.yaml
65 | kubectl apply -f kube/api-svc.yaml
66 | ```
67 |
--------------------------------------------------------------------------------
/app.js:
--------------------------------------------------------------------------------
1 | var express = require('express');
2 | var path = require('path');
3 | var favicon = require('serve-favicon');
4 | var logger = require('morgan');
5 | var cookieParser = require('cookie-parser');
6 | var bodyParser = require('body-parser');
7 |
8 | var index = require('./routes/index');
9 | var users = require('./routes/users');
10 | var healthcheck = require('./routes/healthcheck');
11 |
12 | var app = express();
13 |
14 | // view engine setup
15 | app.set('views', path.join(__dirname, 'views'));
16 | app.set('view engine', 'pug');
17 |
18 | // uncomment after placing your favicon in /public
19 | //app.use(favicon(path.join(__dirname, 'public', 'favicon.ico')));
20 | app.use(logger('dev'));
21 | app.use(bodyParser.json());
22 | app.use(bodyParser.urlencoded({ extended: false }));
23 | app.use(cookieParser());
24 | app.use(express.static(path.join(__dirname, 'public')));
25 |
26 | app.use('/users', users);
27 | app.use('/healthcheck', healthcheck);
28 | app.use('/', index);
29 |
30 | // catch 404 and forward to error handler
31 | app.use(function(req, res, next) {
32 | var err = new Error('Not Found');
33 | err.status = 404;
34 | next(err);
35 | });
36 |
37 | // error handler
38 | app.use(function(err, req, res, next) {
39 | // set locals, only providing error in development
40 | res.locals.message = err.message;
41 | res.locals.error = req.app.get('env') === 'development' ? err : {};
42 |
43 | // render the error page
44 | res.status(err.status || 500);
45 | res.render('error');
46 | });
47 |
48 | module.exports = app;
49 |
--------------------------------------------------------------------------------
/bin/www:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env node
2 |
3 | /**
4 | * Module dependencies.
5 | */
6 |
7 | var app = require('../app');
8 | var debug = require('debug')('go-alias:server');
9 | var http = require('http');
10 |
11 | /**
12 | * Get port from environment and store in Express.
13 | */
14 |
15 | var port = normalizePort(process.env.PORT || '3000');
16 | app.set('port', port);
17 |
18 | /**
19 | * Create HTTP server.
20 | */
21 |
22 | var server = http.createServer(app);
23 |
24 | /**
25 | * Listen on provided port, on all network interfaces.
26 | */
27 |
28 | server.listen(port);
29 | server.on('error', onError);
30 | server.on('listening', onListening);
31 |
32 | /**
33 | * Normalize a port into a number, string, or false.
34 | */
35 |
36 | function normalizePort(val) {
37 | var port = parseInt(val, 10);
38 |
39 | if (isNaN(port)) {
40 | // named pipe
41 | return val;
42 | }
43 |
44 | if (port >= 0) {
45 | // port number
46 | return port;
47 | }
48 |
49 | return false;
50 | }
51 |
52 | /**
53 | * Event listener for HTTP server "error" event.
54 | */
55 |
56 | function onError(error) {
57 | if (error.syscall !== 'listen') {
58 | throw error;
59 | }
60 |
61 | var bind = typeof port === 'string'
62 | ? 'Pipe ' + port
63 | : 'Port ' + port;
64 |
65 | // handle specific listen errors with friendly messages
66 | switch (error.code) {
67 | case 'EACCES':
68 | console.error(bind + ' requires elevated privileges');
69 | process.exit(1);
70 | break;
71 | case 'EADDRINUSE':
72 | console.error(bind + ' is already in use');
73 | process.exit(1);
74 | break;
75 | default:
76 | throw error;
77 | }
78 | }
79 |
80 | /**
81 | * Event listener for HTTP server "listening" event.
82 | */
83 |
84 | function onListening() {
85 | var addr = server.address();
86 | var bind = typeof addr === 'string'
87 | ? 'pipe ' + addr
88 | : 'port ' + addr.port;
89 | debug('Listening on ' + bind);
90 | }
91 |
--------------------------------------------------------------------------------
/kube/api-deployment.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: extensions/v1beta1
2 | kind: Deployment
3 | metadata:
4 | name: gonode
5 | namespace: system
6 | spec:
7 | replicas: 3
8 | template:
9 | metadata:
10 | labels:
11 | app: gonode
12 | spec:
13 | containers:
14 | - name: app
15 | image: quay.io/openai/gonode
16 | imagePullPolicy: Always
17 | ports:
18 | - containerPort: 3000
19 | env:
20 | - name: AWS_DEFAULT_REGION
21 | value: "us-west-2"
22 | - name: FIREBASE_PROJECT_ID
23 | valueFrom:
24 | secretKeyRef:
25 | name: firebase
26 | key: project-id
27 | - name: FIREBASE_DATABASE_NAME
28 | valueFrom:
29 | secretKeyRef:
30 | name: firebase
31 | key: database-name
32 | - name: FIREBASE_CLIENT_EMAIL
33 | valueFrom:
34 | secretKeyRef:
35 | name: firebase
36 | key: client-email
37 | - name: FIREBASE_PRIVATE_KEY
38 | valueFrom:
39 | secretKeyRef:
40 | name: firebase
41 | key: private-key
42 | readinessProbe:
43 | httpGet:
44 | path: /healthcheck
45 | port: 3000
46 | initialDelaySeconds: 3
47 | timeoutSeconds: 1
48 | restartPolicy: Always
49 | imagePullSecrets:
50 | - name: quay-login-secret
51 | nodeSelector:
52 | aws/region: us-west-2
53 |
--------------------------------------------------------------------------------
/kube/api-svc.yaml:
--------------------------------------------------------------------------------
1 | apiVersion: v1
2 | kind: Service
3 | metadata:
4 | name: go
5 | namespace: system
6 | spec:
7 | ports:
8 | - port: 80 # the port that this service should serve on
9 | # the container on each pod to connect to, can be a name
10 | # (e.g. 'www') or a number (e.g. 80)
11 | targetPort: 3000
12 | nodePort: 30212
13 | protocol: TCP
14 | type: NodePort
15 | # just like the selector in the replication controller,
16 | # but this time it identifies the set of pods to load balance
17 | # traffic to.
18 | selector:
19 | app: gonode
20 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "go-alias",
3 | "version": "0.0.1",
4 | "private": true,
5 | "scripts": {
6 | "start": "node ./bin/www",
7 | "dev": "DEBUG=go-alias:* devtool bin/www -w"
8 | },
9 | "engines": {
10 | "node": "6.6.0",
11 | "npm": "3.10.3"
12 | },
13 | "dependencies": {
14 | "body-parser": "~1.17.1",
15 | "cookie-parser": "~1.4.3",
16 | "debug": "~2.6.3",
17 | "express": "~4.15.2",
18 | "firebase-admin": "^4.1.4",
19 | "firebase-encode": "0.0.5",
20 | "morgan": "~1.8.1",
21 | "pug": "~2.0.0-beta11",
22 | "serve-favicon": "~2.4.2"
23 | }
24 | }
25 |
--------------------------------------------------------------------------------
/public/stylesheets/style.css:
--------------------------------------------------------------------------------
1 | body {
2 | padding: 50px;
3 | font: 14px "Lucida Grande", Helvetica, Arial, sans-serif;
4 | }
5 |
6 | a {
7 | color: #00B7FF;
8 | }
9 |
--------------------------------------------------------------------------------
/routes/healthcheck.js:
--------------------------------------------------------------------------------
1 | var express = require('express');
2 | var router = express.Router();
3 |
4 | /* GET users listing. */
5 | router.get('/', function(req, res, next) {
6 | res.send('ok');
7 | });
8 |
9 | module.exports = router;
10 |
--------------------------------------------------------------------------------
/routes/index.js:
--------------------------------------------------------------------------------
1 |
2 | var encoder = require('firebase-encode');
3 | var express = require('express');
4 | var router = express.Router();
5 |
6 | var admin = require("firebase-admin");
7 |
8 | // Handle escaped new lines in private key
9 | var firebasePrivateKey = process.env.FIREBASE_PRIVATE_KEY
10 | if (firebasePrivateKey.search(/\\n/) != -1) {
11 | firebasePrivateKey = firebasePrivateKey.replace(/\\n/g, '\n')
12 | }
13 |
14 | admin.initializeApp({
15 | credential: admin.credential.cert({
16 | projectId: process.env.FIREBASE_PROJECT_ID,
17 | clientEmail: process.env.FIREBASE_CLIENT_EMAIL,
18 | privateKey: firebasePrivateKey
19 | }),
20 | databaseURL: `https://${process.env.FIREBASE_DATABASE_NAME}.firebaseio.com`
21 | });
22 |
23 |
24 | var db = admin.database();
25 | var linksRef = db.ref('links');
26 |
27 | /* GET home page. */
28 | router.get('/', function(req, res, next) {
29 | var fullUrl = req.protocol + '://' + req.get('host')
30 | res.render('welcome', {path: fullUrl})
31 | })
32 |
33 | /* For some reason this never gets pull even with link rel=search tag
34 | https://developer.mozilla.org/en-US/Add-ons/Creating_OpenSearch_plugins_for_Firefox
35 | */
36 | router.get('/opensearch.xml', function(req, res, next) {
37 | var fullUrl = req.protocol + '://' + req.get('host')
38 | res.render('opensearch', {path: fullUrl})
39 | })
40 |
41 | router.get('/:id', function(req, res, next) {
42 | linksRef.child(req.params.id).once('value').then(function(dataSnapshot) {
43 | if (dataSnapshot.exists()) {
44 | res.redirect(encoder.decode(dataSnapshot.val()))
45 | } else {
46 | res.render('index', { title: req.params.id })
47 | }
48 | });
49 |
50 | });
51 |
52 | function getId(path) {
53 | return path.split('/', 2)[1]
54 | }
55 |
56 | function getURL(path) {
57 | for (var i = 1; i < path.length; i++) {
58 | if (path[i] == '/') {
59 | return path.substring(i+1)
60 | }
61 | }
62 | return ''
63 | }
64 |
65 | router.get(/.*/, function(req, res, next) {
66 | var id = getId(req.path)
67 | var url = getURL(req.path)
68 | // make sure this ID isn't already taken
69 | // linksRef.child(req.params.id).once('value').then(function(dataSnapshot) {
70 | // if (dataSnapshot.exists()) {
71 | // res.redirect(dataSnapshot.val())
72 | // } else {
73 | // // Success.
74 | // }
75 | // });
76 |
77 | // Clobber
78 | if (!url.match(/https?/)) {
79 | url = 'http://' + url
80 | }
81 | linksRef.child(id).set(encoder.encode(url));
82 | res.render('index', { title: id, url: url })
83 |
84 |
85 | });
86 |
87 | module.exports = router
88 |
--------------------------------------------------------------------------------
/routes/users.js:
--------------------------------------------------------------------------------
1 | var express = require('express');
2 | var router = express.Router();
3 |
4 | /* GET users listing. */
5 | router.get('/', function(req, res, next) {
6 | res.send('respond with a resource');
7 | });
8 |
9 | module.exports = router;
10 |
--------------------------------------------------------------------------------
/views/error.pug:
--------------------------------------------------------------------------------
1 | extends layout
2 |
3 | block content
4 | h1= message
5 | h2= error.status
6 | pre #{error.stack}
7 |
--------------------------------------------------------------------------------
/views/index.pug:
--------------------------------------------------------------------------------
1 | extends layout
2 |
3 | block content
4 | a(href='/' + title)
5 | h1= title
6 | if (url)
7 | p Will now redirect to #{url}
8 | else
9 | p No URL set. Enter one:
10 | input#url.form-control(type='text', placeholder='url' name='url')
11 | button.btn.btn-primary(
12 | type='button',
13 | onclick='go()') Submit
14 |
15 | script.
16 | title='#{title}'
17 | function go() {
18 | window.location.href = `${title}/${document.getElementById("url").value}`
19 | }
20 | input = document.getElementById('url')
21 | if (input) {
22 | input.addEventListener('keypress', function (e) {
23 | var key = e.which || e.keyCode;
24 | if (key === 13) { // enter
25 | go()
26 | }
27 | });
28 | }
29 |
--------------------------------------------------------------------------------
/views/layout.pug:
--------------------------------------------------------------------------------
1 | doctype html
2 | html
3 | head
4 | title= title
5 | link(rel='stylesheet', href='/stylesheets/style.css')
6 | block head
7 | body
8 | block content
9 |
--------------------------------------------------------------------------------
/views/opensearch.pug:
--------------------------------------------------------------------------------
1 |
3 | xx
4 | Quickly visit link aliases!
5 | UTF-8
6 | 
7 |
8 |
9 | #{path}/
10 |
11 |
--------------------------------------------------------------------------------
/views/welcome.pug:
--------------------------------------------------------------------------------
1 | extends layout
2 |
3 | block content
4 | h1 Welcome to the go service!
5 | p Use this service to easily visit long, complicated URLs.
6 | h2 Setup
7 | p.
8 | Add this site to your Chrome search engines by scrolling down until you see three empty boxes. Enter values go, go, and #{path}/%s, respectively.
9 | h2 Use
10 | p.
11 | When you type in the Chrome URL bar
12 | go<TAB>
13 | anything you type afterwards will bring you to that alias here.
14 | You can then paste your link in the box and the next time you visit this alias, it will bring you straight to the site you pasted before.
15 | p You can override any alias by visiting eg., #{path}/myalias/mynewsite.com
16 | h2 Feedback
17 | p This project is open source! Check out the
18 | a(href="https://github.com/openai/go-alias") Github
19 |
--------------------------------------------------------------------------------