blob: d534da6de77d7575811fff6fd86ed6c694615bc0 [file] [log] [blame]
var util = require('util'),
f = util.format,
EventEmitter = require('events').EventEmitter,
path = require('path'),
uuid = require('node-uuid'),
fork = require('child_process').fork,
pbxWriter = require('./pbxWriter'),
pbxFile = require('./pbxFile'),
fs = require('fs'),
parser = require('./parser/pbxproj'),
COMMENT_KEY = /_comment$/
function pbxProject(filename) {
if (!(this instanceof pbxProject))
return new pbxProject(filename);
this.filepath = path.resolve(filename)
}
util.inherits(pbxProject, EventEmitter)
pbxProject.prototype.parse = function (cb) {
var worker = fork(__dirname + '/parseJob.js', [this.filepath])
worker.on('message', function (msg) {
if (msg.name == 'SyntaxError' || msg.code) {
this.emit('error', msg);
} else {
this.hash = msg;
this.emit('end', null, msg)
}
}.bind(this));
if (cb) {
this.on('error', cb);
this.on('end', cb);
}
return this;
}
pbxProject.prototype.parseSync = function () {
var file_contents = fs.readFileSync(this.filepath, 'utf-8');
this.hash = parser.parse(file_contents);
return this;
}
pbxProject.prototype.writeSync = function () {
this.writer = new pbxWriter(this.hash);
return this.writer.writeSync();
}
pbxProject.prototype.allUuids = function () {
var sections = this.hash.project.objects,
uuids = [],
section;
for (key in sections) {
section = sections[key]
uuids = uuids.concat(Object.keys(section))
}
uuids = uuids.filter(function (str) {
return !COMMENT_KEY.test(str) && str.length == 24;
});
return uuids;
}
pbxProject.prototype.generateUuid = function () {
var id = uuid.v4()
.replace(/-/g,'')
.substr(0,24)
.toUpperCase()
if (this.allUuids().indexOf(id) >= 0) {
return this.generateUuid();
} else {
return id;
}
}
pbxProject.prototype.addPluginFile = function (path, opt) {
var file = new pbxFile(path, opt);
file.plugin = true; // durr
correctForPluginsPath(file, this);
// null is better for early errors
if (this.hasFile(file.path)) return null;
file.fileRef = this.generateUuid();
this.addToPbxFileReferenceSection(file); // PBXFileReference
this.addToPluginsPbxGroup(file); // PBXGroup
return file;
}
pbxProject.prototype.removePluginFile = function (path, opt) {
var file = new pbxFile(path, opt);
correctForPluginsPath(file, this);
this.removeFromPbxFileReferenceSection(file); // PBXFileReference
this.removeFromPluginsPbxGroup(file); // PBXGroup
return file;
}
pbxProject.prototype.addSourceFile = function (path, opt) {
var file = this.addPluginFile(path, opt);
if (!file) return false;
file.uuid = this.generateUuid();
this.addToPbxBuildFileSection(file); // PBXBuildFile
this.addToPbxSourcesBuildPhase(file); // PBXSourcesBuildPhase
return file;
}
pbxProject.prototype.removeSourceFile = function (path, opt) {
var file = this.removePluginFile(path, opt)
this.removeFromPbxBuildFileSection(file); // PBXBuildFile
this.removeFromPbxSourcesBuildPhase(file); // PBXSourcesBuildPhase
return file;
}
pbxProject.prototype.addHeaderFile = function (path, opt) {
return this.addPluginFile(path, opt)
}
pbxProject.prototype.removeHeaderFile = function (path, opt) {
return this.removePluginFile(path, opt)
}
pbxProject.prototype.addResourceFile = function (path, opt) {
opt = opt || {};
var file;
if (opt.plugin) {
file = this.addPluginFile(path, opt);
if (!file) return false;
} else {
file = new pbxFile(path, opt);
if (this.hasFile(file.path)) return false;
}
file.uuid = this.generateUuid();
if (!opt.plugin) {
correctForResourcesPath(file, this);
file.fileRef = this.generateUuid();
}
this.addToPbxBuildFileSection(file); // PBXBuildFile
this.addToPbxResourcesBuildPhase(file); // PBXResourcesBuildPhase
if (!opt.plugin) {
this.addToPbxFileReferenceSection(file); // PBXFileReference
this.addToResourcesPbxGroup(file); // PBXGroup
}
return file;
}
pbxProject.prototype.removeResourceFile = function (path, opt) {
var file = new pbxFile(path, opt);
correctForResourcesPath(file, this);
this.removeFromPbxBuildFileSection(file); // PBXBuildFile
this.removeFromPbxFileReferenceSection(file); // PBXFileReference
this.removeFromResourcesPbxGroup(file); // PBXGroup
this.removeFromPbxResourcesBuildPhase(file); // PBXResourcesBuildPhase
return file;
}
pbxProject.prototype.addFramework = function (path, opt) {
var file = new pbxFile(path, opt);
// catch duplicates
if (this.hasFile(file.path)) return false;
file.uuid = this.generateUuid();
file.fileRef = this.generateUuid();
this.addToPbxBuildFileSection(file); // PBXBuildFile
this.addToPbxFileReferenceSection(file); // PBXFileReference
this.addToFrameworksPbxGroup(file); // PBXGroup
this.addToPbxFrameworksBuildPhase(file); // PBXFrameworksBuildPhase
return file;
}
pbxProject.prototype.removeFramework = function (path, opt) {
var file = new pbxFile(path, opt);
this.removeFromPbxBuildFileSection(file); // PBXBuildFile
this.removeFromPbxFileReferenceSection(file); // PBXFileReference
this.removeFromFrameworksPbxGroup(file); // PBXGroup
this.removeFromPbxFrameworksBuildPhase(file); // PBXFrameworksBuildPhase
return file;
}
pbxProject.prototype.addStaticLibrary = function (path, opt) {
opt = opt || {};
var file;
if (opt.plugin) {
file = this.addPluginFile(path, opt);
if (!file) return false;
} else {
file = new pbxFile(path, opt);
if (this.hasFile(file.path)) return false;
}
file.uuid = this.generateUuid();
if (!opt.plugin) {
file.fileRef = this.generateUuid();
this.addToPbxFileReferenceSection(file); // PBXFileReference
}
this.addToPbxBuildFileSection(file); // PBXBuildFile
this.addToPbxFrameworksBuildPhase(file); // PBXFrameworksBuildPhase
this.addToLibrarySearchPaths(file); // make sure it gets built!
return file;
}
// helper addition functions
pbxProject.prototype.addToPbxBuildFileSection = function (file) {
var commentKey = f("%s_comment", file.uuid);
this.pbxBuildFileSection()[file.uuid] = pbxBuildFileObj(file);
this.pbxBuildFileSection()[commentKey] = pbxBuildFileComment(file);
}
pbxProject.prototype.removeFromPbxBuildFileSection = function (file) {
var uuid;
for(uuid in this.pbxBuildFileSection()) {
if(this.pbxBuildFileSection()[uuid].fileRef_comment == file.basename) {
file.uuid = uuid;
delete this.pbxBuildFileSection()[uuid];
}
}
var commentKey = f("%s_comment", file.uuid);
delete this.pbxBuildFileSection()[commentKey];
}
pbxProject.prototype.addToPbxFileReferenceSection = function (file) {
var commentKey = f("%s_comment", file.fileRef);
this.pbxFileReferenceSection()[file.fileRef] = pbxFileReferenceObj(file);
this.pbxFileReferenceSection()[commentKey] = pbxFileReferenceComment(file);
}
pbxProject.prototype.removeFromPbxFileReferenceSection = function (file) {
var i;
var refObj = pbxFileReferenceObj(file);
for(i in this.pbxFileReferenceSection()) {
if(this.pbxFileReferenceSection()[i].name == refObj.name ||
this.pbxFileReferenceSection()[i].path == refObj.path) {
file.fileRef = file.uuid = i;
delete this.pbxFileReferenceSection()[i];
break;
}
}
var commentKey = f("%s_comment", file.fileRef);
if(this.pbxFileReferenceSection()[commentKey] != undefined) {
delete this.pbxFileReferenceSection()[commentKey];
}
return file;
}
pbxProject.prototype.addToPluginsPbxGroup = function (file) {
var pluginsGroup = this.pbxGroupByName('Plugins');
pluginsGroup.children.push(pbxGroupChild(file));
}
pbxProject.prototype.removeFromPluginsPbxGroup = function (file) {
var pluginsGroupChildren = this.pbxGroupByName('Plugins').children, i;
for(i in pluginsGroupChildren) {
if(pbxGroupChild(file).value == pluginsGroupChildren[i].value &&
pbxGroupChild(file).comment == pluginsGroupChildren[i].comment) {
pluginsGroupChildren.splice(i, 1);
break;
}
}
}
pbxProject.prototype.addToResourcesPbxGroup = function (file) {
var pluginsGroup = this.pbxGroupByName('Resources');
pluginsGroup.children.push(pbxGroupChild(file));
}
pbxProject.prototype.removeFromResourcesPbxGroup = function (file) {
var pluginsGroupChildren = this.pbxGroupByName('Resources').children, i;
for(i in pluginsGroupChildren) {
if(pbxGroupChild(file).value == pluginsGroupChildren[i].value &&
pbxGroupChild(file).comment == pluginsGroupChildren[i].comment) {
pluginsGroupChildren.splice(i, 1);
break;
}
}
}
pbxProject.prototype.addToFrameworksPbxGroup = function (file) {
var pluginsGroup = this.pbxGroupByName('Frameworks');
pluginsGroup.children.push(pbxGroupChild(file));
}
pbxProject.prototype.removeFromFrameworksPbxGroup = function (file) {
var pluginsGroupChildren = this.pbxGroupByName('Frameworks').children;
for(i in pluginsGroupChildren) {
if(pbxGroupChild(file).value == pluginsGroupChildren[i].value &&
pbxGroupChild(file).comment == pluginsGroupChildren[i].comment) {
pluginsGroupChildren.splice(i, 1);
break;
}
}
}
pbxProject.prototype.addToPbxSourcesBuildPhase = function (file) {
var sources = this.pbxSourcesBuildPhaseObj();
sources.files.push(pbxBuildPhaseObj(file));
}
pbxProject.prototype.removeFromPbxSourcesBuildPhase = function (file) {
var sources = this.pbxSourcesBuildPhaseObj(), i;
for(i in sources.files) {
if(sources.files[i].comment == longComment(file)) {
sources.files.splice(i, 1);
break;
}
}
}
pbxProject.prototype.addToPbxResourcesBuildPhase = function (file) {
var sources = this.pbxResourcesBuildPhaseObj();
sources.files.push(pbxBuildPhaseObj(file));
}
pbxProject.prototype.removeFromPbxResourcesBuildPhase = function (file) {
var sources = this.pbxResourcesBuildPhaseObj(), i;
for(i in sources.files) {
if(sources.files[i].comment == longComment(file)) {
sources.files.splice(i, 1);
break;
}
}
}
pbxProject.prototype.addToPbxFrameworksBuildPhase = function (file) {
var sources = this.pbxFrameworksBuildPhaseObj();
sources.files.push(pbxBuildPhaseObj(file));
}
pbxProject.prototype.removeFromPbxFrameworksBuildPhase = function (file) {
var sources = this.pbxFrameworksBuildPhaseObj();
for(i in sources.files) {
if(sources.files[i].comment == longComment(file)) {
sources.files.splice(i, 1);
break;
}
}
}
// helper access functions
pbxProject.prototype.pbxBuildFileSection = function () {
return this.hash.project.objects['PBXBuildFile'];
}
pbxProject.prototype.pbxXCBuildConfigurationSection = function () {
return this.hash.project.objects['XCBuildConfiguration'];
}
pbxProject.prototype.pbxFileReferenceSection = function () {
return this.hash.project.objects['PBXFileReference'];
}
pbxProject.prototype.pbxGroupByName = function (name) {
var groups = this.hash.project.objects['PBXGroup'],
key, groupKey;
for (key in groups) {
// only look for comments
if (!COMMENT_KEY.test(key)) continue;
if (groups[key] == name) {
groupKey = key.split(COMMENT_KEY)[0];
return groups[groupKey];
}
}
return null;
}
pbxProject.prototype.pbxSourcesBuildPhaseObj = function () {
return this.buildPhaseObject('PBXSourcesBuildPhase', 'Sources');
}
pbxProject.prototype.pbxResourcesBuildPhaseObj = function () {
return this.buildPhaseObject('PBXResourcesBuildPhase', 'Resources');
}
pbxProject.prototype.pbxFrameworksBuildPhaseObj = function () {
return this.buildPhaseObject('PBXFrameworksBuildPhase', 'Frameworks');
}
pbxProject.prototype.buildPhaseObject = function (name, group) {
var section = this.hash.project.objects[name],
obj, sectionKey, key;
for (key in section) {
// only look for comments
if (!COMMENT_KEY.test(key)) continue;
if (section[key] == group) {
sectionKey = key.split(COMMENT_KEY)[0];
return section[sectionKey];
}
}
return null;
}
pbxProject.prototype.updateProductName = function(name) {
var config = this.pbxXCBuildConfigurationSection();
propReplace(config, 'PRODUCT_NAME', '"' + name + '"');
}
pbxProject.prototype.removeFromLibrarySearchPaths = function (file) {
var configurations = nonComments(this.pbxXCBuildConfigurationSection()),
INHERITED = '"$(inherited)"',
SEARCH_PATHS = 'LIBRARY_SEARCH_PATHS',
config, buildSettings, searchPaths;
var new_path = searchPathForFile(file, this);
for (config in configurations) {
buildSettings = configurations[config].buildSettings;
if (unquote(buildSettings['PRODUCT_NAME']) != this.productName)
continue;
if (buildSettings[SEARCH_PATHS]) {
var matches = buildSettings[SEARCH_PATHS].filter(function(p) {
return p.indexOf(new_path) > -1;
});
matches.forEach(function(m) {
var idx = buildSettings[SEARCH_PATHS].indexOf(m);
buildSettings[SEARCH_PATHS].splice(idx, 1);
});
}
}
}
pbxProject.prototype.addToLibrarySearchPaths = function (file) {
var configurations = nonComments(this.pbxXCBuildConfigurationSection()),
INHERITED = '"$(inherited)"',
config, buildSettings, searchPaths;
for (config in configurations) {
buildSettings = configurations[config].buildSettings;
if (unquote(buildSettings['PRODUCT_NAME']) != this.productName)
continue;
if (!buildSettings['LIBRARY_SEARCH_PATHS']) {
buildSettings['LIBRARY_SEARCH_PATHS'] = [INHERITED];
}
buildSettings['LIBRARY_SEARCH_PATHS'].push(searchPathForFile(file, this));
}
}
// a JS getter. hmmm
pbxProject.prototype.__defineGetter__("productName", function () {
var configurations = nonComments(this.pbxXCBuildConfigurationSection()),
config, productName;
for (config in configurations) {
productName = configurations[config].buildSettings['PRODUCT_NAME'];
if (productName) {
return unquote(productName);
}
}
});
// check if file is present
pbxProject.prototype.hasFile = function (filePath) {
var files = nonComments(this.pbxFileReferenceSection()),
file, id;
for (id in files) {
file = files[id];
if (file.path == filePath || file.path == ('"' + filePath + '"')) {
return true;
}
}
return false;
}
// helper recursive prop search+replace
function propReplace(obj, prop, value) {
for (var p in obj) {
if (obj.hasOwnProperty(p)) {
if (typeof obj[p] == 'object') {
propReplace(obj[p], prop, value);
} else if (p == prop) {
obj[p] = value;
}
}
}
}
// helper object creation functions
function pbxBuildFileObj(file) {
var obj = Object.create(null);
obj.isa = 'PBXBuildFile';
obj.fileRef = file.fileRef;
obj.fileRef_comment = file.basename;
if (file.settings) obj.settings = file.settings;
return obj;
}
function pbxFileReferenceObj(file) {
var obj = Object.create(null);
obj.isa = 'PBXFileReference';
obj.lastKnownFileType = file.lastType;
obj.name = "\"" + file.basename + "\"";
obj.path = "\"" + file.path + "\"";
obj.sourceTree = file.sourceTree;
if (file.fileEncoding)
obj.fileEncoding = file.fileEncoding;
return obj;
}
function pbxGroupChild(file) {
var obj = Object.create(null);
obj.value = file.fileRef;
obj.comment = file.basename;
return obj;
}
function pbxBuildPhaseObj(file) {
var obj = Object.create(null);
obj.value = file.uuid;
obj.comment = longComment(file);
return obj;
}
function pbxBuildFileComment(file) {
return longComment(file);
}
function pbxFileReferenceComment(file) {
return file.basename;
}
function longComment(file) {
return f("%s in %s", file.basename, file.group);
}
// respect <group> path
function correctForPluginsPath(file, project) {
var r_plugin_dir = /^Plugins\//;
if (project.pbxGroupByName('Plugins').path)
file.path = file.path.replace(r_plugin_dir, '');
return file;
}
function correctForResourcesPath(file, project) {
var r_resources_dir = /^Resources\//;
if (project.pbxGroupByName('Resources').path)
file.path = file.path.replace(r_resources_dir, '');
return file;
}
function searchPathForFile(file, proj) {
var pluginsPath = proj.pbxGroupByName('Plugins').path,
fileDir = path.dirname(file.path);
if (fileDir == '.') {
fileDir = '';
} else {
fileDir = '/' + fileDir;
}
if (file.plugin && pluginsPath) {
return '"\\"$(SRCROOT)/' + unquote(pluginsPath) + '\\""';
} else {
return '"\\"$(SRCROOT)/' + proj.productName + fileDir + '\\""';
}
}
function nonComments(obj) {
var keys = Object.keys(obj),
newObj = {}, i = 0;
for (i; i < keys.length; i++) {
if (!COMMENT_KEY.test(keys[i])) {
newObj[keys[i]] = obj[keys[i]];
}
}
return newObj;
}
function unquote(str) {
if (str) return str.replace(/^"(.*)"$/, "$1");
}
module.exports = pbxProject;