This repository contains the Java Source Code for Google App Engine standard environment, the production runtime, the AppEngine APIs, and the local SDK.
jdk8, but using a JDK21 or JDK25 is also possible.
The shared code base is also used for GAE Java 17, Java 17 and Java 25 build and test targets, using GitHub actions:
This repository is the open source mirror of the Google App Engine Java source code that was used to produce Maven artifacts and runtime jars. On Maven Central, the released artifacts from the internal source repository are using the versions like 1.9.xx. The open source release mechanism used with this GitHub repository is using the version starting at 2.0.x. Soon we will stop entirely pushing internal 1.9.xx artifacts and encourage all App Engine customers to use the new artifacts built from the GitHub project.
Orange items are public modules artifacts and yellow are internal ones. Modules ending with * are only used on the production server side.
Source code for all public APIs for com.google.appengine.api.* packages.
- Public Documentation
- Latest javadoc.io API Javadocs from this repository
- Javadocs
- Source Code
- Source Code for repackaged API jar
- 
Maven pom.xml <packaging>war</packaging><!-- Servlet 3.1 WAR packaging--> ... <dependencies> <dependency> <groupId>com.google.appengine</groupId> <artifactId>appengine-api-1.0-sdk</artifactId> <version>2.0.39</version><!-- or later--> </dependency> <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>3.1</version> <scope>provided</scope> </dependency> ...
- 
Maven Java 21 with Jarkata EE 10 support pom.xml <packaging>war</packaging><!-- Servlet 6.0 WAR packaging--> ... <dependencies> <dependency> <groupId>com.google.appengine</groupId> <artifactId>appengine-api-1.0-sdk</artifactId> <version>2.0.39</version><!-- or later--> </dependency> <dependency> <groupId>jakarta.servlet</groupId> <artifactId>jakarta.servlet-api</artifactId> <version>6.0.0</version> <scope>provided</scope> </dependency> ...
- 
Maven Java 25 Alpha with Jakarta EE 11 support pom.xml (EE10 is not supported in Java25, EE11 is fully compatible with EE10) <packaging>war</packaging><!-- Servlet 6.1 WAR packaging--> ... <dependencies> <dependency> <groupId>com.google.appengine</groupId> <artifactId>appengine-api-1.0-sdk</artifactId> <version>3.0.0-beta</version><!-- or later--> </dependency> <dependency> <groupId>jakarta.servlet</groupId> <artifactId>jakarta.servlet-api</artifactId> <version>6.1.0</version> <scope>provided</scope> </dependency> ...
- 
Java 21/25 with javax EE8 profile appengine-web.xml <?xml version="1.0" encoding="utf-8"?> <appengine-web-app xmlns="http://appengine.google.com/ns/1.0"> <runtime>java21</runtime> <-- or java25 alpha--> <app-engine-apis>true</app-engine-apis> <!-- Add optionally: <system-properties> <property name="appengine.use.EE8" value="true"/> </system-properties> If you want to keep javax.servlet APIs and not jarkata.servlet by default --> </appengine-web-app>
- 
Java 17 appengine-web.xml <?xml version="1.0" encoding="utf-8"?> <appengine-web-app xmlns="http://appengine.google.com/ns/1.0"> <runtime>java17</runtime> <app-engine-apis>true</app-engine-apis> </appengine-web-app>
- 
Java 21 appengine-web.xml (will default to EE10, but EE8 possible) <?xml version="1.0" encoding="utf-8"?> <appengine-web-app xmlns="http://appengine.google.com/ns/1.0"> <runtime>java21</runtime> <app-engine-apis>true</app-engine-apis> </appengine-web-app>
- 
Java 25 appengine-web.xml (will default to EE11, but EE8 possible) Alpha <?xml version="1.0" encoding="utf-8"?> <appengine-web-app xmlns="http://appengine.google.com/ns/1.0"> <runtime>java25</runtime> <app-engine-apis>true</app-engine-apis> </appengine-web-app>
Source code for remote APIs for App Engine.
- Public Documentation
- Latest javadoc.io Javadocs from this repository
- Public Sample remote server
- Public Sample remote client
- Source Code
- Servlet web.xml
   <servlet>
     <display-name>Remote API Servlet</display-name>
     <servlet-name>RemoteApiServlet</servlet-name>
     <servlet-class>com.google.apphosting.utils.remoteapi.RemoteApiServlet</servlet-class>
     <load-on-startup>1</load-on-startup>
   </servlet>
   <servlet-mapping>
     <servlet-name>RemoteApiServlet</servlet-name>
     <url-pattern>/remote_api</url-pattern>
   </servlet-mapping>
