com.codename1.maven.CN1BuildMojo Maven / Gradle / Ivy
The newest version!
package com.codename1.maven;
import com.codename1.ant.AntExecutor;
import com.codename1.ant.SortedProperties;
import com.codename1.builders.*;
import org.apache.commons.io.FileUtils;
import org.apache.commons.vfs2.FileObject;
import org.apache.commons.vfs2.FileSystemManager;
import org.apache.commons.vfs2.PatternFileSelector;
import org.apache.commons.vfs2.VFS;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.plugins.annotations.*;
import org.apache.tools.ant.taskdefs.Expand;
import org.apache.tools.ant.taskdefs.Zip;
import org.apache.tools.ant.types.FileSet;
import org.apache.tools.ant.types.ZipFileSet;
import javax.imageio.ImageIO;
import java.awt.image.BufferedImage;
import java.io.*;
import java.util.*;
import static com.codename1.maven.PathUtil.path;
/**
* Mojo that uses the CodenameOneBuildClient to send builds to the CodenameOne build server.
*
* It also supports a few local build targets, such as "ios-source", which generates an Xcode project,
* and "android-source", which generates an Android gradle project.
*/
@Mojo(name="build", requiresDependencyResolution = ResolutionScope.COMPILE_PLUS_RUNTIME,
requiresDependencyCollection = ResolutionScope.COMPILE_PLUS_RUNTIME)
@Execute(phase = LifecyclePhase.PACKAGE)
public class CN1BuildMojo extends AbstractCN1Mojo {
public static final String BUILD_TARGET_XCODE_PROJECT = Executor.BUILD_TARGET_XCODE_PROJECT;
public static final String BUILD_TARGET_ANDROID_PROJECT = Executor.BUILD_TARGET_ANDROID_PROJECT;
private String serverMustProvideKotlinVersion;
/**
* The target platform. E.g. javase, javascript, ios, android, win
*/
@Parameter(property = "codename1.platform", required = true)
private String platform;
/**
* The build target, corresponding to ANT build targets in build-template.xml. E.g. javascript,
* mac-os-x-desktop, windows-desktop, windows-device, ios-device, ios-device-release, android-device, war
*/
@Parameter(property = "codename1.buildTarget", required = true, defaultValue = "${codename1.defaultBuildTarget}")
private String buildTarget;
/**
* Flag to indicate whether to use an automated build or not.
*/
@Parameter(property = "automated", defaultValue = "false")
private boolean automated;
/**
* Flag of whether to open the xcode/android studio project.
*/
@Parameter(property = "open", defaultValue = "true")
private boolean open;
private String cn1MavenPluginVersion = "";
private String cn1MavenVersion = "";
@Override
protected void executeImpl() throws MojoExecutionException, MojoFailureException {
if ("none".equalsIgnoreCase(buildTarget)) {
getLog().info("BuildTarget is None. Skipping cn1build goal");
return;
}
cn1MavenPluginVersion = project.getProperties().getProperty("cn1.plugin.version", "");
cn1MavenVersion = project.getProperties().getProperty("cn1.version", "");
File retrolambdaJar = getJar("net.orfjackal.retrolambda", "retrolambda");
if (retrolambdaJar != null && retrolambdaJar.exists()) {
System.setProperty("retrolambdaJarPath", retrolambdaJar.getAbsolutePath());
} else {
getLog().warn("Could not find retrolambda Jar from dependencies. Falling back to default version 2.5.1 that may have issues building Android.");
}
String projectPlatform = project.getProperties().getProperty("codename1.projectPlatform");
if (projectPlatform == null) {
getLog().debug("Skipping build because codename1.projectPlatform property is not defined");
return;
}
if (!projectPlatform.equals(platform)) {
getLog().debug("Skipping build because codename1.projectPlatform doesn't match the given platform");
return;
}
if (platform.contains("android")) {
if (!BUILD_TARGET_ANDROID_PROJECT.equals(buildTarget)) {
String apkName = project.getBuild().getFinalName() + ".apk";
File apkFile = new File(project.getBuild().getDirectory() + File.separator + apkName);
try {
if (apkFile.exists() && apkFile.lastModified() >= getSourcesModificationTime()) {
getLog().info("Sources have not been modified since APK at " + apkFile + " was created. Skipping Android build");
return;
}
} catch (IOException ex) {
throw new MojoExecutionException("Failed to check sources modification time", ex);
}
}
}
try {
createAntProject();
} catch (IOException ex) {
getLog().error("Failed to create and build ANT project", ex);
throw new MojoFailureException("Failed to create and build ANT project", ex);
} catch (LibraryPropertiesException ex) {
getLog().error("Failed to merge properties from library "+ex.libName+". " + ex.getMessage());
throw new MojoExecutionException("Failed to merge properties from library "+ex.libName+". " + ex.getMessage(), ex);
}
}
/**
* Merge a set of jars into a single jar file.
* @param dest The destination jar file. Also the first source if it already exists.
* @param src The source jar files to be merged into the destination.
*/
private void mergeJars(File dest, File... src) {
Zip task = (Zip)antProject.createTask("zip");
task.setDestFile(dest);
task.setUpdate(true);
for (File srcFile : src) {
if (srcFile.isDirectory()) {
FileSet fs = new FileSet();
// Multiversioned jars trip out the ASM class parsing.
// Specifically module-info.class files
fs.setExcludes("**/META-INF/versions/**");
fs.setProject(this.antProject);
fs.setDir(srcFile);
task.addFileset(fs);
} else {
ZipFileSet fileset = new ZipFileSet();
// Multiversioned jars trip out the ASM class parsing.
// Specifically module-info.class files
fileset.setExcludes("**/META-INF/versions/**");
fileset.setProject(antProject);
fileset.setSrc(srcFile);
task.addZipfileset(fileset);
}
}
task.execute();
}
/**
* The dependency scopes to include in the jar file that is sent to the build server.
*/
private static String[] BUNDLE_ARTIFACT_SCOPES = new String[] { "compile" };
/**
* Artifact IDs that should not be sent to the build server.
*/
private static String[] BUNDLE_ARTIFACT_ID_BLACKLIST = new String[] {"codenameone-core", "java-runtime"};
/**
* Gets the app extensions jar file that should be included in any iOS builds.
* @return The app extensions jar file if it exists. null otherwise.
* @throws IOException
*/
private File getAppExtensionsJar() throws IOException {
if (!project.getProperties().getProperty("codename1.platform", "").equalsIgnoreCase("ios")){
// App extensions are only for iOS
return null;
}
if (!project.getBasedir().getName().equalsIgnoreCase("ios")) {
// Make sure we're building the ios project
return null;
}
File appExtensionsDir = new File(project.getBasedir(), "app_extensions");
if (!appExtensionsDir.isDirectory()) return null;
File appExtensionsJar = new File(project.getBuild().getDirectory() + File.separator + "codenameone" + File.separator + "app_extensions.jar");
if (appExtensionsJar.exists() && appExtensionsJar.lastModified() < lastModifiedRecursive(appExtensionsDir)) {
// The app extensions jar is out of date.
appExtensionsJar.delete();
}
if (!appExtensionsJar.exists()) {
File tmpDir = new File(appExtensionsJar.getParentFile(), "app_extensions");
if (tmpDir.exists()) {
FileUtils.deleteDirectory(tmpDir);
}
tmpDir.mkdirs();
for (File appExtension : appExtensionsDir.listFiles()) {
Zip task = (Zip)antProject.createTask("zip");
File dest = new File(tmpDir, appExtension.getName()+".ios.appext");
task.setDestFile(dest);
task.setUpdate(false);
if (appExtension.isDirectory()) {
FileSet fs = new FileSet();
fs.setProject(this.antProject);
fs.setDir(appExtension);
task.addFileset(fs);
task.execute();
} else if (appExtension.getName().endsWith(".zip")) {
ZipFileSet fileset = new ZipFileSet();
fileset.setProject(antProject);
fileset.setSrc(appExtension);
task.addZipfileset(fileset);
task.execute();
}
}
Zip task = (Zip)antProject.createTask("zip");
task.setDestFile(appExtensionsJar);
task.setUpdate(false);
FileSet fs = new FileSet();
fs.setProject(this.antProject);
fs.setDir(tmpDir);
task.addFileset(fs);
task.execute();
}
if (appExtensionsJar.exists()) {
return appExtensionsJar;
}
return null;
}
/**
* Gets the localizations jar file that should be included in any iOS builds.
* @return The localizations jar file if it exists. null otherwise.
* @throws IOException
*/
private File getStringsJar() throws IOException {
if (!project.getProperties().getProperty("codename1.platform", "").equalsIgnoreCase("ios")){
// App extensions are only for iOS
return null;
}
if (!project.getBasedir().getName().equalsIgnoreCase("ios")) {
// Make sure we're building the ios project
return null;
}
File stringsDir = new File(project.getBasedir(), path("src", "main", "strings"));
if (!stringsDir.isDirectory()) return null;
File stringsJar = new File(project.getBuild().getDirectory() + File.separator + "codenameone" + File.separator + "strings.jar");
if (stringsJar.exists() && stringsJar.lastModified() < lastModifiedRecursive(stringsDir)) {
// The app extensions jar is out of date.
stringsJar.delete();
}
if (!stringsJar.exists()) {
File tmpDir = new File(stringsJar.getParentFile(), "strings");
if (tmpDir.exists()) {
FileUtils.deleteDirectory(tmpDir);
}
tmpDir.mkdirs();
for (File lproj : stringsDir.listFiles()) {
Zip task = (Zip)antProject.createTask("zip");
File dest = new File(tmpDir, lproj.getName()+".zip");
task.setDestFile(dest);
task.setUpdate(false);
if (lproj.isDirectory()) {
FileSet fs = new FileSet();
fs.setProject(this.antProject);
fs.setDir(lproj);
task.addFileset(fs);
task.execute();
} else if (lproj.getName().endsWith(".zip")) {
ZipFileSet fileset = new ZipFileSet();
fileset.setProject(antProject);
fileset.setSrc(lproj);
task.addZipfileset(fileset);
task.execute();
}
}
Zip task = (Zip)antProject.createTask("zip");
task.setDestFile(stringsJar);
task.setUpdate(false);
FileSet fs = new FileSet();
fs.setProject(this.antProject);
fs.setDir(tmpDir);
task.addFileset(fs);
task.execute();
}
if (stringsJar.exists()) {
return stringsJar;
}
return null;
}
private boolean isLocalBuildTarget(String buildTarget) {
return (buildTarget.startsWith("local-") || BUILD_TARGET_XCODE_PROJECT.equals(buildTarget) || BUILD_TARGET_ANDROID_PROJECT.equals(buildTarget));
}
private void createAntProject() throws IOException, LibraryPropertiesException, MojoExecutionException {
File cn1dir = new File(project.getBuild().getDirectory() + File.separator + "codenameone");
File antProject = new File(cn1dir, "antProject");
antProject.mkdirs();
File codenameOneSettings = new File(getCN1ProjectDir(), "codenameone_settings.properties");
File icon = new File(getCN1ProjectDir(), "icon.png");
if (icon.exists()) {
FileUtils.copyFile(icon, new File(antProject, "icon.png"));
} else {
FileUtils.copyInputStreamToFile(getClass().getResourceAsStream("codenameone-icon.png"), new File(antProject, "icon.png"));
}
File codenameOneSettingsCopy = new File(antProject, codenameOneSettings.getName());
FileUtils.copyFile(codenameOneSettings, codenameOneSettingsCopy);
FileUtils.copyInputStreamToFile(getClass().getResourceAsStream("buildxml-template.xml"), new File(antProject, "build.xml"));
File distDir = new File(antProject, "dist");
distDir.mkdirs();
// Build a jar with all dependencies that we will send to the build server.
File jarWithDependencies = new File(path(project.getBuild().getDirectory(), project.getBuild().getFinalName() + "-"+buildTarget+"-jar-with-dependencies.jar"));
List cpElements;
try {
//getLog().info("Classpath Elements: "+ project.getCompileClasspathElements());
cpElements = project.getCompileClasspathElements();
} catch (Exception ex) {
throw new MojoExecutionException("Failed to get classpath elements", ex);
}
File appExtensionsJar = getAppExtensionsJar();
if (appExtensionsJar != null) {
cpElements.add(appExtensionsJar.getAbsolutePath());
}
File stringsJar = getStringsJar();
if (stringsJar != null) {
cpElements.add(stringsJar.getAbsolutePath());
}
getLog().debug("Classpath Elements: "+cpElements);
if (jarWithDependencies.exists()) {
getLog().debug("Found jar file with dependencies at "+jarWithDependencies+". Will use that one unless it is out of date.");
// Evidently pom.xml file has already built the jar file - we will use that one. This allows
// developers to override what is included in the jar file that is sent to the server.
for (String artifact : cpElements) {
File jar = new File(artifact);
if (jar.isDirectory()) {
if (jarWithDependencies.lastModified() < lastModifiedRecursive(jar)) {
getLog().debug("Jar file out of date. Dependencies have changed. "+jarWithDependencies+". Deleting");
jarWithDependencies.delete();
break;
}
} else if (jar.exists() && jar.lastModified() > jarWithDependencies.lastModified()) {
// One of the dependency jar files is newer... so we delete the dependencies jar file
// and will generate a new one.
getLog().debug("Jar file out of date. Dependencies have changed. "+jarWithDependencies+". Deleting");
jarWithDependencies.delete();
break;
}
}
}
if (!jarWithDependencies.exists()) {
getLog().info(jarWithDependencies + " not found. Generating jar with dependencies now");
// Jars that should be stripped out and not sent to the server
List blackListJars = new ArrayList();
getLog().info("Project artifacts: "+project.getArtifacts());
for (Artifact artifact : project.getArtifacts()) {
boolean addToBlacklist = false;
if (artifact.getGroupId().equals("com.codenameone") && contains(artifact.getArtifactId(), BUNDLE_ARTIFACT_ID_BLACKLIST)) {
addToBlacklist = true;
}
if (!addToBlacklist && !isLocalBuildTarget(buildTarget)) {
// When sending to the build server, we'll strip the kotlin-stdlib and the server will provide it
// for local builds, it's easier to just include it.
if (artifact.getGroupId().equals("org.jetbrains.kotlin") && artifact.getArtifactId().equals("kotlin-stdlib")) {
addToBlacklist = true;
serverMustProvideKotlinVersion = artifact.getVersion();
getLog().debug("Adding kotlin-stdlib to blacklist. Server will provide this:" + artifact);
}
}
if (!addToBlacklist && !"compile".equals(artifact.getScope())) {
addToBlacklist = true;
}
if (addToBlacklist) {
File jar = getJar(artifact);
if (jar != null) {
blackListJars.add(jar.getAbsolutePath());
blackListJars.add(jar.getPath());
try {
blackListJars.add(jar.getCanonicalPath());
getLog().debug("Added "+jar+" to blacklist");
} catch (Exception ex){}
}
}
}
getLog().debug("Merging compile classpath elements into jar with dependencies: "+cpElements);
List jarsToMerge = new ArrayList();
for (String element : cpElements) {
String canonicalEl = element;
try {
canonicalEl = new File(canonicalEl).getCanonicalPath();
} catch (Exception ex){}
if (blackListJars.contains(element) || blackListJars.contains(canonicalEl)) {
getLog().debug("NOT adding jar "+element+" because it is on the blacklist");
continue;
}
if (!new File(element).exists()) {
continue;
}
getLog().debug("Adding jar " + element + " to " + jarWithDependencies + " Jar file="+element);
jarsToMerge.add(new File(element));
}
mergeJars(jarWithDependencies, jarsToMerge.toArray(new File[jarsToMerge.size()]));
}
try {
updateCodenameOne(false);
} catch (MojoExecutionException ex) {
getLog().error("Failed to update Codename One");
throw new IOException("Failed to update Codename One", ex);
}
File antDistDir = new File(antProject, "dist");
File antDistJar = new File(antDistDir, project.getBuild().getFinalName() + "-"+buildTarget+"-jar-with-dependencies.jar");
antDistDir.mkdirs();
FileUtils.copyFile(jarWithDependencies, antDistJar);
Properties p = new Properties();
p.setProperty("codenameone_settings.properties", codenameOneSettingsCopy.getAbsolutePath());
p.setProperty("CodeNameOneBuildClient.jar", path(System.getProperty("user.home"), ".codenameone", "CodeNameOneBuildClient.jar"));
p.setProperty("dist.jar", antDistJar.getAbsolutePath());
if (automated) {
p.setProperty("automated", "true");
}
getLog().info("Running ANT build target " + buildTarget);
String logPasskey = UUID.randomUUID().toString();
Properties cn1SettingsProps = new Properties();
try (FileInputStream fis = new FileInputStream(codenameOneSettingsCopy)) {
cn1SettingsProps.load(fis);
}
if (serverMustProvideKotlinVersion != null) {
cn1SettingsProps.setProperty("codename1.arg.requireKotlinStdlib", serverMustProvideKotlinVersion);
}
FileSystemManager fsManager = VFS.getManager();
FileObject jarFile = fsManager.resolveFile( "jar:"+jarWithDependencies.getAbsolutePath() + "!/META-INF/codenameone" );
if (jarFile != null) {
FileObject[] appendedPropsFiles = jarFile.findFiles(new PatternFileSelector(".*\\/codenameone_library_appended.properties"));
if (appendedPropsFiles != null) {
for (FileObject appendedPropsFile : appendedPropsFiles) {
SortedProperties appendedProps = new SortedProperties();
try (InputStream appendedPropsIn = appendedPropsFile.getContent().getInputStream()) {
appendedProps.load(appendedPropsIn);
}
for (String propName : appendedProps.stringPropertyNames()) {
String propVal = appendedProps.getProperty(propName);
if (!cn1SettingsProps.containsKey(propName)) {
cn1SettingsProps.put(propName, propVal);
} else {
String existing = cn1SettingsProps.getProperty(propName);
if (!existing.contains(propVal)) {
cn1SettingsProps.setProperty(propName, existing + propVal);
}
}
}
}
}
FileObject[] requiredPropsFiles = jarFile.findFiles(new PatternFileSelector(".*\\/codenameone_library_required.properties"));
if (requiredPropsFiles != null) {
for (FileObject requiredPropsFile : requiredPropsFiles) {
SortedProperties requiredProps = new SortedProperties();
try (InputStream appendedPropsIn = requiredPropsFile.getContent().getInputStream()) {
requiredProps.load(appendedPropsIn);
}
String artifactId = requiredPropsFile.getParent().getName().getBaseName();
String groupId = requiredPropsFile.getParent().getParent().getName().getBaseName();
String libraryName = groupId + ":" + artifactId;
cn1SettingsProps = mergeRequiredProperties(libraryName, requiredProps, cn1SettingsProps);
}
}
}
cn1SettingsProps.setProperty("codename1.arg.hyp.beamId", logPasskey);
cn1SettingsProps.setProperty("codename1.arg.maven.codenameone-core.version", cn1MavenVersion);
cn1SettingsProps.setProperty("codename1.arg.maven.codenameone-maven-plugin", cn1MavenPluginVersion);
try (FileOutputStream fos = new FileOutputStream(codenameOneSettingsCopy)) {
cn1SettingsProps.store(fos,"");
}
final Process[] proc = new Process[1];
final boolean[] closingHypLog = new boolean[1];
Thread hyperBeamThread = new Thread(()->{
ProcessBuilder pb = new ProcessBuilder("hyp", "beam", logPasskey);
pb.redirectErrorStream(true);
try {
proc[0] = pb.start();
InputStream out = proc[0].getInputStream();
byte[] buffer = new byte[4000];
while (isAlive(proc[0])) {
int no = out.available();
if (no > 0) {
int n = out.read(buffer, 0, Math.min(no, buffer.length));
getLog().info(new String(buffer, 0, n));
}
try {
Thread.sleep(10);
}
catch (InterruptedException e) {
}
}
} catch (Exception ex) {
if (!closingHypLog[0]) {
getLog().warn("Failed to start hyperlog. The build log will not stream to your console. If the build fails, you can download the error log at https://cloud.codenameone.com/secure/index.html");
getLog().debug(ex);
}
}
});
try {
if (isLocalBuildTarget(buildTarget)) {
automated = false;
if (buildTarget.contains("android") || BUILD_TARGET_ANDROID_PROJECT.equals(buildTarget)) {
doAndroidLocalBuild(antProject, cn1SettingsProps, antDistJar);
} else if (buildTarget.contains("ios") || BUILD_TARGET_XCODE_PROJECT.equals(buildTarget)) {
doIOSLocalBuild(antProject, cn1SettingsProps, antDistJar);
} else {
throw new MojoExecutionException("Build target not supported "+buildTarget);
}
} else {
if (automated) {
getLog().debug("Attempting to start hyper beam stream the build log to the console");
hyperBeamThread.start();
}
AntExecutor.executeAntTask(new File(antProject, "build.xml").getAbsolutePath(), buildTarget, p);
}
} finally {
if (automated) {
try {
closingHypLog[0] = true;
proc[0].destroyForcibly();
} catch (Exception ex) {
}
}
}
if (automated) {
getLog().info("Extracting server result");
File result = new File(antDistDir, "result.zip");
if (!result.exists()) {
throw new IOException("Failed to find result.zip after automated build");
}
Expand unzip = (Expand)this.antProject.createTask("unzip");
unzip.setSrc(result);
File resultDir = new File(antDistDir, "result");
resultDir.mkdir();
unzip.setDest(resultDir);
unzip.execute();
for (File child : resultDir.listFiles()) {
String name = child.getName();
int dotpos = name.lastIndexOf(".");
if (dotpos < 0) {
continue;
}
String extension = name.substring(dotpos);
String base = name.substring(0, dotpos);
File copyTo = new File(project.getBuild().getDirectory() + File.separator + project.getBuild().getFinalName() + extension);
FileUtils.copyFile(child, copyTo);
if (".war".equals(extension)) {
projectHelper.attachArtifact(project, "war", copyTo);
} else if (".zip".equals(extension) && "javascript".equals(buildTarget)) {
projectHelper.attachArtifact(project, "zip", "webapp", copyTo);
} else if (".dmg".equals(extension) && "mac-os-x-desktop".equals(buildTarget)) {
projectHelper.attachArtifact(project, "dmg", "mac-app", copyTo);
} else if (".pkg".equals(extension) && "mac-os-x-desktop".equals(buildTarget)) {
projectHelper.attachArtifact(project, "pkg", "mac-app-installer", copyTo);
}
}
FileUtils.deleteDirectory(resultDir);
result.delete();
afterBuild();
}
}
private static boolean isAlive(Process proc) {
try {
proc.exitValue();
return false;
}
catch (IllegalThreadStateException e) {
return true;
}
}
private String generateCertificate(String password, String alias, String fullName, String orgName, String company, String city, String state, String twoLetterCountryCode, boolean sha512) throws Exception {
File keyTool = new File(System.getProperty("java.home") + File.separator + "bin" + File.separator + "keytool");
if (!keyTool.exists()) {
keyTool = new File(System.getProperty("java.home") + File.separator + "bin" + File.separator + "keytool.exe");
}
File keyfileLocation = new File(System.getProperty("user.home") + File.separator + "Keychain.ks");
int counter = 1;
while (keyfileLocation.exists()) {
keyfileLocation = new File(System.getProperty("user.home") + File.separator + "Keychain_" + counter + ".ks");
counter++;
}
ProcessBuilder pb = new ProcessBuilder(keyTool.getAbsolutePath(),
"-genkey", "-keystore", keyfileLocation.getAbsolutePath(), "-storetype", "jks", "-alias", alias,
"-keyalg", "RSA", "-keysize", "2048", "-validity", "15000", "-dname", "CN=" + fullName.replace(",", "\\,")
+ ", OU=" + orgName.replace(",", "\\,")
+ ", O=" + company.replace(",", "\\,")
+ ", L=" + city.replace(",", "\\,")
+ ", S=" + state.replace(",", "\\,")
+ ", C=" + twoLetterCountryCode, "-storepass", password, "-keypass", password, "-v");
if(sha512) {
pb.command().add("-sigalg");
pb.command().add("SHA512withRSA");
}
Process p = pb.start();
int res = p.waitFor();
//error occured
if(res > 0){
StringBuilder msg = new StringBuilder();
final InputStream input = p.getInputStream();
final InputStream stream = p.getErrorStream();
byte[] buffer = new byte[8192];
int i = input.read(buffer);
while (i > -1) {
String str = new String(buffer, 0, i);
System.out.print(str);
msg.append(str);
i = stream.read(buffer);
}
i = stream.read(buffer);
while (i > -1) {
String str = new String(buffer, 0, i);
System.out.print(str);
msg.append(str);
i = stream.read(buffer);
}
return null;
}
return keyfileLocation.getAbsolutePath();
}
private File getGeneratedAndroidProjectSourceDirectory() {
return new File(project.getBuild().getDirectory(), project.getBuild().getFinalName() + "-android-source");
}
private File getGeneratedIOSProjectSourceDirectory() {
return new File(project.getBuild().getDirectory(), project.getBuild().getFinalName() + "-ios-source");
}
private void doAndroidLocalBuild(File tmpProjectDir, Properties props, File distJar) throws MojoExecutionException {
if (BUILD_TARGET_ANDROID_PROJECT.equals(buildTarget)) {
File generatedProject = getGeneratedAndroidProjectSourceDirectory();
getLog().info("Generating android gradle Project to "+generatedProject+"...");
try {
if (generatedProject.exists()) {
getLog().info("Android gradle project already exists. Checking to see if it needs updating...");
if (getSourcesModificationTime() <= lastModifiedRecursive(generatedProject)) {
getLog().info("Sources have not changed. Skipping android gradle project generation");
if (open) {
openAndroidStudioProject(generatedProject);
}
return;
}
}
} catch (IOException ex) {
throw new MojoExecutionException("Failed to find last modification time of "+generatedProject);
}
}
File codenameOneJar = getJar("com.codenameone", "codenameone-core");
AndroidGradleBuilder e = new AndroidGradleBuilder();
e.setBuildTarget(buildTarget);
e.setLogger(getLog());
File buildDirectory = new File(tmpProjectDir, "dist" + File.separator + "android-build");
e.setBuildDirectory(buildDirectory);
e.setCodenameOneJar(codenameOneJar);
e.setPlatform("android");
BuildRequest r = new BuildRequest();
r.setDisplayName(props.getProperty("codename1.displayName"));
r.setPackageName(props.getProperty("codename1.packageName"));
r.setMainClass(props.getProperty("codename1.mainName"));
r.setVersion(props.getProperty("codename1.version"));
String iconPath = props.getProperty("codename1.icon");
File iconFile = new File(iconPath);
if (!iconFile.isAbsolute()) {
iconFile = new File(getCN1ProjectDir(), iconPath);
}
try {
BufferedImage bi = ImageIO.read(iconFile);
if(bi.getWidth() != 512 || bi.getHeight() != 512) {
throw new MojoExecutionException("The icon must be a 512x512 pixel PNG image. It will be scaled to the proper sizes for devices");
}
r.setIcon(iconFile.getAbsolutePath());
} catch (IOException ex) {
throw new MojoExecutionException("Error reading the icon: the icon must be a 512x512 pixel PNG image. It will be scaled to the proper sizes for devices");
}
r.setVendor(props.getProperty("codename1.vendor"));
r.setSubTitle(props.getProperty("codename1.secondaryTitle"));
r.setType("android");
r.setKeystoreAlias(props.getProperty("codename1.android.keystoreAlias"));
String keystorePath = props.getProperty("codename1.android.keystore");
if (keystorePath != null) {
File keystoreFile = new File(keystorePath);
if (!keystoreFile.isAbsolute()) {
keystoreFile = new File(getCN1ProjectDir(), keystorePath);
}
if (keystoreFile.exists() && keystoreFile.isFile()) {
try {
r.setCertificate(keystoreFile.getAbsolutePath());
} catch (IOException ex) {
throw new MojoExecutionException("Failed to load keystore file. ", ex);
}
} else {
File androidCerts = new File(getCN1ProjectDir(), "androidCerts");
androidCerts.mkdirs();
keystoreFile = new File(androidCerts, "KeyChain.ks");
if (!keystoreFile.exists()) {
try {
String alias = r.getKeystoreAlias();
if (alias == null || alias.isEmpty()) {
alias = "androidKey";
r.setKeystoreAlias(alias);
props.setProperty("codename1.android.keystoreAlias", alias);
}
String password = props.getProperty("codename1.android.keystorePassword");
if (password == null || password.isEmpty()) {
password = "password";
props.setProperty("codename1.android.keystorePassword", password);
}
getLog().info("No Keystore found. Generating one now");
String keyPath = generateCertificate(password, alias, r.getVendor(), "", r.getVendor(), "Vancouver", "BC", "CA", false);
FileUtils.copyFile(new File(keyPath), keystoreFile);
r.setCertificate(keystoreFile.getAbsolutePath());
getLog().info("Generated keystore with password 'password' at "+keystoreFile+". alias=androidKey");
new File(keyPath).delete();
SortedProperties sp = new SortedProperties();
try (FileInputStream fis = new FileInputStream(new File(getCN1ProjectDir(), "codenameone_settings.properties"))) {
sp.load(fis);
}
sp.setProperty("codename1.android.keystore", keystoreFile.getAbsolutePath());
sp.setProperty("codename1.android.keystorePassword", password);
sp.setProperty("codename1.android.keystoreAlias", alias);
try (FileOutputStream fos = new FileOutputStream(new File(getCN1ProjectDir(), "codenameone_settings.properties"))) {
sp.store(fos, "Updated keystore");
}
} catch (Exception ex) {
getLog().error("Failed to generate keystore", ex);
throw new MojoExecutionException("Failed to generate keystore", ex);
}
}
}
}
r.setCertificatePassword(props.getProperty("codename1.android.keystorePassword"));
for (Object k : props.keySet()) {
String key = (String)k;
if(key.startsWith("codename1.arg.")) {
String value = props.getProperty(key);
String currentKey = key.substring(14);
if(currentKey.indexOf(' ') > -1) {
throw new MojoExecutionException("The build argument contains a space in the key: '" + currentKey + "'");
}
r.putArgument(currentKey, value);
}
}
BuildRequest request = r;
request.setIncludeSource(true);
String testBuild = request.getArg("build.unitTest", null);
if(testBuild != null && testBuild.equals("1")) {
e.setUnitTestMode(true);
}
try {
getLog().info("Starting android project builder...");
boolean result = e.build(distJar, request);
getLog().info("Android project builder completed with result "+result);
if (!result) {
getLog().error("Received false return value from build()");
throw new MojoExecutionException("Android build failed. Received false return value for build");
}
if (BUILD_TARGET_ANDROID_PROJECT.equals(buildTarget) && e.getGradleProjectDirectory() != null) {
File gradleProject = e.getGradleProjectDirectory();
File output = getGeneratedAndroidProjectSourceDirectory();
output.getParentFile().mkdirs();
try {
getLog().info("Copying Gradle Project to "+output);
FileUtils.copyDirectory(gradleProject, output);
} catch (IOException ex) {
throw new MojoExecutionException("Failed to copy gradle project at "+gradleProject+" to "+output, ex);
}
}
if (open) {
openAndroidStudioProject(getGeneratedAndroidProjectSourceDirectory());
}
} catch (BuildException ex) {
getLog().error("Failed to build Android project with error: "+ex.getMessage(), ex);
getLog().error(e.getErrorMessage());
throw new MojoExecutionException("Failed to build android app", ex);
} finally {
e.cleanup();
}
}
private void openAndroidStudioProject(File generatedProject) {
if (isMac) {
getLog().info("Trying to open project in Android studio");
ProcessBuilder pb = new ProcessBuilder("open", "-a", "/Applications/Android Studio.app", generatedProject.getAbsolutePath());
try {
pb.start();
} catch (Exception ex) {
getLog().warn("Failed to open project in Android studio", ex);
getLog().warn("Please open the project in Android studio manually.");
getLog().warn("The project is located at "+generatedProject.getAbsolutePath());
}
} else if (isWindows) {
getLog().info("Trying to open project in Android studio");
ProcessBuilder pb = new ProcessBuilder("C:\\Program Files\\Android\\Android Studio\\bin\\studio.bat", generatedProject.getAbsolutePath());
try {
pb.start();
} catch (Exception ex) {
getLog().warn("Failed to open project in Android studio", ex);
getLog().warn("Please open the project in Android studio manually.");
getLog().warn("The project is located at "+generatedProject.getAbsolutePath());
}
} else {
getLog().warn("Opening automatically in Android studio not supported on this platform.");
getLog().warn("Please open the project in Android studio manually.");
getLog().warn("The project is located at "+generatedProject.getAbsolutePath());
}
}
private File getWorkspace(Properties props, File xcprojectRoot) {
return new File(xcprojectRoot, props.getProperty("codename1.mainName")+".xcworkspace");
}
private void openWorkspace(File workspace) throws MojoExecutionException {
try {
ProcessBuilder pb = new ProcessBuilder("open", workspace.getAbsolutePath());
Process p = pb.start();
int result = p.waitFor();
if (result != 0) {
throw new MojoExecutionException("Failed to open project at "+workspace+". Result code: "+result);
}
} catch (Exception ex) {
throw new MojoExecutionException("Failed to open project at "+workspace, ex);
}
}
private void doIOSLocalBuild(File tmpProjectDir, Properties props, File distJar) throws MojoExecutionException {
if (BUILD_TARGET_XCODE_PROJECT.equals(buildTarget)) {
File generatedProject = getGeneratedIOSProjectSourceDirectory();
getLog().info("Generating Xcode Project to "+generatedProject+"...");
try {
if (generatedProject.exists()) {
getLog().info("Xcode project already exists. Checking to see if it needs updating...");
if (getSourcesModificationTime() <= lastModifiedRecursive(generatedProject)) {
getLog().info("Sources have not changed. Skipping Xcode project generation");
if (open) {
getLog().info("Opening workspace project "+getWorkspace(props, generatedProject));
openWorkspace(getWorkspace(props, generatedProject));
}
return;
}
}
} catch (IOException ex) {
throw new MojoExecutionException("Failed to find last modification time of "+generatedProject);
}
}
File codenameOneJar = getJar("com.codenameone", "codenameone-core");
IPhoneBuilder e = new IPhoneBuilder();
e.setLogger(getLog());
File buildDirectory = new File(tmpProjectDir, "dist" + File.separator + "ios-build");
e.setBuildDirectory(buildDirectory);
e.setCodenameOneJar(codenameOneJar);
e.setPlatform("ios");
BuildRequest r = new BuildRequest();
r.setAppid(props.getProperty("codename1.ios.appid"));
r.setDisplayName(props.getProperty("codename1.displayName"));
r.setPackageName(props.getProperty("codename1.packageName"));
r.setMainClass(props.getProperty("codename1.mainName"));
r.setVersion(props.getProperty("codename1.version"));
String iconPath = props.getProperty("codename1.icon");
File iconFile = new File(iconPath);
if (!iconFile.isAbsolute()) {
iconFile = new File(getCN1ProjectDir(), iconPath);
}
try {
BufferedImage bi = ImageIO.read(iconFile);
if(bi.getWidth() != 512 || bi.getHeight() != 512) {
throw new MojoExecutionException("The icon must be a 512x512 pixel PNG image. It will be scaled to the proper sizes for devices");
}
r.setIcon(iconFile.getAbsolutePath());
} catch (IOException ex) {
throw new MojoExecutionException("Error reading the icon: the icon must be a 512x512 pixel PNG image. It will be scaled to the proper sizes for devices");
}
r.setVendor(props.getProperty("codename1.vendor"));
r.setSubTitle(props.getProperty("codename1.secondaryTitle"));
r.setType("ios");
for (Object k : props.keySet()) {
String key = (String)k;
if(key.startsWith("codename1.arg.")) {
String value = props.getProperty(key);
String currentKey = key.substring(14);
if(currentKey.indexOf(' ') > -1) {
throw new MojoExecutionException("The build argument contains a space in the key: '" + currentKey + "'");
}
r.putArgument(currentKey, value);
}
}
BuildRequest request = r;
String incSources = request.getArg("build.incSources", null);
request.setIncludeSource(true);
String testBuild = request.getArg("build.unitTest", null);
if(testBuild != null && testBuild.equals("1")) {
e.setUnitTestMode(true);
}
try {
boolean result = e.build(distJar, request);
if (!result) {
throw new MojoExecutionException("iOS build failed");
}
if (BUILD_TARGET_XCODE_PROJECT.equals(buildTarget) && e.getXcodeProjectDir() != null) {
File xcodeProject = e.getXcodeProjectDir();
File output = getGeneratedIOSProjectSourceDirectory();
output.getParentFile().mkdirs();
try {
getLog().info("Copying Xcode Project to "+output);
FileUtils.copyDirectory(xcodeProject, output);
} catch (IOException ex) {
throw new MojoExecutionException("Failed to copy xcode project at "+xcodeProject+" to "+output, ex);
}
if (open) {
getLog().info("Opening workspace project "+getWorkspace(props, output));
openWorkspace(getWorkspace(props, output));
}
}
} catch (BuildException ex) {
throw new MojoExecutionException("Failed to build ios app", ex);
} finally {
e.cleanup();
}
}
protected void afterBuild() {
}
private static class LibraryPropertiesException extends Exception {
private String libName;
LibraryPropertiesException(String libName, String message) {
super(message);
this.libName = libName;
}
}
private static class VersionMismatchException extends LibraryPropertiesException {
VersionMismatchException(String libName, String message) {
super(libName, message);
}
}
private static class PropertyConflictException extends LibraryPropertiesException {
PropertyConflictException(String libName, String message) {
super(libName, message);
}
}
private SortedProperties mergeRequiredProperties(String libraryName, Properties libProps, Properties projectProps) throws LibraryPropertiesException {
String javaVersion = (String)projectProps.getProperty("codename1.arg.java.version", "8");
String javaVersionLib = (String)libProps.get("codename1.arg.java.version");
if(javaVersionLib != null){
int v1 = 5;
if(javaVersion != null){
v1 = Integer.parseInt(javaVersion);
}
int v2 = Integer.parseInt(javaVersionLib);
//if the lib java version is bigger, this library cannot be used
if(v1 < v2){
throw new VersionMismatchException(libraryName, "Cannot use a cn1lib with java version "
+ "greater then the project java version");
}
}
//merge and save
SortedProperties merged = new SortedProperties();
merged.putAll(projectProps);
Enumeration keys = libProps.propertyNames();
while(keys.hasMoreElements()){
String key = (String) keys.nextElement();
if(!merged.containsKey(key)){
merged.put(key, libProps.getProperty(key));
}else{
//if this property already exists with a different value the
//install will fail
if(!merged.get(key).equals(libProps.getProperty(key))){
throw new PropertyConflictException(libraryName, "Property " + key + " has a conflict");
}
}
}
return merged;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy