Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
ch.swaechter:angularj-universal
With the introduction of Angular Universal, a solution for dynamically prerendering Angular applications on the server side (SSR) and sending the content directly to the browser as 'already-bootstrapped' application, Angular became more interesting for many developers that were using a Node.js environment. It solved the problems of SEO optimization, empty page previews from Facebooks & others and resource management (For more information see https://scotch.io/tutorials/server-side-rendering-in-angular-2-with-angular-universal).
For traditional web developers in the Java enterprise environment with technologies like Spring Boot or Java EE, these technologies required the use of a separate Node.js runtime in combination with a web framework like Express.js or even worse, inter process communication between two servers. Requirements that often just weren't possible or not allowed: Two run times with two server sockets, technology fragmentation or just skepticism against Node.js on the server side or JavaScript in general. All in all, Java developers weren't able to use the benefits made in the Node.js ecosystem.
This project provides a Java API to a local Node.js instance that is responsible for server side rendering the Angular application. For this, the JNI language bindings of the J2V8 projects were used. This makes it possible to use the latest Angular Universal features without having a second [JavaScript] web server (Only a second VM by Node.js) - an integration of a prerendered Angular application into a pure Java stack is now possible.
This repository is divided into several modules that all share the same Angular application
Module name | Content and responsibility |
---|---|
angularj-universal-application | Contains the main Angular application with several pages |
angularj-universal-renderer | Contains the main render SDK that provides the core functionality and allows a developer to implement an own render solution |
angularj-universal-renderer-v8 | Contains a specific render solution that uses a Node.js instance via J2V8 language bindings for rendering requests |
angularj-universal-example-spring-boot | Contains a Spring Boot web application that serves the Angular application with the help of a custom written Spring Boot starter (See module bellow) |
angularj-universal-example-spring-boot-simple | Contains a Spring Boot web application that serves the Angular application without the help of the Spring Boot starter. This leads to a more simple example, but a lot more boiler plated code is required. For simplicity, this module is used as example |
angularj-universal-example-servlet | TODO: Provide a servlet example |
angularj-universal-spring-boot-starter | Custom Spring Boot starter that allows are better integration of a server side rendered application into the Spring Boot ecosystem |
It is really hard to cover all possible work flows and frameworks, so we are focusing on Spring Boot in this example, but other examples are provided as source code. Please don't hesitate to open a new issue and ask for help with a new framework. With your help we will be able to provide an example for your framework!
If you want to skip this tutorial, checkout the working example https://github.com/swaechter/angularj-universal/tree/master/angularj-universal-example-spring-boot-simple
The whole getting started process can be divided into three phases:
As mentioned, you can adapt these steps for your framework. If you want to skip the tutorial, just take a look at the example in angularj-universal-example-spring-boot-simple
and the Angular application angularj-universal-application
.
Head over to http://start.spring.io/
and create a new Spring Boot project with Maven. It is important to select Web
and DevTools
as dependency:
http://start.spring.io/
Download and import that Maven project in your favorite Java IDE. In addition, add the AngularJ Universal dependencies:
<dependency>
<groupId>ch.swaechter</groupId>
<artifactId>angularj-universal-renderer</artifactId>
<version>0.0.3</version>
</dependency>
<dependency>
<groupId>ch.swaechter</groupId>
<artifactId>angularj-universal-renderer-v8</artifactId>
<version>0.0.3</version>
</dependency>
Now that we have a project layout and the required Java dependencies, we can continue with phase 2 and create a new Angular application inside the Java project.
To make the integration with Java as smooth as possible, we will stick to the Maven default layout https://maven.apache.org/guides/introduction/introduction-to-the-standard-directory-layout.html
. The Angular application will be located in angularj-universal-example-spring-boot-simple/src/main/angular
and the build output in angularj-universal-example-spring-boot-simple/src/main/resources
.
Install the Angular CLI and create a new Angular application:
sudo npm install -g @angular/cli
cd angularj-universal-example-spring-boot-simple/src/main
ng new angular
Generate an universal component, add the required third party libraries and update your modules:
cd angular
ng generate universal server
npm install --save-dev ts-loader @nguniversal/module-map-ngfactory-loader
npm update
This will create a traditional single page application (SPA) that needs some adjustment and preparations. To enable lazy modules, change the file src/app/app.server.module.ts
to this:
import {NgModule} from '@angular/core';
import {ServerModule} from '@angular/platform-server';
import {AppModule} from './app.module';
import {AppComponent} from './app.component';
import {ModuleMapLoaderModule} from "@nguniversal/module-map-ngfactory-loader";
@NgModule({
imports: [
AppModule,
ServerModule,
ModuleMapLoaderModule // Add this line
],
bootstrap: [AppComponent],
})
export class AppServerModule {
}
To stick with the traditional Maven layout, change the output directory of your client application (App 0) in the .angular-cli.json
:
"outDir": "../resources/public",
At the moment, the generated server bundle (App 1) will not be relocatable (The required libraries from ǹode_modules
are not packaged into the build artifact), hence we have to build a relocatable bundle with webpack. Create the file webpack.config.js
:
const path = require('path');
const webpack = require('webpack');
module.exports = {
entry: {server: './library/server.ts'},
resolve: {extensions: ['.js', '.ts']},
target: 'node',
// this makes sure we include node_modules and other 3rd party libraries
externals: [/(node_modules|main\..*\.js)/],
output: {
path: path.join(__dirname, '../resources'),
filename: '[name].js'
},
module: {
rules: [
{test: /\.ts$/, loader: 'ts-loader'}
]
},
plugins: [
// Temporary Fix for issue: https://github.com/angular/angular/issues/11580
// for "WARNING Critical dependency: the request of a dependency is an expression"
new webpack.ContextReplacementPlugin(
/(.+)?angular(\\|\/)core(.+)?/,
path.join(__dirname, 'src'), // location of your src
{} // a map of your routes
),
new webpack.ContextReplacementPlugin(
/(.+)?express(\\|\/)(.+)?/,
path.join(__dirname, 'src'),
{}
)
]
}
Now we need to create a communication mechanism with which the Java JVM can speak to the Angular application (Render a page request) and handle the answer (Receive rendered page request). For this, the Java JVM executes a script that establish this communication.
Create a file library/renderadapter.ts
that provides the communication mechanism (I should move this code to an own consumable NPM library):
require("zone.js/dist/zone-node");
import {provideModuleMap} from "@nguniversal/module-map-ngfactory-loader";
import {renderModuleFactory} from "@angular/platform-server";
export declare function registerRenderAdapter(renderadapter: RenderAdapter): void;
export declare function receiveRenderedPage(uuid: string, html: string, error: any): void;
export class RenderAdapter {
constructor(private appservermodulengfactory: any, private lazymodulemap: any, private html: string) {
registerRenderAdapter(this);
}
setHtml(html: string) {
this.html = html;
}
renderPage(uuid: string, uri: string) {
renderModuleFactory(this.appservermodulengfactory, {
document: this.html,
url: uri,
extraProviders: [
provideModuleMap(this.lazymodulemap)
]
}).then(html => {
receiveRenderedPage(uuid, html, null);
});
}
}
Then create another file library/server.ts
that is going to establish the communication (This script will be loaded by the Java JVM):
import {RenderAdapter} from "./renderadapter";
const {AppServerModuleNgFactory, LAZY_MODULE_MAP} = require("./../dist-server/main.bundle");
new RenderAdapter(AppServerModuleNgFactory, LAZY_MODULE_MAP, "<app-root></app-root>");
In short, the generated server.js bundle will be loaded by the Java JVM. The script then passes an instance of the render adapter to the JVM (See registerRenderAdapter)
, so the JVM can use this instance to render page requests. After a page is rendered, it is passed back to the JVM (See receiveRenderedPage
) and served to the client. Hence, the functions registerRenderAdapter
and receiveRenderedPage
can't be found in the JavaScript code because they are registered Java methods in the script engine.
Now let's update our Node scripts for building all three applications (Snipped from package.json
):
"scripts": {
"start": "npm run build",
"build": "ng build --app 0 --prod --build-optimizer && ng build --app 1 --prod --output-hashing none && webpack"
},
Now run npm start
or npm run build
and the traditional client side SPA application will land in angularj-universal-example-spring-boot-simple/src/main/resources/public
, the publicly accessible part of our web server. In addition, the relocatable server side bundle that will be loaded by the Java JVM can be found in angularj-universal-example-spring-boot-simple/src/main/resources/server.js
, the not publicly accessible part of our web server (Only the subdirectory public
is).
We can continue with the third phase and integrate the Angular application into the Java application and automate the NPM build (Because In cause you don't execute it, your application will fail).
Now we switch back to the Java part and create a web application with a controller and a service that is going to render our application.
Create a new Java class file angularj-universal-example-spring-boot-simple/src/main/java/ch/swaechter/angularjuniversal/example/springboot/simple/DemoApplication.java
with the following content bellow:
Note 1: In this example we only provide the /
route, but the example in angularj-universal-application
has several pages and hence routes (Home, keywords, about etc.). For every Angular route you have to define an own route in Spring Boot
Note 2: To avoid such manual and duplicated routes, check out the Spring Boot starter example in angularj-universal-example-spring-boot
. In the starter example, you can define all routes via the application.properties
and don't have to define any ModelAndView
at all.
package ch.swaechter.angularjuniversal.example.springboot.simple;
import ch.swaechter.angularjuniversal.renderer.Renderer;
import ch.swaechter.angularjuniversal.renderer.configuration.RenderConfiguration;
import ch.swaechter.angularjuniversal.renderer.engine.RenderEngineFactory;
import ch.swaechter.angularjuniversal.renderer.utils.RenderUtils;
import ch.swaechter.angularjuniversal.v8renderer.V8RenderEngineFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.stereotype.Service;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.bind.annotation.RestController;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.Future;
@SpringBootApplication
public class WebApplication {
public static void main(String[] args) {
SpringApplication.run(WebApplication.class, args);
}
@RestController
public class ContentController {
private final RenderService renderservice;
@Autowired
public ContentController(RenderService renderservice) {
this.renderservice = renderservice;
}
@ResponseBody
@GetMapping("/")
public String showIndex() throws Exception {
return renderservice.renderPage("/").get();
}
// Note: For every Angular route you have to provide a Java Spring Boot route - or just provide a wildcard route
}
@Service
public class RenderService {
private final Renderer renderer;
public RenderService() throws IOException {
// Load the template and create a temporary server bundle file from the resource (This file will of course never change until manually edited)
InputStream templateinputstream = getClass().getResourceAsStream("/public/index.html");
InputStream serverbundleinputstream = getClass().getResourceAsStream("/server.js");
String templatecontent = RenderUtils.getStringFromInputStream(templateinputstream, StandardCharsets.UTF_8);
File serverbundlefile = RenderUtils.createTemporaryFileFromInputStream("serverbundle", "tmp", serverbundleinputstream);
// File localserverbundlefile = new File("<Local server bundle on the file system>"); --> Also enable auto reload in the configuration
// Create the configuration. For real live reloading, don't use a temporary file but the real generated on from the file system
RenderConfiguration configuration = new RenderConfiguration.RenderConfigurationBuilder(templatecontent, serverbundlefile).engines(4).liveReload(false).build();
// Create the V8 render engine factory for spawning render engines
RenderEngineFactory factory = new V8RenderEngineFactory();
// Create and start the renderer
this.renderer = new Renderer(factory, configuration);
this.renderer.startRenderer();
}
Future<String> renderPage(String uri) {
// Render a request and return a resolvable future
return renderer.addRenderRequest(uri);
}
}
}
Now start the Spring Boot application and head to the URL:
http://localhost:8080
You can test if the server side rendering is working if:
Be aware of these information:
npm run build
, otherwise the AngularJ Universal won't find the index template and relocatable server bundle and will fail. In case you want to integrate this into your regular Maven build, take a look at the Maven NPM/Node plugin eirslett/frontend-maven-plugin
or just take a look at the usage in angularj-universal-application/pom.xml
RenderConfiguration
and enable the reload flagangularj-universal-application
to avoid duplication of Angular applications. Just be aware of these fact in case you are reading the source code and asking yourself where the application is. In case you want to use the examples for your own project, just copy away the example and integrate the angularj-universal-application
content and adjust the pom.xml
If you encounter a problem/bug or something isn't documented well enough, feel free to create an issue in the issue tracker (https://github.com/swaechter/angularj-universal/issues) or send me a mail (waechter.simon@gmail.com)
This project was heavily influenced by many other projects (A huge thank you to the authors):
FAQs
A Java library for rendering Angular 4+ on the server side and providing them on the client side as universal application
We found that ch.swaechter:angularj-universal demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 0 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.