- Servlet Jarkata EE10 and EE11 web.xml
   <servlet>
     <display-name>Remote API Servlet</display-name>
     <servlet-name>RemoteApiServlet</servlet-name>
     <servlet-class>com.google.apphosting.utils.remoteapi.JakartaRemoteApiServlet</servlet-class>
     <load-on-startup>1</load-on-startup>
   </servlet>
   <servlet-mapping>
     <servlet-name>RemoteApiServlet</servlet-name>
     <url-pattern>/remote_api</url-pattern>
   </servlet-mapping>
- Maven javax and jakarta API pom.xml
    <dependency>
       <groupId>com.google.appengine</groupId>
       <artifactId>appengine-remote-api</artifactId>
       <version>2.0.39</version><!-- Or later-->
    </dependency>
We moved com.google.appengine.api.memcache.stdimpl and its old dependency
javax.cache from appengine-api-1.0-sdk.jar to  a new jar appengine-api-legacy.jar.
- 
Latest javadoc.io Javadocs from this repository Users who depend on the moved classes will need to also include appengine-api-legacy.jarwhen they build/deploy. Separating these classes allowsappengine-api-1.0-sdkusers to choose any version ofjavax.cacherather than being constrained by an obsolete included version.- Maven pom.xml
 
    <dependency>
       <groupId>com.google.appengine</groupId>
       <artifactId>appengine-api-legacy.jar/artifactId>
       <version>2.0.39</version><!-- Or later-->
    </dependency>
- 
Latest javadoc.io Javadocs from this repository - Maven pom.xml
 
    <dependency>
      <groupId>com.google.appengine</groupId>
      <artifactId>appengine-testing</artifactId>
       <version>2.0.39</version><!-- Or later-->
      <scope>test</scope>
    </dependency>
    <dependency>
      <groupId>com.google.appengine</groupId>
      <artifactId>appengine-api-stubs</artifactId>
       <version>2.0.39</version><!-- Or later-->
      <scope>test</scope>
    </dependency>
    <dependency>
      <groupId>com.google.appengine</groupId>
      <artifactId>appengine-tools-sdk</artifactId>
       <version>2.0.39</version><!-- Or later-->
      <scope>test</scope>
    </dependency>
Implementation of all the App Engine APIs for local environment (devappserver) and local testing of an application before deployment.
Source code for the App Engine local dev application server and local utilities.
- Public Documentation
- Source Code for tools APIs (appcfg)
- Source Code for XML validator (appcfg)
- Source Code for local devappserver
- Source Code for shared utilities (appcfg)
- Source Code for shared utilities (config)
Source code for the App Engine production application server and utilities. It is based on the Jetty9.4 Web Server.
- Public Documentation
- Source Code for the runtime implementation
- Source Code for the Java Main
- End-to-End test Web Applications
- End-to-End tests
- Source Code for runtime utilities
The Java 17, Java 21 and 25 runtimes can benefit from extra user configuration when starting the JVM for web apps.
The default entrypoint used to boot the JVM is generated by App Engine Buildpacks.
Essentially, it is equivalent to define this entrypoint in the appengine-web.xml file. For example:
java --add-opens java.base/java.lang=ALL-UNNAMED --add-opens java.base/java.nio.charset=ALL-UNNAMED -showversion -Xms32M -Xmx204M -XX:+UseG1GC -XX:+ParallelRefProcEnabled -XX:+PrintCommandLineFlags -Dclasspath.runtimebase=/base/java_runtime -Djava.class.path=/base/java_runtime/runtime-main.jar -Djava.library.path=/base/java_runtime: com/google/apphosting/runtime/JavaRuntimeMainWithDefaults --fixed_application_path=/workspace /base/java_runtime
We do not recommend changing this default entrypoint as the memory settings are calculated based on the instance type (F1, F2, F4) and memory available.
By default, we use --add-opens java.base/java.lang=ALL-UNNAMED  --add-opens java.base/java.nio.charset=ALL-UNNAMED to open some necessary JDK APIs.
The entry point for the Java 17, Java 21, 25 runtimes can be customized with user-defined environment variables added in the appengine-web.xml configuration file.
The following table indicates the environment variables that can be used to enable/disable/configure features, and the default values if they are not set:
| Env Var | Description | Type | Default | 
|---|---|---|---|
| CPROF_ENABLE | Stackdriver Profiler | boolean | false | 
| GAE_MEMORY_MB | Available memory | size | Set by GAE or /proc/meminfo-400M | 
| HEAP_SIZE_RATIO | Memory for the heap | percent | 80 | 
| HEAP_SIZE_MB | Available heap | size | ${HEAP_SIZE_RATIO}% of${GAE_MEMORY_MB} | 
| JAVA_HEAP_OPTS | JVM heap args | JVM args | -Xms${HEAP_SIZE_MB}M -Xmx${HEAP_SIZE_MB}M | 
| JAVA_GC_OPTS | JVM GC args | JVM args | -XX:+UseG1GCplus configuration | 
| JAVA_USER_OPTS | JVM other args | JVM args | |
| JAVA_OPTS | JVM args | JVM args | See below | 
If not explicitly set, JAVA_OPTS is defaulted to:
JAVA_OPTS:=-showversion \
           ${DBG_AGENT} \
           ${PROFILER_AGENT} \
           ${JAVA_HEAP_OPTS} \
           ${JAVA_GC_OPTS} \
           ${JAVA_USER_OPTS}
