re-natal/re-natal.coffee

498 lines
14 KiB
CoffeeScript
Raw Permalink Normal View History

# Re-Natal
2015-08-25 23:48:43 -04:00
# Bootstrap ClojureScript React Native apps
# Dan Motzenbecker
# http://oxism.com
# MIT License
2015-10-04 18:53:54 -04:00
fs = require 'fs'
2015-10-17 15:10:31 -04:00
net = require 'net'
http = require 'http'
2015-10-04 18:53:54 -04:00
crypto = require 'crypto'
child = require 'child_process'
cli = require 'commander'
chalk = require 'chalk'
semver = require 'semver'
pkgJson = require __dirname + '/package.json'
2015-08-21 23:45:42 -04:00
nodeVersion = pkgJson.engines.node
2015-08-21 23:45:42 -04:00
resources = __dirname + '/resources/'
validNameRx = /^[A-Z][0-9A-Z]*$/i
2015-08-21 23:45:42 -04:00
camelRx = /([a-z])([A-Z])/g
projNameRx = /\$PROJECT_NAME\$/g
projNameHyphRx = /\$PROJECT_NAME_HYPHENATED\$/g
projNameUsRx = /\$PROJECT_NAME_UNDERSCORED\$/g
platformRx = /\$PLATFORM\$/g
devHostRx = /\$DEV_HOST\$/g
2015-12-07 21:49:25 +01:00
rnVersion = '0.16.0'
2015-10-17 15:10:31 -04:00
rnPackagerPort = 8081
2015-09-19 14:19:54 -04:00
podMinVersion = '0.38.2'
process.title = 're-natal'
sampleCommand = '(dispatch [:set-greeting "Hello Native World!"])'
2015-08-21 23:45:42 -04:00
2015-08-25 21:28:58 -04:00
log = (s, color = 'green') ->
console.log chalk[color] s
2015-08-21 23:45:42 -04:00
2015-08-25 21:28:58 -04:00
logErr = (err, color = 'red') ->
console.error chalk[color] err
2015-09-19 14:47:07 -04:00
process.exit 1
2015-08-21 23:45:42 -04:00
2015-10-04 15:32:23 -04:00
exec = (cmd, keepOutput) ->
if keepOutput
2015-10-04 18:53:54 -04:00
child.execSync cmd
2015-10-04 15:32:23 -04:00
else
2015-10-04 18:53:54 -04:00
child.execSync cmd, stdio: 'ignore'
2015-10-04 15:32:23 -04:00
2015-10-03 20:34:10 -04:00
readFile = (path) ->
fs.readFileSync path, encoding: 'ascii'
2015-10-04 15:33:20 -04:00
edit = (path, pairs) ->
2015-08-21 23:45:42 -04:00
fs.writeFileSync path, pairs.reduce (contents, [rx, replacement]) ->
contents.replace rx, replacement
2015-10-03 20:34:10 -04:00
, readFile path
2015-08-21 23:45:42 -04:00
pluckUuid = (line) ->
line.match(/\[(.+)\]/)[1]
2015-11-14 18:17:38 -05:00
getUuidForDevice = (deviceName) ->
device = getDeviceList().find (line) -> line.match deviceName
unless device
logErr "Cannot find device `#{deviceName}`"
pluckUuid device
toUnderscored = (s) ->
s.replace(camelRx, '$1_$2').toLowerCase()
2015-10-17 15:10:31 -04:00
checkPort = (port, cb) ->
sock = net.connect {port}, ->
sock.end()
2015-11-14 14:52:30 -05:00
http.get "http://localhost:#{port}/status", (res) ->
2015-10-17 15:10:31 -04:00
data = ''
res.on 'data', (chunk) -> data += chunk
res.on 'end', ->
cb data.toString() isnt 'packager-status:running'
.on 'error', -> cb true
.setTimeout 3000
sock.on 'error', ->
sock.end()
cb false
ensureFreePort = (cb) ->
checkPort rnPackagerPort, (inUse) ->
if inUse
logErr "
Port #{rnPackagerPort} is currently in use by another process
and is needed by the React Native packager.
"
cb()
2015-11-14 18:18:19 -05:00
generateConfig = (name) ->
log 'Creating Re-Natal config'
2015-11-14 18:18:19 -05:00
config =
name: name
device: getUuidForDevice 'iPhone 6'
2015-11-14 18:18:19 -05:00
writeConfig config
config
2015-09-19 14:02:28 -04:00
writeConfig = (config) ->
try
fs.writeFileSync '.re-natal', JSON.stringify config, null, 2
2015-09-19 14:02:28 -04:00
catch {message}
logErr \
if message.match /EACCES/i
'Invalid write permissions for creating .re-natal config file'
2015-09-19 14:02:28 -04:00
else
message
2015-09-19 14:21:19 -04:00
readConfig = ->
try
JSON.parse readFile '.re-natal'
2015-09-19 14:21:19 -04:00
catch {message}
logErr \
if message.match /ENOENT/i
'No Re-Natal config was found in this directory (.re-natal)'
2015-09-19 14:21:19 -04:00
else if message.match /EACCES/i
'No read permissions for .re-natal'
2015-09-19 14:21:19 -04:00
else if message.match /Unexpected/i
'.re-natal contains malformed JSON'
2015-09-19 14:21:19 -04:00
else
message
getBundleId = (name) ->
2015-10-03 20:35:46 -04:00
try
if line = readFile "ios/#{name}.xcodeproj/project.pbxproj"
2015-10-04 19:44:22 -04:00
.match /PRODUCT_BUNDLE_IDENTIFIER = (.+);/
2015-10-03 20:35:46 -04:00
line[1]
2015-10-04 19:44:22 -04:00
else if line = readFile "ios/#{name}/Info.plist"
2015-10-04 19:44:22 -04:00
.match /\<key\>CFBundleIdentifier\<\/key\>\n?\s*\<string\>(.+)\<\/string\>/
2015-10-03 20:35:46 -04:00
rfcIdRx = /\$\(PRODUCT_NAME\:rfc1034identifier\)/
if line[1].match rfcIdRx
line[1].replace rfcIdRx, name
else
line[1]
else
throw new Error 'Cannot find bundle identifier in project.pbxproj or Info.plist'
catch {message}
logErr message
copyDevEnvironmentFiles = (projNameHyph, projName, devHost) ->
mainIosDevPath = "env/dev/env/ios/main.cljs"
mainAndroidDevPath = "env/dev/env/android/main.cljs"
exec "cp #{resources}cljs/main_dev.cljs #{mainIosDevPath}"
edit mainIosDevPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName], [platformRx, "ios"], [devHostRx, devHost] ]
exec "cp #{resources}cljs/main_dev.cljs #{mainAndroidDevPath}"
edit mainAndroidDevPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName], [platformRx, "android"], [devHostRx, devHost]]
requestImgMacroDevPath = "env/dev/env/require_img.clj"
exec "cp #{resources}require_img_dev.clj #{requestImgMacroDevPath}"
edit requestImgMacroDevPath, [[devHostRx, devHost]]
copyProdEnvironmentFiles = (projNameHyph, projName) ->
mainIosProdPath = "env/prod/env/ios/main.cljs"
mainAndroidProdPath = "env/prod/env/android/main.cljs"
exec "cp #{resources}cljs/main_prod.cljs #{mainIosProdPath}"
edit mainIosProdPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName], [platformRx, "ios"]]
exec "cp #{resources}cljs/main_prod.cljs #{mainAndroidProdPath}"
edit mainAndroidProdPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName], [platformRx, "android"]]
requestImgMacroProdPath = "env/prod/env/require_img.clj"
exec "cp #{resources}require_img_prod.clj #{requestImgMacroProdPath}"
copyFigwheelBridge = (projNameUs) ->
exec "cp #{resources}figwheel-bridge.js ."
edit "figwheel-bridge.js", [[projNameUsRx, projNameUs]]
2015-11-21 12:01:44 +01:00
init = (projName) ->
if projName.toLowerCase() is 'react' or !projName.match validNameRx
logErr 'Invalid project name. Use an alphanumeric CamelCase name.'
projNameHyph = projName.replace(camelRx, '$1-$2').toLowerCase()
projNameUs = toUnderscored projName
try
2015-09-18 23:10:03 -04:00
log "Creating #{projName}", 'bgMagenta'
2015-08-25 21:44:11 -04:00
log ''
if fs.existsSync projNameHyph
2015-09-18 23:10:03 -04:00
throw new Error "Directory #{projNameHyph} already exists"
2015-10-04 15:32:23 -04:00
exec 'type lein'
exec 'type watchman'
2015-10-04 19:33:37 -04:00
exec 'type xcodebuild'
2015-10-03 15:48:28 -04:00
2015-08-22 11:06:30 -04:00
log 'Creating Leiningen project'
2015-10-04 15:32:23 -04:00
exec "lein new #{projNameHyph}"
2015-08-22 11:06:30 -04:00
log 'Updating Leiningen project'
process.chdir projNameHyph
2015-10-04 15:32:23 -04:00
exec "cp #{resources}project.clj project.clj"
2015-10-31 11:35:07 -04:00
edit \
'project.clj',
2015-10-31 13:56:27 -04:00
[
[projNameHyphRx, projNameHyph]
]
2015-10-31 11:35:07 -04:00
2015-11-28 21:22:03 +01:00
exec "rm -rf resources"
2015-09-18 23:10:03 -04:00
corePath = "src/#{projNameUs}/core.clj"
2015-08-22 11:06:30 -04:00
fs.unlinkSync corePath
2015-11-20 22:43:47 +01:00
handlersPath = "src/#{projNameUs}/handlers.cljs"
subsPath = "src/#{projNameUs}/subs.cljs"
exec "cp #{resources}cljs/handlers.cljs #{handlersPath}"
exec "cp #{resources}cljs/subs.cljs #{subsPath}"
edit handlersPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName]]
edit subsPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName]]
fs.mkdirSync 'src/cljsjs'
exec "echo '(ns cljsjs.react)' > src/cljsjs/react.cljs"
fs.mkdirSync "src/#{projNameUs}/android"
fs.mkdirSync "src/#{projNameUs}/ios"
coreAndroidPath = "src/#{projNameUs}/android/core.cljs"
coreIosPath = "src/#{projNameUs}/ios/core.cljs"
exec "cp #{resources}cljs/core.cljs #{coreAndroidPath}"
edit coreAndroidPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName], [platformRx, "android"]]
exec "cp #{resources}cljs/core.cljs #{coreIosPath}"
edit coreIosPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName], [platformRx, "ios"]]
fs.mkdirSync "env"
fs.mkdirSync "env/dev"
fs.mkdirSync "env/dev/env"
fs.mkdirSync "env/dev/env/ios"
fs.mkdirSync "env/dev/env/android"
fs.mkdirSync "env/prod"
fs.mkdirSync "env/prod/env"
fs.mkdirSync "env/prod/env/ios"
fs.mkdirSync "env/prod/env/android"
copyDevEnvironmentFiles(projNameHyph, projName, "localhost")
copyProdEnvironmentFiles(projNameHyph, projName)
exec "cp -r #{resources}images ."
log 'Creating React Native skeleton. Relax, this takes a while...'
2015-10-04 17:54:48 -04:00
fs.writeFileSync 'package.json', JSON.stringify
name: projName
version: '0.0.1'
private: true
scripts:
start: 'node_modules/react-native/packager/packager.sh'
dependencies:
'react-native': rnVersion
, null, 2
exec 'npm i'
fs.unlinkSync '.gitignore'
exec "
node -e
2015-11-22 18:17:45 +01:00
\"require('react-native/local-cli/cli').init('.', '#{projName}')\"
"
2015-11-20 22:43:47 +01:00
generateConfig projName
copyFigwheelBridge(projNameUs)
2015-08-22 13:30:42 -04:00
log 'Compiling ClojureScript'
exec 'lein prod-build'
2015-10-04 19:05:04 -04:00
log ''
log 'To get started with your new app, first cd into its directory:', 'yellow'
2015-09-18 23:10:03 -04:00
log "cd #{projNameHyph}", 'inverse'
2015-10-04 19:05:04 -04:00
log ''
log 'Open iOS app in xcode and run it:' , 'yellow'
log 're-natal xcode', 'inverse'
log ''
log 'To use figwheel type:' , 'yellow'
log 're-natal use-figwheel', 'inverse'
log 'lein figwheel ios', 'inverse'
log ''
log 'Reload the app in simulator'
2015-10-04 19:05:04 -04:00
log ''
2015-08-25 21:44:11 -04:00
log 'At the REPL prompt type this:', 'yellow'
2015-11-21 12:01:44 +01:00
log "(in-ns '#{projNameHyph}.ios.core)", 'inverse'
2015-10-04 19:05:04 -04:00
log ''
2015-08-25 21:44:11 -04:00
log 'Changes you make via the REPL or by changing your .cljs files should appear live.', 'yellow'
2015-10-04 19:05:04 -04:00
log ''
2015-08-25 21:44:11 -04:00
log 'Try this command as an example:', 'yellow'
2015-11-21 12:01:44 +01:00
log sampleCommand, 'inverse'
log ''
2015-08-25 23:48:43 -04:00
log '✔ Done', 'bgMagenta'
log ''
2015-08-25 21:44:11 -04:00
2015-09-18 23:05:39 -04:00
catch {message}
logErr \
if message.match /type.+lein/i
2015-10-03 15:48:28 -04:00
'Leiningen is required (http://leiningen.org)'
else if message.match /type.+watchman/i
2015-10-03 15:48:28 -04:00
'Watchman is required (https://facebook.github.io/watchman)'
else if message.match /type.+xcodebuild/i
2015-10-04 19:33:37 -04:00
'Xcode Command Line Tools are required'
else if message.match /npm/i
"npm install failed. This may be a network issue. Check #{projNameHyph}/npm-debug.log for details."
2015-09-18 23:05:39 -04:00
else
message
2015-09-19 14:47:28 -04:00
launch = ({name, device}) ->
unless device in getDeviceUuids()
log 'Device ID not available, defaulting to iPhone 6 simulator', 'yellow'
{device} = generateConfig name
try
fs.statSync 'node_modules'
catch
logErr 'Dependencies are missing. Something went horribly wrong...'
log 'Compiling ClojureScript'
exec 'lein prod-build'
2015-10-04 17:54:48 -04:00
log 'Compiling Xcode project'
try
2015-10-04 15:32:23 -04:00
exec "
xcodebuild
-project ios/#{name}.xcodeproj
2015-10-04 15:32:23 -04:00
-scheme #{name}
-destination platform='iOS Simulator',OS=latest,id='#{device}'
2015-10-15 21:36:58 -04:00
test
2015-10-04 15:32:23 -04:00
"
log 'Launching simulator'
2015-10-04 15:32:23 -04:00
exec "xcrun simctl launch #{device} #{getBundleId name}"
catch {message}
logErr message
2015-09-19 14:47:28 -04:00
openXcode = (name) ->
try
exec "open ios/#{name}.xcodeproj"
2015-09-19 14:47:28 -04:00
catch {message}
logErr \
if message.match /ENOENT/i
"""
Cannot find #{name}.xcodeproj in ios.
2015-09-19 14:47:28 -04:00
Run this command from your project's root directory.
"""
else if message.match /EACCES/i
"Invalid permissions for opening #{name}.xcodeproj in ios"
2015-09-19 14:47:28 -04:00
else
message
2015-08-25 23:48:43 -04:00
2015-09-19 15:35:00 -04:00
getDeviceList = ->
try
2015-10-04 15:32:23 -04:00
exec 'xcrun instruments -s devices', true
2015-09-19 15:35:00 -04:00
.toString()
.split '\n'
.filter (line) -> /^i/.test line
catch {message}
logErr 'Device listing failed: ' + message
getDeviceUuids = ->
getDeviceList().map (line) -> line.match(/\[(.+)\]/)[1]
generateDevScripts = (devHost) ->
try
projName = readConfig().name
projNameHyph = projName.replace(camelRx, '$1-$2').toLowerCase()
log 'Cleaning...'
exec 'lein clean'
fs.writeFileSync 'index.ios.js', "require('figwheel-bridge').start('" + projName + "','ios', '" + devHost + "');"
log 'index.ios.js was regenerated'
fs.writeFileSync 'index.android.js', "require('figwheel-bridge').start('" + projName + "','android', '" + devHost + "');"
log 'index.android.js was regenerated'
copyDevEnvironmentFiles(projNameHyph, projName, devHost)
log 'Dev server host: ' + devHost
catch {message}
logErr \
if message.match /EACCES/i
'Invalid write permissions for creating development scripts'
else
message
doUpgrade = (config) ->
projName = config.name;
projNameHyph = projName.replace(camelRx, '$1-$2').toLowerCase()
projNameUs = toUnderscored projName
copyDevEnvironmentFiles(projNameHyph, projName, "localhost")
copyProdEnvironmentFiles(projNameHyph, projName)
log 'upgraded files in env/'
copyFigwheelBridge(projNameUs)
log 'upgraded figwheel-bridge.js'
2015-10-04 18:53:54 -04:00
cli._name = 're-natal'
2015-10-03 21:18:26 -04:00
cli.version pkgJson.version
2015-08-25 23:48:43 -04:00
2015-09-19 13:51:11 -04:00
cli.command 'init <name>'
.description 'create a new ClojureScript React Native project'
2015-11-21 12:01:44 +01:00
.action (name) ->
2015-09-19 13:51:11 -04:00
if typeof name isnt 'string'
logErr '''
re-natal init requires a project name as the first argument.
2015-09-19 13:51:11 -04:00
e.g.
re-natal init HelloWorld
2015-09-19 13:51:11 -04:00
'''
2015-08-25 23:48:43 -04:00
2015-11-21 12:01:44 +01:00
ensureFreePort -> init name
2015-09-19 13:51:11 -04:00
2015-10-04 15:19:04 -04:00
2015-09-19 15:35:00 -04:00
cli.command 'launch'
2015-11-21 12:01:44 +01:00
.description 'compile project and run in iOS simulator'
2015-09-19 15:35:00 -04:00
.action ->
ensureFreePort -> launch readConfig()
2015-09-19 15:35:00 -04:00
cli.command 'upgrade'
.description 'upgrades project files to current installed version of re-natal (the upgrade of re-natal itself is done via npm)'
.action ->
doUpgrade readConfig()
2015-09-19 15:35:00 -04:00
cli.command 'listdevices'
2015-10-29 23:57:09 -04:00
.description 'list available simulator devices by index'
2015-09-19 15:35:00 -04:00
.action ->
console.log (getDeviceList()
2015-10-03 21:22:34 -04:00
.map (line, i) -> "#{i}\t#{line.replace /\[.+\]/, ''}"
2015-09-19 15:35:00 -04:00
.join '\n')
2015-10-03 22:36:46 -04:00
cli.command 'setdevice <index>'
2015-10-29 23:57:09 -04:00
.description 'choose simulator device by index'
2015-10-03 22:36:46 -04:00
.action (index) ->
unless device = getDeviceList()[parseInt index, 10]
logErr 'Invalid device index. Run re-natal listdevices for valid indexes.'
2015-10-03 22:36:46 -04:00
config = readConfig()
config.device = pluckUuid device
writeConfig config
2015-10-04 15:19:04 -04:00
cli.command 'xcode'
2015-10-29 23:57:09 -04:00
.description 'open Xcode project'
2015-10-04 15:19:04 -04:00
.action ->
openXcode readConfig().name
cli.command 'deps'
.description 'install all dependencies for the project'
.action ->
try
log 'Installing npm packages'
exec 'npm i'
catch {message}
logErr message
cli.command 'use-figwheel'
.description 'generate index.ios.js and index.android.js for development with figwheel'
.option "-H, --host [host or IP address}]", 'specify server host (default localhost)', "localhost"
.action (cmd) ->
generateDevScripts(cmd.host)
2015-10-03 22:57:22 -04:00
cli.on '*', (command) ->
logErr "unknown command #{command[0]}. See re-natal --help for valid commands"
2015-10-03 22:57:22 -04:00
2015-09-19 13:51:11 -04:00
unless semver.satisfies process.version[1...], nodeVersion
logErr """
Re-Natal requires Node.js version #{nodeVersion}
You have #{process.version[1...]}
"""
if process.argv.length <= 2
cli.outputHelp()
else
cli.parse process.argv