All can be installed through Homebrew.
- Ruby (see
Gemfile
or.ruby-version
for version) - Node 0.10.x (for the Angular app in
callveyor
) - Redis 2.8
- MySQL 5.5.40
- Heroku toolbelt
- Ngrok
Create a separate Twilio account for dev/staging and production - it will be much easier to track down errors in development. Use the ngrok.io subdomain as the host for development, and the herokuapp.com host for production.
Create TwiML apps:
- Browser phone: request and fallback to
/twiml/caller_sessions
, status callback to/caller/end_session
- Dial in: request and fallback to
/callin/create
, status callback to/caller/end_session
- Dashboard: request and fallback to
/client/monitors/callers/start
Set TWILIO_APP_SID to the browser phone app's SID.
Set TWILIO_MONITOR_APP_SID to the dashboard app's SID.
For each call-in number, configure the number to use the dial in app.
Make sure to copy .env.example
to .env
and update credentials if needed.
Install gems with bundle
from the app root. Set up the database with rake db:create && db:schema:load
. Install caller interface deps with npm install && bower install
from callveyor
.
Start MySQL server and Redis server.
Launch the web app with rails s
(for simple testing only). Launch the web app and background processes with heroku local -f Procfile.dev
(customize Procfile.dev
to choose which processes to launch) and visit localhost:5000
for the admin interface, and localhost:5000/app
for the caller interface.
Launch the caller interface from callveyor
with grunt serve
and visit localhost:9000/app/login
. (Don't worry about assets not loading.) After logging in, you'll get the error Cannot GET /app
. Remove app
from the URL to visit localhost:9000/
to reach the logged-in caller interface.
Receive Twilio callbacks through Ngrok by running ngrok http -subdomain=impactdialing 5000
.
After making changes to Callveyor, build the Angular app into the Rails app with rake callveyor:build
.
Run rspec spec
for Ruby tests.
Run foreman run rspec features
for acceptance tests.
Run grunt test
from callveyor
to continuously run Callveyor tests in Firefox, Safari and Chrome.
We run Impact Dialing on Heroku. We deploy to two apps. The main one ("impactdialing") serves admin.impactdialing.com and caller.impactdialing. The other one ("impactdialing-twiml") is solely responsible for handling Twilio webhooks, and runs a single Perforance dyno.
Performance dynos run on a dedicated VM and don't suffer from performance leakage from neighboring dynos, and so have a consistently fast response time that we couldn't achieve on standard dynos. By isolating the two apps, we can be sure that slow requests on the main app don't disrupt call flow, which is very latency-sensitive.
The main impactdialing app should be configured to have the Cloudflare proxy enabled, to protect from attacks. impactdialing-twiml should not have the Cloudflare proxy enabled, as it only services requests from Twilio, and we want those requests to stay within the AWS datacenter and not take a roundtrip through Cloudflare first. Make sure to keep this URL a secret, since it does not have Cloudflare protection.
- Heroku - hosting/platform
- HireFire - autoscaling Heroku
- Cloudflare - DNS, etc
- RDS - MySQL hosting
- S3 - list and audio storage, daily Redis backups, log backups
- RedisLabs - Redis hosting
- Pusher - realtime
- Twilio - calls
- Mandrill - emails
- Bugsnag - exceptions
- Papertrail - logs
- Librato - dashboards
- PagerDuty - alerts
- Blazemeter - load testing
- Sauce - browser testing
- CircleCI - continuous integration
- Ngrok - tunnel from a public domain to localhost
- Freshdesk - email support
- Olark - chat support
- Usersnap - screenshots/JS dump support
CALLIN_PHONE
: The Twilio phone number associated with the "Production call-in" TwiML appCAMPAIGN_EXPIRY
: A number of days; campaigns that have not made any dials in this number of days will be auto-archivedDATABASE_READ_SLAVE1_URL
: URL to a MySQL read slaveDATABASE_READ_SLAVE2_URL
: URL to a second MySQL read slaveDATABASE_SIMULATOR_SLAVE_URL
: URL to a third MySQL read slave, intended for use by predictive simulator workersDATABASE_URL
: URL to MySQL masterDO_NOT_CALL_PORTED_LISTS_PROVIDER_URL
: HTTP AUTH URL to tcpacompliance ported listsDO_NOT_CALL_REDIS_URL
: URL to redis instance where block and ported cell lists are cachedDO_NOT_CALL_WIRELESS_BLOCK_LIST_PROVIDER_URL
: HTTP AUTH URL to qscdl block listsHIREFIRE_TOKEN
: Auth token provided by HireFire for auto-scalingINCOMING_CALLBACK_HOST
: HOST of end-points to process TwiMLINSTRUMENT_ACTIONS
: Toggle librato-rails experimentalinstrument_action
usage; 0 = do not instrument controller actions; 1 = instrument controller actionsLIBRATO_SOURCE
: Names the source of the Librato metrics being collectedLIBRATO_TOKEN
: Auth token provided by LibratoLIBRATO_USER
: Username for Librato account ([email protected])MANDRILL_API_KEY
: ...MAX_THREADS
: How many threads should puma start (1 - app not proven thread-safe yet)PUSHER_APP_ID
: ...PUSHER_KEY
: ...PUSHER_SECRET
: ...RACK_ENV
: ...RACK_TIMEOUT
: Number of seconds before rack considers request timed out (max 30 for heroku)RAILS_ENV
: ...RECORDING_ENV
: Root-level folder to store recordings in on s3REDIS_PHONE_KEY_INDEX_STOP
: CAUTION! Changing this requires migrating household data in redis, should be negative four (-4); this determines the position phone numbers are partitioned when creating redis keys and redis hash keys.REDIS_URL
: URL of primary (default) redis instance to connectS3_ACCESS_KEY
: ...S3_BUCKET
: ...S3_SECRET_ACCESS_KEY
: ...STRIPE_PUBLISHABLE_KEY
: ...STRIPE_SECRET_KEY
: ...TWILIO_ACCOUNT
: ...TWILIO_APP_SID
: SID of the Browser Phone TwiML appTWILIO_AUTH
: ...TWILIO_CALLBACK_HOST
: the hostname of the impactdialing-twiml Heroku appTWILIO_CALLBACK_PORT
: the port of the impactdialing-twiml Heroku appTWILIO_CAPABILITY_TOKEN_TTL
: TTL of Twilio Client capability tokens (caller app & admin dashboard)TWILIO_FAILOVER_HOST
: the same as the hostname of the impactdialing-twiml Heroku appTWILIO_MONITOR_APP_SID
: SID of the Dashboard TwiML appTWILIO_RETRIES
: Number of retries Twilio ruby client should perform before considering API request as failedUPSERT_GEM_ON
: Upsert is a SLOWER & MORE ERROR-PRONE alternative to activerecord-import; 0 = use activerecord-import; 1 = use upsertVOIP_API_URL
: Twilio's API host (api.twilio.com)VOTER_BATCH_SIZE
: Number of rows of CSV data to process before committing to redis during uploads. Keep at a max of 100 down to a min of 20 or 30. Lower value will increase overall upload time but decrease commit time thereby improving redis throughput.WEB_CONCURRENCY
: Number of puma workers to start.
- billing -> Billing::Jobs::AutoRecharge, Billing::Jobs::StripeEvent, DebitJob
- call_flow -> CallerPusherJob, CampaignOutOfNumbersJob, Providers::Phones::Jobs::DropMessage, EndRunningCallJob, RedirectCallerJob, VoterConnectedPusherJob
- dial_queue -> CallFlow::DialQueue::Jobs::Recycle, CallFlow::Web::Jobs::CacheContactFields, DoNotCall::Jobs::BlockedNumberCreatedOrDestroyed, CachePhonesOnlyScriptQuestions, CallerGroupJob
- dialer_worker -> CalculateDialsJob, DialerJob
- general -> Archival::Jobs::CampaignArchived, Archival::Jobs::CampaignSweeper, DoNotCall::Jobs::CachePortedLists, DoNotCall::Jobs::CacheWirelessBlockList, DoNotCall::Jobs::RefreshPortedLists, DoNotCall::Jobs::RefreshWirelessBlockList, DeliverInvitationEmailJob, PhantomCallerJob, ResetVoterListCounterCache
- import -> CallList::Jobs::Import, CallList::Jobs::Prune, CallList::Jobs::ToggleActive, CallList::Jobs::Upload
- persist_jobs -> none!
- persistence -> CallFlow::Jobs::Persistence
- reports -> AdminReportJob, ReportAccountUsageJob, ReportDownloadJob
- simulator_worker -> SimulatorJob
- twilio_stats -> UpdateStatsAttemptsEm, UpdateStatsTransfersEm, UpdateTwilioStatsCallerSession
- Logo should look good at 300x57 and be png
- Use imagemagick to convert format if needed
convert img.jpg img.png
- Use imagemagick to resize if needed
convert img.png -resize xx% img-h1.png
- Update
en.yml
(use previous whitelabel entries as template) - Billing Link is only for certain customers
- Add logo to
public/img
folder, naming filedomain-name-h1.ext
- Update CSS in
public/styles/style.css
(use previous whitelabel entries as template - class names are dynamically generated in erb) - Verify logo displays nicely on localhost
- Update
/etc/hosts
- Buy domain from badger.com (use visa xx8669)
- Setup domain in Cloudflare
- Use other domains as template
- Verify security rules whitelist upload urls
- Verify High Security Profile is used
- Verify CDN + High Performance is used
- Update DNS w/ Badger
- Add domain to heroku production app
- Add domain to heroku staging app
- Add the following to
/etc/hosts
impactdialing-staging.herokuapp.com whitelabel-domain.com
- Visit
whitelabel-domain.com
The resque-loner
gem is used for a few jobs. This gem defines .redis_key
and uses that to track job uniqueness. Careful not to override this in implementation classes.