When CPROF_ENABLE is true, the default entrypoint adds the PROFILER_AGENT as:
-agentpath:/opt/cprof/profiler_java_agent.so=--logtostderr
For example, if your application code needs more -add-opens flags, you can use the JAVA_USER_OPTS environment variable defined in the appengine-web.xml file:
  <env-variables>
     <env-var name="JAVA_USER_OPTS" value="--add-opens java.base/java.util=ALL-UNNAMED" />
   </env-variables>
Note:
- Only one of appengine.use.EE8,appengine.use.EE10, orappengine.use.EE11can be set totrueat a time.
- Flags can be set in WEB-INF/appengine-web.xmlor via Java system properties (e.g.,-Dappengine.use.EE10=true). System properties overrideappengine-web.xml.
- EE6 = Servlet 3.1 (javax.*), EE8 = Servlet 4.0 (javax.*), EE10 = Servlet 5.0 (jakarta.*), EE11 = Servlet 6.0 (jakarta.*).
- Jetty 12.1 should be fully backward compatible with Jetty 12.0 and EE11 version should also be backward compatible with EE10.
- EE8 should also be backward compatible with EE6.
| Flag(s) Set in appengine-web.xmlor System Properties | Resulting Jetty | Support | Resulting EE Version | Notes | 
|---|---|---|---|---|
| None (default) | 9.4 | GA | 6 | |
| appengine.use.EE8=true | 12.0 | GA | 8 | |
| appengine.use.EE10=true | 12.0 | GA | 10 | |
| appengine.use.EE8=true,appengine.use.jetty121=true | 12.1 | Early Access | 8 | |
| appengine.use.EE10=true,appengine.use.jetty121=true | 12.1 | Early Access | 11 | Upgraded: EE10 is upgraded to EE11 on Jetty 12.1 | 
| appengine.use.EE11=true | 12.1 | Early Access | 11 | appengine.use.jetty121=trueis used automatically | 
| Flag(s) Set in appengine-web.xmlor System Properties | Resulting Jetty | Support | Resulting EE Version | Notes | 
|---|---|---|---|---|
| None (default) | 12.0 | GA | 10 | |
| appengine.use.EE8=true | 12.0 | GA | 8 | |
| appengine.use.EE10=true | 12.0 | GA | 10 | |
| appengine.use.jetty121=true | 12.1 | Early Access | 11 | If no EE flag is set, jetty121defaults to EE11 | 
| appengine.use.EE8=true,appengine.use.jetty121=true | 12.1 | Early Access | 8 | |
| appengine.use.EE10=true,appengine.use.jetty121=true | 12.1 | Early Access | 11 | Upgraded: EE10 is upgraded to EE11 on Jetty 12.1 | 
| appengine.use.EE11=true | 12.1 | Early Access | 11 | appengine.use.jetty121=trueis used automatically | 
| Flag(s) Set in appengine-web.xmlor System Properties | Resulting Jetty | Support | Resulting EE Version | Notes | 
|---|---|---|---|---|
| None (default) | 12.1 | Early Access | 11 | appengine.use.jetty121=trueis used | 
| appengine.use.EE8=true | 12.1 | Early Access | 8 | appengine.use.jetty121=trueis used | 
| appengine.use.EE11=true | 12.1 | Early Access | 11 | appengine.use.jetty121=trueis used | 
| appengine.use.EE10=true | ERROR | Unsupported | ERROR | EE10 is not supported, use compatible version EE11 instead | 
Check out the contributing guide to learn how you can report issues and help make changes.
Always be sure to follow the Code of Conduct.
