2015-11-20 20:44:36 +01:00
|
|
|
# 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'
|
2016-01-29 20:25:00 +01:00
|
|
|
ckDeps = require 'check-dependencies'
|
2015-10-04 18:53:54 -04:00
|
|
|
pkgJson = require __dirname + '/package.json'
|
2015-08-21 23:45:42 -04:00
|
|
|
|
2015-10-03 15:36:24 -04:00
|
|
|
nodeVersion = pkgJson.engines.node
|
2015-08-21 23:45:42 -04:00
|
|
|
resources = __dirname + '/resources/'
|
2015-11-11 23:12:04 -05:00
|
|
|
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
|
2015-11-27 23:57:14 +01:00
|
|
|
projNameUsRx = /\$PROJECT_NAME_UNDERSCORED\$/g
|
|
|
|
platformRx = /\$PLATFORM\$/g
|
2015-12-06 20:22:53 +01:00
|
|
|
devHostRx = /\$DEV_HOST\$/g
|
2016-01-24 11:21:14 +01:00
|
|
|
figwheelUrlRx = /ws:\/\/[0-9a-zA-Z\.]*:/g
|
2016-01-31 12:16:16 +01:00
|
|
|
rnVersion = '0.19.0'
|
2015-10-17 15:10:31 -04:00
|
|
|
rnPackagerPort = 8081
|
2015-11-20 20:44:36 +01:00
|
|
|
process.title = 're-natal'
|
2015-11-22 10:57:51 +01:00
|
|
|
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
|
|
|
|
|
|
|
|
2015-10-03 22:35:19 -04:00
|
|
|
pluckUuid = (line) ->
|
|
|
|
line.match(/\[(.+)\]/)[1]
|
|
|
|
|
2015-12-31 15:55:43 +01:00
|
|
|
mkdirSync = (path) ->
|
|
|
|
try
|
|
|
|
fs.mkdirSync(path)
|
|
|
|
catch {message}
|
|
|
|
if not message.match /EEXIST/i
|
|
|
|
throw new Error "Could not create dir #{path}: #{message}" ;
|
|
|
|
|
2015-10-03 22:35:19 -04:00
|
|
|
|
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
|
|
|
|
|
|
|
|
|
2015-10-17 00:37:20 -04:00
|
|
|
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
|
|
|
|
|
|
|
|
|
2015-10-17 15:20:02 -04:00
|
|
|
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-12-09 22:32:25 +01:00
|
|
|
ensureXcode = (cb) ->
|
|
|
|
try
|
|
|
|
exec 'type xcodebuild'
|
|
|
|
config = readConfig()
|
|
|
|
unless config.device?
|
|
|
|
config.device = getUuidForDevice 'iPhone 6'
|
|
|
|
writeConfig config
|
|
|
|
cb();
|
|
|
|
catch {message}
|
|
|
|
if message.match /type.+xcodebuild/i
|
|
|
|
logErr 'Xcode Command Line Tools are required'
|
2015-10-17 15:20:02 -04:00
|
|
|
|
2015-11-14 18:18:19 -05:00
|
|
|
generateConfig = (name) ->
|
2015-11-20 20:44:36 +01:00
|
|
|
log 'Creating Re-Natal config'
|
2015-11-14 18:18:19 -05:00
|
|
|
config =
|
|
|
|
name: name
|
2016-01-24 13:46:34 +01:00
|
|
|
androidHost: "localhost"
|
2016-01-04 23:10:13 +01:00
|
|
|
modules: []
|
2016-01-17 13:32:42 +01:00
|
|
|
imageDirs: ["images"]
|
2015-11-14 18:18:19 -05:00
|
|
|
|
|
|
|
writeConfig config
|
|
|
|
config
|
|
|
|
|
|
|
|
|
2015-09-19 14:02:28 -04:00
|
|
|
writeConfig = (config) ->
|
|
|
|
try
|
2015-11-20 20:44:36 +01:00
|
|
|
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
|
2015-11-20 20:44:36 +01:00
|
|
|
'Invalid write permissions for creating .re-natal config file'
|
2015-09-19 14:02:28 -04:00
|
|
|
else
|
|
|
|
message
|
|
|
|
|
2016-01-29 20:25:00 +01:00
|
|
|
verifyConfig = (config) ->
|
|
|
|
if !config.androidHost? || !config.modules? || !config.imageDirs?
|
|
|
|
throw new Error 're-natal project needs to be upgraded, please run: re-natal upgrade'
|
|
|
|
config
|
2015-09-19 14:02:28 -04:00
|
|
|
|
2016-01-29 20:25:00 +01:00
|
|
|
readConfig = (verify = true)->
|
2015-09-19 14:21:19 -04:00
|
|
|
try
|
2016-01-29 20:25:00 +01:00
|
|
|
config = JSON.parse readFile '.re-natal'
|
|
|
|
if (verify)
|
|
|
|
verifyConfig(config)
|
|
|
|
else
|
|
|
|
config
|
2015-09-19 14:21:19 -04:00
|
|
|
catch {message}
|
|
|
|
logErr \
|
|
|
|
if message.match /ENOENT/i
|
2015-11-20 20:44:36 +01:00
|
|
|
'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
|
2015-11-20 20:44:36 +01:00
|
|
|
'No read permissions for .re-natal'
|
2015-09-19 14:21:19 -04:00
|
|
|
else if message.match /Unexpected/i
|
2015-11-20 20:44:36 +01:00
|
|
|
'.re-natal contains malformed JSON'
|
2015-09-19 14:21:19 -04:00
|
|
|
else
|
|
|
|
message
|
|
|
|
|
|
|
|
|
2015-10-04 15:11:54 -04:00
|
|
|
getBundleId = (name) ->
|
2015-10-03 20:35:46 -04:00
|
|
|
try
|
2015-11-27 23:57:14 +01:00
|
|
|
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
|
|
|
|
2015-11-27 23:57:14 +01: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
|
|
|
|
|
2016-01-17 13:32:42 +01:00
|
|
|
scanImageDir = (dir) ->
|
|
|
|
fnames = fs.readdirSync(dir)
|
|
|
|
.map (fname) -> "#{dir}/#{fname}"
|
|
|
|
.filter (path) -> fs.statSync(path).isFile()
|
|
|
|
.map (path) -> path.replace /@2x|@3x/i, ''
|
|
|
|
.filter (v, idx, slf) -> slf.indexOf(v) == idx
|
|
|
|
|
|
|
|
dirs = fs.readdirSync(dir)
|
|
|
|
.map (fname) -> "#{dir}/#{fname}"
|
|
|
|
.filter (path) -> fs.statSync(path).isDirectory()
|
|
|
|
|
|
|
|
fnames.concat scanImages(dirs)
|
|
|
|
|
|
|
|
scanImages = (dirs) ->
|
|
|
|
imgs = []
|
|
|
|
for dir in dirs
|
|
|
|
imgs = imgs.concat(scanImageDir(dir));
|
|
|
|
imgs
|
2015-10-03 20:35:46 -04:00
|
|
|
|
2016-01-24 13:46:34 +01:00
|
|
|
configureDevHostForAndroidDevice = (deviceType) ->
|
|
|
|
try
|
|
|
|
allowedTypes = {'real': 'localhost', 'avd': '10.0.2.2', 'genymotion': '10.0.3.2'}
|
|
|
|
devHost = allowedTypes[deviceType]
|
|
|
|
if (! devHost?)
|
|
|
|
throw new Error "Unknown android device type #{deviceType}, known types are #{Object.keys(allowedTypes)}"
|
|
|
|
log "Using host '#{devHost}' for android device type '#{deviceType}'"
|
|
|
|
config = readConfig()
|
|
|
|
config.androidHost = devHost
|
|
|
|
writeConfig(config)
|
|
|
|
catch {message}
|
|
|
|
logErr message
|
|
|
|
|
2015-12-06 20:22:53 +01:00
|
|
|
copyDevEnvironmentFiles = (projNameHyph, projName, devHost) ->
|
2015-12-31 15:55:43 +01:00
|
|
|
mkdirSync "env/dev"
|
|
|
|
mkdirSync "env/dev/env"
|
|
|
|
mkdirSync "env/dev/env/ios"
|
|
|
|
mkdirSync "env/dev/env/android"
|
|
|
|
|
2015-12-14 23:02:40 +01:00
|
|
|
userNsPath = "env/dev/user.clj"
|
|
|
|
exec "cp #{resources}user.clj #{userNsPath}"
|
|
|
|
|
2015-11-29 22:36:49 +01:00
|
|
|
mainIosDevPath = "env/dev/env/ios/main.cljs"
|
|
|
|
mainAndroidDevPath = "env/dev/env/android/main.cljs"
|
|
|
|
|
|
|
|
exec "cp #{resources}cljs/main_dev.cljs #{mainIosDevPath}"
|
2015-12-06 20:22:53 +01:00
|
|
|
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]]
|
|
|
|
|
|
|
|
copyProdEnvironmentFiles = (projNameHyph, projName) ->
|
2015-12-31 15:55:43 +01:00
|
|
|
mkdirSync "env/prod"
|
|
|
|
mkdirSync "env/prod/env"
|
|
|
|
mkdirSync "env/prod/env/ios"
|
|
|
|
mkdirSync "env/prod/env/android"
|
|
|
|
|
2015-12-06 20:22:53 +01:00
|
|
|
mainIosProdPath = "env/prod/env/ios/main.cljs"
|
|
|
|
mainAndroidProdPath = "env/prod/env/android/main.cljs"
|
|
|
|
|
2015-11-29 22:36:49 +01:00
|
|
|
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"]]
|
|
|
|
|
|
|
|
copyFigwheelBridge = (projNameUs) ->
|
|
|
|
exec "cp #{resources}figwheel-bridge.js ."
|
|
|
|
edit "figwheel-bridge.js", [[projNameUsRx, projNameUs]]
|
|
|
|
|
2015-12-31 18:57:50 +01:00
|
|
|
updateGitIgnore = () ->
|
2016-01-29 21:27:34 +01:00
|
|
|
fs.appendFileSync(".gitignore", "\n# Generated by re-natal\n#\nindex.android.js\nindex.ios.js\ntarget/\n")
|
2015-12-31 18:57:50 +01:00
|
|
|
fs.appendFileSync(".gitignore", "\n# Figwheel\n#\nfigwheel_server.log")
|
|
|
|
|
2016-01-21 00:14:35 +01:00
|
|
|
patchReactNativePackager = () ->
|
|
|
|
log "Patching react-native packager to serve *.map files"
|
2016-01-21 22:27:36 +01:00
|
|
|
edit "node_modules/react-native/packager/react-packager/src/Server/index.js",
|
|
|
|
[[/match.*\.map\$\/\)/m, "match(/index\\..*\\.map$/)"]]
|
2016-01-21 00:14:35 +01:00
|
|
|
|
2015-11-21 12:01:44 +01:00
|
|
|
init = (projName) ->
|
2015-11-11 23:12:04 -05:00
|
|
|
if projName.toLowerCase() is 'react' or !projName.match validNameRx
|
|
|
|
logErr 'Invalid project name. Use an alphanumeric CamelCase name.'
|
|
|
|
|
2015-08-21 23:46:38 -04:00
|
|
|
projNameHyph = projName.replace(camelRx, '$1-$2').toLowerCase()
|
2015-10-17 00:37:20 -04:00
|
|
|
projNameUs = toUnderscored projName
|
2015-08-21 23:46:38 -04:00
|
|
|
|
|
|
|
try
|
2015-09-18 23:10:03 -04:00
|
|
|
log "Creating #{projName}", 'bgMagenta'
|
2015-08-25 21:44:11 -04:00
|
|
|
log ''
|
|
|
|
|
2015-08-22 13:45:37 -04:00
|
|
|
if fs.existsSync projNameHyph
|
2015-09-18 23:10:03 -04:00
|
|
|
throw new Error "Directory #{projNameHyph} already exists"
|
2015-08-22 13:45:37 -04:00
|
|
|
|
2015-10-04 15:32:23 -04:00
|
|
|
exec 'type lein'
|
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 20:44:36 +01:00
|
|
|
|
2015-11-20 22:43:47 +01:00
|
|
|
handlersPath = "src/#{projNameUs}/handlers.cljs"
|
|
|
|
subsPath = "src/#{projNameUs}/subs.cljs"
|
2015-12-13 22:49:54 +01:00
|
|
|
dbPath = "src/#{projNameUs}/db.cljs"
|
2015-11-28 16:20:39 +01:00
|
|
|
exec "cp #{resources}cljs/handlers.cljs #{handlersPath}"
|
|
|
|
exec "cp #{resources}cljs/subs.cljs #{subsPath}"
|
2015-12-13 22:49:54 +01:00
|
|
|
exec "cp #{resources}cljs/db.cljs #{dbPath}"
|
2015-11-20 20:44:36 +01:00
|
|
|
|
|
|
|
edit handlersPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName]]
|
|
|
|
edit subsPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName]]
|
2015-12-13 22:49:54 +01:00
|
|
|
edit dbPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName]]
|
2015-11-20 20:44:36 +01:00
|
|
|
|
|
|
|
fs.mkdirSync 'src/cljsjs'
|
|
|
|
exec "echo '(ns cljsjs.react)' > src/cljsjs/react.cljs"
|
|
|
|
|
2015-11-22 10:57:51 +01:00
|
|
|
fs.mkdirSync "src/#{projNameUs}/android"
|
|
|
|
fs.mkdirSync "src/#{projNameUs}/ios"
|
2015-11-20 20:44:36 +01:00
|
|
|
|
2015-11-22 10:57:51 +01:00
|
|
|
coreAndroidPath = "src/#{projNameUs}/android/core.cljs"
|
|
|
|
coreIosPath = "src/#{projNameUs}/ios/core.cljs"
|
2015-11-20 20:44:36 +01:00
|
|
|
|
2015-11-28 16:20:39 +01:00
|
|
|
exec "cp #{resources}cljs/core.cljs #{coreAndroidPath}"
|
2015-11-27 23:57:14 +01:00
|
|
|
edit coreAndroidPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName], [platformRx, "android"]]
|
|
|
|
|
2015-11-28 16:20:39 +01:00
|
|
|
exec "cp #{resources}cljs/core.cljs #{coreIosPath}"
|
2015-11-27 23:57:14 +01:00
|
|
|
edit coreIosPath, [[projNameHyphRx, projNameHyph], [projNameRx, projName], [platformRx, "ios"]]
|
|
|
|
|
2015-11-28 16:20:39 +01:00
|
|
|
fs.mkdirSync "env"
|
|
|
|
|
2015-12-06 20:22:53 +01:00
|
|
|
copyDevEnvironmentFiles(projNameHyph, projName, "localhost")
|
|
|
|
copyProdEnvironmentFiles(projNameHyph, projName)
|
2015-11-28 16:20:39 +01:00
|
|
|
|
|
|
|
exec "cp -r #{resources}images ."
|
2015-08-21 23:46:38 -04:00
|
|
|
|
2015-11-24 22:44:57 +01:00
|
|
|
log 'Creating React Native skeleton. Relax, this takes a while...'
|
2015-10-04 17:54:48 -04:00
|
|
|
|
2015-11-22 10:57:51 +01: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'
|
2015-11-24 22:44:57 +01:00
|
|
|
|
|
|
|
fs.unlinkSync '.gitignore'
|
2015-12-13 22:49:54 +01:00
|
|
|
exec "node -e
|
2015-11-22 18:17:45 +01:00
|
|
|
\"require('react-native/local-cli/cli').init('.', '#{projName}')\"
|
2015-11-22 10:57:51 +01:00
|
|
|
"
|
2015-11-20 22:43:47 +01:00
|
|
|
|
2016-01-21 00:14:35 +01:00
|
|
|
patchReactNativePackager();
|
|
|
|
|
2015-12-31 18:57:50 +01:00
|
|
|
updateGitIgnore()
|
|
|
|
|
2015-11-27 23:57:14 +01:00
|
|
|
generateConfig projName
|
|
|
|
|
2015-11-29 22:36:49 +01:00
|
|
|
copyFigwheelBridge(projNameUs)
|
2015-08-22 13:30:42 -04:00
|
|
|
|
2015-11-24 22:44:57 +01:00
|
|
|
log 'Compiling ClojureScript'
|
2015-11-27 23:57:14 +01:00
|
|
|
exec 'lein prod-build'
|
2015-10-04 15:11:54 -04:00
|
|
|
|
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 ''
|
2015-11-27 23:57:14 +01:00
|
|
|
log 'Open iOS app in xcode and run it:' , 'yellow'
|
|
|
|
log 're-natal xcode', 'inverse'
|
|
|
|
log ''
|
2015-12-06 12:09:28 +01:00
|
|
|
log 'To use figwheel type:' , 'yellow'
|
2015-11-27 23:57:14 +01:00
|
|
|
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'
|
2015-08-25 21:57:22 -04:00
|
|
|
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 \
|
2015-10-18 18:33:58 -04:00
|
|
|
if message.match /type.+lein/i
|
2015-10-03 15:48:28 -04:00
|
|
|
'Leiningen is required (http://leiningen.org)'
|
2015-11-14 17:51:59 -05:00
|
|
|
else if message.match /npm/i
|
2015-11-27 23:57:14 +01:00
|
|
|
"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-08-21 23:46:38 -04:00
|
|
|
|
2015-09-19 14:47:28 -04:00
|
|
|
|
2015-10-04 15:11:54 -04:00
|
|
|
launch = ({name, device}) ->
|
2015-11-14 18:20:09 -05:00
|
|
|
unless device in getDeviceUuids()
|
2015-12-07 10:37:32 -05:00
|
|
|
log 'Device ID not available, defaulting to iPhone 6 simulator', 'yellow'
|
2015-11-14 18:20:09 -05:00
|
|
|
{device} = generateConfig name
|
|
|
|
|
2015-11-14 18:21:08 -05:00
|
|
|
try
|
2015-11-27 23:57:14 +01:00
|
|
|
fs.statSync 'node_modules'
|
2015-11-14 18:21:08 -05:00
|
|
|
catch
|
2015-11-27 23:57:14 +01:00
|
|
|
logErr 'Dependencies are missing. Something went horribly wrong...'
|
2015-11-14 18:21:08 -05:00
|
|
|
|
2015-11-14 18:10:07 -05:00
|
|
|
log 'Compiling ClojureScript'
|
2015-11-27 23:57:14 +01:00
|
|
|
exec 'lein prod-build'
|
2015-11-14 18:10:07 -05:00
|
|
|
|
2015-10-04 17:54:48 -04:00
|
|
|
log 'Compiling Xcode project'
|
2015-10-04 15:11:54 -04:00
|
|
|
try
|
2015-10-04 15:32:23 -04:00
|
|
|
exec "
|
|
|
|
xcodebuild
|
2015-11-27 23:57:14 +01:00
|
|
|
-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
|
|
|
"
|
2015-10-04 15:11:54 -04:00
|
|
|
|
|
|
|
log 'Launching simulator'
|
2015-10-04 15:32:23 -04:00
|
|
|
exec "xcrun simctl launch #{device} #{getBundleId name}"
|
2015-10-04 15:11:54 -04:00
|
|
|
|
|
|
|
catch {message}
|
|
|
|
logErr message
|
|
|
|
|
2015-09-19 14:47:28 -04:00
|
|
|
openXcode = (name) ->
|
|
|
|
try
|
2015-11-27 23:57:14 +01:00
|
|
|
exec "open ios/#{name}.xcodeproj"
|
2015-09-19 14:47:28 -04:00
|
|
|
catch {message}
|
|
|
|
logErr \
|
|
|
|
if message.match /ENOENT/i
|
|
|
|
"""
|
2015-11-27 23:57:14 +01:00
|
|
|
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
|
2015-11-27 23:57:14 +01:00
|
|
|
"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
|
|
|
|
|
|
|
|
|
2015-11-14 18:20:09 -05:00
|
|
|
getDeviceUuids = ->
|
|
|
|
getDeviceList().map (line) -> line.match(/\[(.+)\]/)[1]
|
|
|
|
|
|
|
|
|
2016-01-04 23:10:13 +01:00
|
|
|
generateRequireModulesCode = (modules) ->
|
|
|
|
jsCode = "var modules={'react-native': require('react-native')};"
|
|
|
|
for m in modules
|
|
|
|
jsCode += "modules['#{m}']=require('#{m}');";
|
|
|
|
jsCode += '\n'
|
|
|
|
|
2016-01-24 11:21:14 +01:00
|
|
|
updateFigwheelUrlForAndroid= (devHost) ->
|
|
|
|
mainAndroidDevPath = "env/dev/env/android/main.cljs"
|
|
|
|
|
|
|
|
edit mainAndroidDevPath, [[figwheelUrlRx, "ws://#{devHost}:"]]
|
|
|
|
|
2016-01-24 13:46:34 +01:00
|
|
|
generateDevScripts = () ->
|
2015-11-27 23:57:14 +01:00
|
|
|
try
|
2016-01-04 23:10:13 +01:00
|
|
|
config = readConfig()
|
|
|
|
projName = config.name
|
2015-12-06 20:22:53 +01:00
|
|
|
|
2016-01-29 20:25:00 +01:00
|
|
|
depState = ckDeps.sync {install: false, verbose: false}
|
|
|
|
if (!depState.depsWereOk)
|
|
|
|
throw new Error "Missing dependencies, please run: re-natal deps"
|
|
|
|
|
2015-12-06 20:53:49 +01:00
|
|
|
log 'Cleaning...'
|
|
|
|
exec 'lein clean'
|
|
|
|
|
2016-01-17 13:32:42 +01:00
|
|
|
images = scanImages(config.imageDirs).map (fname) -> './' + fname;
|
|
|
|
modulesAndImages = config.modules.concat images;
|
|
|
|
moduleMap = generateRequireModulesCode modulesAndImages
|
2016-01-04 23:10:13 +01:00
|
|
|
|
2016-01-24 13:46:34 +01:00
|
|
|
androidDevHost = config.androidHost
|
|
|
|
|
2016-01-24 11:21:14 +01:00
|
|
|
fs.writeFileSync 'index.ios.js', "#{moduleMap}require('figwheel-bridge').withModules(modules).start('#{projName}','ios','localhost');"
|
2015-11-27 23:57:14 +01:00
|
|
|
log 'index.ios.js was regenerated'
|
2016-01-24 13:46:34 +01:00
|
|
|
fs.writeFileSync 'index.android.js', "#{moduleMap}require('figwheel-bridge').withModules(modules).start('#{projName}','android','#{androidDevHost}');"
|
2015-11-27 23:57:14 +01:00
|
|
|
log 'index.android.js was regenerated'
|
2015-12-06 20:22:53 +01:00
|
|
|
|
2016-01-24 13:46:34 +01:00
|
|
|
updateFigwheelUrlForAndroid(androidDevHost)
|
2016-01-24 11:21:14 +01:00
|
|
|
log 'Dev server host for iOS: localhost'
|
2016-01-24 13:46:34 +01:00
|
|
|
log 'Dev server host for Android: ' + androidDevHost
|
2016-01-24 11:21:14 +01:00
|
|
|
|
2015-11-27 23:57:14 +01:00
|
|
|
catch {message}
|
|
|
|
logErr \
|
|
|
|
if message.match /EACCES/i
|
|
|
|
'Invalid write permissions for creating development scripts'
|
|
|
|
else
|
|
|
|
message
|
|
|
|
|
2015-11-29 22:36:49 +01:00
|
|
|
doUpgrade = (config) ->
|
|
|
|
projName = config.name;
|
|
|
|
projNameHyph = projName.replace(camelRx, '$1-$2').toLowerCase()
|
|
|
|
projNameUs = toUnderscored projName
|
|
|
|
|
2016-01-29 20:25:00 +01:00
|
|
|
ckDeps.sync {install: true, verbose: false}
|
|
|
|
|
2016-01-21 00:14:35 +01:00
|
|
|
patchReactNativePackager()
|
|
|
|
|
2015-12-06 20:22:53 +01:00
|
|
|
copyDevEnvironmentFiles(projNameHyph, projName, "localhost")
|
|
|
|
copyProdEnvironmentFiles(projNameHyph, projName)
|
2015-11-29 22:36:49 +01:00
|
|
|
log 'upgraded files in env/'
|
|
|
|
|
|
|
|
copyFigwheelBridge(projNameUs)
|
|
|
|
log 'upgraded figwheel-bridge.js'
|
2015-10-04 18:53:54 -04:00
|
|
|
|
2016-01-05 19:41:32 +01:00
|
|
|
if (!config.modules)
|
|
|
|
config.modules = []
|
|
|
|
|
2016-01-17 13:32:42 +01:00
|
|
|
if (!config.imageDirs)
|
|
|
|
config.imageDirs = ["images"]
|
|
|
|
|
2016-01-24 13:46:34 +01:00
|
|
|
if (!config.androidHost)
|
|
|
|
config.androidHost = "localhost"
|
|
|
|
|
2016-01-05 19:41:32 +01:00
|
|
|
writeConfig(config)
|
|
|
|
log 'upgraded .re-natal'
|
|
|
|
|
2016-01-30 18:21:27 +01:00
|
|
|
edit "src/#{projNameUs}/ios/core.cljs", [[/\^:figwheel-load\s/g, ""]]
|
|
|
|
edit "src/#{projNameUs}/android/core.cljs", [[/\^:figwheel-load\s/g, ""]]
|
|
|
|
log 'upgraded core.cljs'
|
|
|
|
|
2016-01-24 11:21:14 +01:00
|
|
|
edit '.gitignore', [[/^\s*env\/dev\s*$/m, ""]]
|
2016-01-29 21:27:34 +01:00
|
|
|
gignore = readFile '.gitignore'
|
|
|
|
if (!gignore.match /^\s*target\/\s*$/m)
|
|
|
|
fs.appendFileSync(".gitignore", "\ntarget/\n")
|
2016-01-24 11:21:14 +01:00
|
|
|
log 'upgraded .gitignore'
|
|
|
|
|
2016-01-04 23:10:13 +01:00
|
|
|
useComponent = (name) ->
|
|
|
|
log "Component '#{name}' is now configured for figwheel, please re-run 'use-figwheel' command to take effect"
|
|
|
|
try
|
|
|
|
config = readConfig()
|
|
|
|
config.modules.push name
|
|
|
|
writeConfig(config)
|
|
|
|
catch {message}
|
|
|
|
logErr message
|
|
|
|
|
2015-11-20 20:44:36 +01: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>'
|
2015-10-31 11:36:32 -04:00
|
|
|
.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 '''
|
2015-11-20 20:44:36 +01:00
|
|
|
re-natal init requires a project name as the first argument.
|
2015-09-19 13:51:11 -04:00
|
|
|
e.g.
|
2015-11-20 20:44:36 +01:00
|
|
|
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 ->
|
2015-12-09 22:32:25 +01:00
|
|
|
ensureXcode ->
|
|
|
|
ensureFreePort -> launch readConfig()
|
2015-09-19 15:35:00 -04:00
|
|
|
|
2015-11-29 22:36:49 +01: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 ->
|
2016-01-29 20:25:00 +01:00
|
|
|
doUpgrade readConfig(false)
|
2015-11-29 22:36:49 +01:00
|
|
|
|
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 ->
|
2015-12-09 22:32:25 +01:00
|
|
|
ensureXcode ->
|
|
|
|
console.log (getDeviceList()
|
|
|
|
.map (line, i) -> "#{i}\t#{line.replace /\[.+\]/, ''}"
|
|
|
|
.join '\n')
|
2015-09-19 15:35:00 -04:00
|
|
|
|
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) ->
|
2015-12-09 22:32:25 +01:00
|
|
|
ensureXcode ->
|
|
|
|
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
|
|
|
|
2015-12-09 22:32:25 +01:00
|
|
|
config = readConfig()
|
|
|
|
config.device = pluckUuid device
|
|
|
|
writeConfig config
|
2015-10-03 22:36:46 -04:00
|
|
|
|
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 ->
|
2015-12-09 22:32:25 +01:00
|
|
|
ensureXcode ->
|
|
|
|
openXcode readConfig().name
|
2015-10-04 15:19:04 -04:00
|
|
|
|
2015-11-14 18:21:39 -05:00
|
|
|
cli.command 'deps'
|
|
|
|
.description 'install all dependencies for the project'
|
|
|
|
.action ->
|
2016-01-29 20:25:00 +01:00
|
|
|
ckDeps.sync {install: true, verbose: true}
|
2015-11-14 18:21:39 -05:00
|
|
|
|
2015-11-27 23:57:14 +01:00
|
|
|
cli.command 'use-figwheel'
|
|
|
|
.description 'generate index.ios.js and index.android.js for development with figwheel'
|
2016-01-24 13:46:34 +01:00
|
|
|
.action () ->
|
|
|
|
generateDevScripts()
|
|
|
|
|
|
|
|
cli.command 'use-android-device <type>'
|
|
|
|
.description 'sets up the host for android device type: \'real\' - localhost, \'avd\' - 10.0.2.2, \'genymotion\' - 10.0.3.2'
|
|
|
|
.action (type) ->
|
|
|
|
configureDevHostForAndroidDevice type
|
2015-11-14 18:21:39 -05:00
|
|
|
|
2016-01-04 23:10:13 +01:00
|
|
|
cli.command 'use-component <name>'
|
|
|
|
.description 'configures a custom component to work with figwheel. name is the value you pass to (js/require) function.'
|
|
|
|
.action (name) ->
|
|
|
|
useComponent(name)
|
|
|
|
|
2015-10-03 22:57:22 -04:00
|
|
|
cli.on '*', (command) ->
|
2015-11-20 20:44:36 +01:00
|
|
|
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
|
|
|
|
2015-10-03 15:36:24 -04:00
|
|
|
unless semver.satisfies process.version[1...], nodeVersion
|
|
|
|
logErr """
|
2015-11-20 20:44:36 +01:00
|
|
|
Re-Natal requires Node.js version #{nodeVersion}
|
2015-10-03 15:36:24 -04:00
|
|
|
You have #{process.version[1...]}
|
|
|
|
"""
|
|
|
|
|
2015-10-03 21:19:46 -04:00
|
|
|
if process.argv.length <= 2
|
|
|
|
cli.outputHelp()
|
|
|
|
else
|
|
|
|
cli.parse process.argv
|