Security News
The Risks of Misguided Research in Supply Chain Security
Snyk's use of malicious npm packages for research raises ethical concerns, highlighting risks in public deployment, data exfiltration, and unauthorized testing.
africa.absa:inception
The Absa Inception Framework enables the rapid development of applications with a Java back-end and Angular front-end.
Complete the following steps to setup a development environment on MacOS.
xcode-select --install
/usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install)"
brew install openjdk
brew install maven
brew install jenv
export PATH="$HOME/.jenv/bin:$PATH"
eval "$(jenv init -)"
Intel Mac:
jenv add /usr/local/opt/openjdk/libexec/openjdk.jdk/Contents/Home
jenv global 17
sudo ln -s /usr/local/opt/openjdk/libexec/openjdk.jdk /Library/Java/JavaVirtualMachines/openjdk
M1 Mac:
jenv add /opt/homebrew/opt/openjdk/libexec/openjdk.jdk/Contents/Home
jenv global 17
sudo ln -s /opt/homebrew/opt/openjdk/libexec/openjdk.jdk /Library/Java/JavaVirtualMachines/openjdk
jenv enable-plugin maven
brew install postgres
brew install liquibase
brew install npm
npm install -g @angular/cli@13
npm config set save-prefix="~"
MailSlurper is a handy SMTP mail server useful for local and team application development. Mails sent by an application running locally on a developer's machine can be viewed using a web interface. This is useful when developing and testing an application locally.
Complete the following steps to setup MailSlurper on MacOS.
Download the MailSlurper package from https://github.com/mailslurper/mailslurper/releases.
Extract the MailSlurper package.
Edit the config.json file and set the wwwPort value to 8090.
Launch the createcredentials binary in a Terminal window and enter the username as inception and password as inception when prompted.
NOTE: You may need to enable the execution of the mailslurper binary under System Preferences > Security & Privacy > General
Launch the mailslurper binary in a Terminal window.
NOTE: You may need to enable the execution of the mailslurper binary under System Preferences > Security & Privacy > General
Complete the following steps to checkout and build the Absa Inception Framework on MacOS.
git clone https://github.com/AbsaOSS/inception.git
mvn clean compile package install
npm i
ng build
cd dist/ngx-inception
npm link
java -jar demo-1.1.0-SNAPSHOT.jar
ng serve --host 0.0.0.0
Download the Git for Windows package from https://git-scm.com/download/win and install it.
Select the following options when installing Git:
Use the native Windows Secure Channel library
Checkout Windows-style, commit Unix-style line endings
Use MinTTY (the default terminal of MSYS2)
Default (fast-forward or merge)
Git Credential Manager
Enable file system caching
Download the OpenJDK 17 package from https://adoptium.net/ and install it.
Download the Maven package from https://maven.apache.org and install it.
NOTE: Ensure that Maven is on the path.
NOTE: Add the proxy settings to the ~/.m2/settings.xml file if required.
Download the Node.js and npm package from https://www.npmjs.com/get-npm and install it.
NOTE: Set the proxy for NPM if required.
Install the Angular CLI 13 globally by executing the following command in a Git Bash window.
npm install -g @angular/cli@13
Execute the following command to change to the more conservative tilde (~) patch update approach for dependencies for npm.
npm config set save-prefix="~"
Complete the following steps to setup a development environment on Windows.
git clone https://github.com/AbsaOSS/inception.git
mvn clean compile package
npm i
ng build
cd dist/ngx-inception
npm link
java -jar demo-1.1.0-SNAPSHOT.jar
ng serve --host 0.0.0.0
Complete the following steps to create a new application based on the Absa Inception Framework.
Select a name for the new application, e.g. demo, and create the top-level directory for the application with the same name. The directory MUST be created under the same directory that the Absa Inception Framework project was cloned into. This is because the Absa Inception Framework will be installed as an npm dependency using a relative path.
Execute the following commands under the top-level directory for the new application to create the project directory structure.
mkdir -p src/main/java
mkdir -p src/main/java-templates
mkdir -p src/main/resources/META-INF
mkdir -p src/test/java
mkdir -p src/test/resources
Setup the Spring Boot back-end for the application.
Execute the following commands under the top-level directory for the new application to generate the RSA keypair used to sign and verify OAuth2 JWTs issued by the application.
openssl genpkey -algorithm RSA -pkeyopt rsa_keygen_bits:2048 -pkeyopt rsa_keygen_pubexp:65537 | openssl pkcs8 -topk8 -nocrypt -outform pem > src/main/resources/META-INF/oauth2-jwt-key
openssl pkey -pubout -inform pem -outform pem -in src/main/resources/META-INF/oauth2-jwt-key -out src/main/resources/META-INF/oauth2-jwt-key.pub
Create the pom.xml file under the top-level directory for the new application with the following contents, changing the groupId, artifactId, version, name and description values as appropriate.
NOTE: If you do not require all the capabilities provided by the Inception Framework you can remove the appropriate dependencies from the pom.xml file, e.g. removing the inception-reporting-rs dependency will remove the JasperReports-based reporting functionality.
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>demo</groupId>
<artifactId>demo</artifactId>
<version>1.0.0-SNAPSHOT</version>
<name>demo</name>
<parent>
<groupId>africa.absa</groupId>
<artifactId>inception</artifactId>
<version>1.1.0</version>
<relativePath/>
</parent>
<dependencies>
<!-- Inception Core Dependencies -->
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-api</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-application</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-jta</artifactId>
</dependency>
<!-- Inception Module Dependencies -->
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-codes-api</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-config-api</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-error-api</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-mail-api</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-oauth2-authorization-server</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-oauth2-resource-server</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-reporting-api</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-scheduler-api</artifactId>
</dependency>
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-security-api</artifactId>
</dependency>
<!-- Dependencies -->
<dependency>
<groupId>com.h2database</groupId>
<artifactId>h2</artifactId>
</dependency>
<dependency>
<groupId>org.postgresql</groupId>
<artifactId>postgresql</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<!-- Test Dependencies -->
<dependency>
<groupId>africa.absa</groupId>
<artifactId>inception-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
<plugin>
<groupId>com.github.eirslett</groupId>
<artifactId>frontend-maven-plugin</artifactId>
<version>1.6</version>
<configuration>
<workingDirectory>src/main/frontend</workingDirectory>
<nodeVersion>v16.9.0</nodeVersion>
<npmVersion>7.21.1</npmVersion>
</configuration>
<executions>
<execution>
<id>install node and npm</id>
<goals>
<goal>install-node-and-npm</goal>
</goals>
</execution>
<execution>
<id>npm install</id>
<goals>
<goal>npm</goal>
</goals>
</execution>
<execution>
<id>npm run build</id>
<goals>
<goal>npm</goal>
</goals>
<configuration>
<arguments>run build</arguments>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
</project>
Create the class and resource packages for the application. The classes and resources for the application will be created under these packages, e.g. the Spring Boot application class, the in-memory H2 database scripts for the application, etc.
mkdir -p src/main/java/demo
mkdir -p src/main/resources/demo
Add the Spring Boot Configuration class using the naming convention ApplicationNameConfiguration, e.g. src/main/java/demo/DemoConfiguration.java.
package demo;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.jpa.repository.config.EnableJpaRepositories;
@Configuration
@ComponentScan(
basePackages = {"africa.absa.inception", "demo"})
@EnableJpaRepositories(
entityManagerFactoryRef = "applicationEntityManagerFactory",
basePackages = {"demo"})
public class DemoConfiguration {}
Add the Spring Boot Application class using the naming convention ApplicationNameApplication, e.g. src/main/java/demo/DemoApplication.java, which extends the Absa Inception Framework application class (africa.absa.inception.application.Application).
package demo;
import africa.absa.inception.application.Application;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ApplicationContext;
/**
* The <code>DemoApplication</code> provides the implementation of the Absa Inception Framework
* application class for the demo application.
*/
@SpringBootApplication
public class DemoApplication extends Application {
/* Logger */
private static final Logger logger = LoggerFactory.getLogger(DemoApplication.class);
/**
* Constructs a new <code>DemoApplication</code>.
*
* @param applicationContext the Spring application context
*/
public DemoApplication(
ApplicationContext applicationContext) {
super(applicationContext);
}
/**
* The main method.
*
* @param args the command-line arguments
*/
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}
Add the Liquibase changelog for the application using the naming convention src/main/resources/db/application-name.changelog.xml, e.g. src/main/resources/db/demo.changelog.xml, to the project. This file will contain all the Liquibase changesets used to initialize both the in-memory H2 database and the environment-specific databases for the application. The in-memory H2 database allows developers to run the application locally while developing the application and is also used to execute all Junit tests for the application as part of the build process.
NOTE: You need to update the changeSet id and the application schema name.
<?xml version="1.0" encoding="UTF-8"?>
<databaseChangeLog
xmlns="http://www.liquibase.org/xml/ns/dbchangelog"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog
http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-3.8.xsd">
<property name="blob_type" value="bytea" dbms="h2"/>
<property name="blob_type" value="bytea" dbms="postgresql"/>
<property name="blob_type" value="blob" dbms="oracle"/>
<property name="blob_type" value="varbinary(max)" dbms="mssql"/>
<property name="now" value="now()" dbms="h2"/>
<property name="now" value="now()" dbms="postgresql"/>
<property name="now" value="sysdate" dbms="oracle"/>
<property name="now" value="CURRENT_TIMESTAMP" dbms="mssql"/>
<changeSet id="demo-1.0.0" author="">
<sql dbms="h2" endDelimiter=";">
CREATE SCHEMA IF NOT EXISTS demo
</sql>
<sql dbms="postgresql" endDelimiter=";">
CREATE SCHEMA IF NOT EXISTS demo
</sql>
<sql dbms="mssql" endDelimiter=";">
CREATE SCHEMA demo
</sql>
</changeSet>
</databaseChangeLog>
Add the Spring application configuration file, src/main/resources/application.yml, changing the spring.application.name value to the name of the application.
NOTE: You need to update the spring.application.name property.
server:
port: 8080
inception:
application:
data-source:
class-name: org.h2.jdbcx.JdbcDataSource
url: jdbc:h2:mem:application;MODE=PostgreSQL;DB_CLOSE_DELAY=-1;DB_CLOSE_ON_EXIT=FALSE;DATABASE_TO_LOWER=TRUE;CASE_INSENSITIVE_IDENTIFIERS=TRUE
min-pool-size: 5
max-pool-size: 10
debug:
enabled: true
oauth2:
authorization-server:
jwt:
rsa-private-key-location: classpath:META-INF/oauth2-jwt-key
rsa-public-key-location: classpath:META-INF/oauth2-jwt-key.pub
resource-server:
jwt:
rsa-public-key-location: classpath:META-INF/oauth2-jwt-key.pub
spring:
application:
name: demo
springdoc:
writer-with-order-by-keys: true
Setup the Angular front-end for the application.
Execute the following command under the src/main directory to create the new Angular frontend.
ng new --skip-git --routing --style scss frontend
Edit the src/main/frontend/angular.json file and add the fontawesome path to the styles array as shown below.
{
...
"projects": {
"frontend": {
...
"architect": {
"build": {
...
"options": {
...
"styles": [
"node_modules/@fortawesome/fontawesome-free/css/all.css",
"src/styles.scss"
],
...
},
...
}
Edit the src/main/frontend/angular.json file and add the preserveSymlinks option with a value of true, add the aot option with a value of true, and change the outputPath option to ../../../target/classes/static as shown below. This will result in the Angular font-end being packaged as part of the Spring Boot application as a set of static resources.
{
...
"projects": {
"frontend": {
...
"architect": {
"build": {
...
"options": {
"preserveSymlinks": true,
"aot": true,
"outputPath": "../../../target/classes/static",
...
},
...
}
Edit the src/main/frontend/angular.json file and set the maximumWarning* to 2mb and maximumError to 4mb as shown below.
{
...
"projects": {
"frontend": {
...
"architect": {
"build": {
...
"configurations": {
"production": {
"budgets": [
{
"type": "initial",
"maximumWarning": "2mb",
"maximumError": "4mb"
},
{
"type": "anyComponentStyle",
"maximumWarning": "2kb",
"maximumError": "4kb"
}
],
...
}
Execute the following commands under the src/main/frontend directory to install the dependencies for the ngx-inception library.
npm install --save @angular/cdk@12
npm install --save @angular/localize@12
npm install --save @angular/material@12
npm install --save @angular/material-moment-adapter@12
npm install --save @auth0/angular-jwt@5
npm install --save @fortawesome/fontawesome-free@5
npm install --save bootstrap@4
npm install --save moment@2
npm install --save ngx-perfect-scrollbar@10
npm install --save string-template@1
npm install --save uuid@8
npm install --save-dev @types/uuid@8
npm install --save-dev @types/string-template@1
npm install --save-dev codelyzer
Execute the following command under the src/main/frontend directory to install the ngx-inception library dependency.
npm install --save @absaoss/ngx-inception@1.0
Add the path mapping for the ngx-inception library to the src/main/frontend/tsconfig.app.json file.
"compilerOptions": {
...,
"paths": {
"@absaoss/ngx-inception/*": [
"./node_modules/@absaoss/ngx-inception/*"
]
}
},
Edit the src/main/frontend/src/polyfills.ts file and add the line below under the APPLICATION IMPORTS section.
import '@angular/localize/init';
Add the BrowserAnimationsModule and Absa Inception Framework imports to the src/main/frontend/src/app/app.module.ts file.
...
import { BrowserAnimationsModule } from '@angular/platform-browser/animations';
import {
CoreModule, InceptionAppModule, InceptionConfig, NavigationBadge, NavigationItem, NavigationTitle
} from '@absaoss/ngx-inception/core';
import {environment} from "../environments/environment";
...
Add the ngxInceptionConfig to the src/main/frontend/src/app/app.module.ts file.
NOTE: If the application has a logged out landing page, which unauthenticated users can access, then the logoutRedirectUri should be changed to the URL for this page.
...
const ngxInceptionConfig: InceptionConfig = {
// Application Information
applicationId: 'demo',
applicationVersion: '1.0.0',
// OAuth Token URL
oauthTokenUrl: environment.inception_oauthTokenUrl,
// Logout redirect URI
logoutRedirectUri: '/login',
// Inception API URLs
codesApiUrlPrefix: environment.inception_codesApiUrlPrefix,
configApiUrlPrefix: environment.inception_configApiUrlPrefix,
errorApiUrlPrefix: environment.inception_errorApiUrlPrefix,
mailApiUrlPrefix: environment.inception_mailApiUrlPrefix,
partyApiUrlPrefix: environment.inception_partyApiUrlPrefix,
partyReferenceApiUrlPrefix: environment.inception_partyReferenceApiUrlPrefix,
referenceApiUrlPrefix: environment.inception_referenceApiUrlPrefix,
reportingApiUrlPrefix: environment.inception_reportingApiUrlPrefix,
schedulerApiUrlPrefix: environment.inception_schedulerApiUrlPrefix,
securityApiUrlPrefix: environment.inception_securityApiUrlPrefix,
// Flags
forgottenPasswordEnabled: true,
userProfileEnabled: true
};
@NgModule({
...
Add the BrowserAnimationsModule and CoreModule.forRoot(ngxInceptionConfig) module imports to the @NgModule annotation in the src/main/frontend/src/app/app.module.ts file.
@NgModule({
bootstrap: [AppComponent],
declarations: [AppComponent
],
imports: [
AppRoutingModule,
BrowserAnimationsModule,
BrowserModule,
CoreModule.forRoot(ngxInceptionConfig)
]
})
Change the AppModule class so that it extends the InceptionAppModule class, then implement the initNavigation method. The initNavigation method specifies which Absa Inception Framework and application-specific views to include in the navigation.
export class AppModule extends InceptionAppModule {
constructor() {
super();
}
/**
* Initialise the navigation for the application.
*
* @returns The navigation for the application.
*/
protected initNavigation(): NavigationItem[] {
const navigation: NavigationItem[] = [];
navigation.push(new NavigationItem('fa fa-tachometer-alt', 'Dashboard', '/dashboard',
['ROLE_Administrator', 'FUNCTION_Dashboard.Dashboard'], undefined, undefined, undefined));
navigation.push(new NavigationItem('fa fa-cogs', 'Administration', '/administration',
['ROLE_Administrator', 'FUNCTION_Codes.CodeAdministration', 'FUNCTION_Config.ConfigAdministration',
'FUNCTION_Security.GroupAdministration', 'FUNCTION_Security.TenantAdministration',
'FUNCTION_Security.ResetUserPassword', 'FUNCTION_Security.UserAdministration',
'FUNCTION_Security.UserDirectoryAdministration', 'FUNCTION_Security.UserGroups',
'FUNCTION_Scheduler.SchedulerAdministration', 'FUNCTION_Scheduler.JobAdministration',
'FUNCTION_Mail.MailAdministration', 'FUNCTION_Mail.MailTemplateAdministration'
], [new NavigationItem('fa fa-shield-alt', 'Security', '/administration/security',
['ROLE_Administrator', 'FUNCTION_Security.GroupAdministration', 'FUNCTION_Security.TenantAdministration',
'FUNCTION_Security.ResetUserPassword', 'FUNCTION_Security.UserAdministration',
'FUNCTION_Security.UserDirectoryAdministration', 'FUNCTION_Security.UserGroups'
], [new NavigationItem('fas fa-user', 'Users', '/administration/security/users',
['ROLE_Administrator', 'FUNCTION_Security.ResetUserPassword', 'FUNCTION_Security.UserAdministration',
'FUNCTION_Security.UserGroups'
]), new NavigationItem('fas fa-users', 'Groups', '/administration/security/groups',
['ROLE_Administrator', 'FUNCTION_Security.GroupAdministration']),
new NavigationItem('far fa-building', 'Tenants', '/administration/security/tenants',
['ROLE_Administrator', 'FUNCTION_Security.TenantAdministration']),
new NavigationItem('far fa-address-book', 'User Directories', '/administration/security/user-directories',
['ROLE_Administrator', 'FUNCTION_Security.UserDirectoryAdministration'])
]), new NavigationItem('fa fa-cog', 'System', '/administration/system',
['ROLE_Administrator', 'FUNCTION_Codes.CodeAdministration',
'FUNCTION_Configuration.ConfigurationAdministration', 'FUNCTION_Mail.MailAdministration',
'FUNCTION_Mail.MailTemplateAdministration', 'FUNCTION_Scheduler.SchedulerAdministration',
'FUNCTION_Scheduler.JobAdministration'
], [new NavigationItem('fa fa-list', 'Codes', '/administration/system/code-categories',
['ROLE_Administrator', 'FUNCTION_Codes.CodeAdministration']),
new NavigationItem('fa fa-list', 'Config', '/administration/system/config',
['ROLE_Administrator', 'FUNCTION_Config.ConfigAdministration']),
new NavigationItem('fas fa-envelope', 'Mail', '/administration/system/mail',
['ROLE_Administrator', 'FUNCTION_Mail.MailAdministration', 'FUNCTION_Mail.MailTemplateAdministration'
], [new NavigationItem('fas fa-envelope-open-text', 'Mail Templates',
'/administration/system/mail/mail-templates',
['ROLE_Administrator', 'FUNCTION_Mail.MailAdministration', 'FUNCTION_Mail.MailTemplateAdministration'
])
]), new NavigationItem('fas fa-clock', 'Scheduler', '/administration/system/scheduler',
['ROLE_Administrator', 'FUNCTION_Scheduler.SchedulerAdministration', 'FUNCTION_Scheduler.JobAdministration'
], [new NavigationItem('fas fa-cogs', 'Jobs', '/administration/system/scheduler/jobs',
['ROLE_Administrator', 'FUNCTION_Scheduler.SchedulerAdministration',
'FUNCTION_Scheduler.JobAdministration'
])
]), new NavigationItem('fas fa-file-invoice', 'Reporting', '/administration/system/reporting',
['ROLE_Administrator', 'FUNCTION_Reporting.ReportingAdministration',
'FUNCTION_Reporting.ReportDefinitionAdministration'
], [new NavigationItem('far fa-copy', 'Report Definitions',
'/administration/system/reporting/report-definitions',
['ROLE_Administrator', 'FUNCTION_Reporting.ReportingAdministration',
'FUNCTION_Reporting.ReportDefinitionAdministration'
])
])
])
]));
return navigation;
}
}
Create the src/main/frontend/src/app/views/administration directory.
mkdir -p src/main/frontend/src/app/views/administration
Create the src/main/frontend/src/app/views/administration/administration.component.ts file with the following contents.
import {Component} from '@angular/core';
/**
* The AdministrationComponent class implements the administration component.
*/
@Component({
template: `Administration
`
})
export class AdministrationComponent {
constructor() {
}
}
Create the wrapper modules for the Absa Inception Framework modules:
mkdir -p src/main/frontend/src/app/views/wrappers
import {NgModule} from '@angular/core';
import {CodesViewsModule} from '@absaoss/ngx-inception/codes';
@NgModule({
imports: [CodesViewsModule]
})
export class CodesViewsWrapperModule {
}
import {NgModule} from '@angular/core';
import {ConfigViewsModule} from '@absaoss/ngx-inception/config';
@NgModule({
imports: [ConfigViewsModule]
})
export class ConfigViewsWrapperModule {
}
import {NgModule} from '@angular/core';
import {ErrorViewsModule} from '@absaoss/ngx-inception/error';
@NgModule({
imports: [ErrorViewsModule]
})
export class ErrorViewsWrapperModule {
}
import {NgModule} from '@angular/core';
import {LoginViewsModule} from '@absaoss/ngx-inception/login';
@NgModule({
imports: [LoginViewsModule]
})
export class LoginViewsWrapperModule {
}
import {NgModule} from '@angular/core';
import {MailViewsModule} from '@absaoss/ngx-inception/mail';
@NgModule({
imports: [MailViewsModule]
})
export class MailViewsWrapperModule {
}
import {NgModule} from '@angular/core';
import {ReportingViewsModule} from '@absaoss/ngx-inception/reporting';
@NgModule({
imports: [ReportingViewsModule]
})
export class ReportingViewsWrapperModule {
}
import {NgModule} from '@angular/core';
import {SchedulerViewsModule} from '@absaoss/ngx-inception/scheduler';
@NgModule({
imports: [SchedulerViewsModule]
})
export class SchedulerViewsWrapperModule {
}
import {NgModule} from '@angular/core';
import {SecurityViewsModule} from '@absaoss/ngx-inception/security';
@NgModule({
imports: [SecurityViewsModule]
})
export class SecurityViewsWrapperModule {
}
Create the src/main/frontend/src/app/views/administration/administration.module.ts file with the following contents.
import {CommonModule} from '@angular/common';
import {NgModule} from '@angular/core';
import {RouterModule, Routes} from '@angular/router';
import {CodeCategoriesTitleResolver} from '@absaoss/ngx-inception/codes';
import {ConfigsTitleResolver} from '@absaoss/ngx-inception/config';
import {MailTitleResolver} from '@absaoss/ngx-inception/mail';
import {ReportingTitleResolver} from '@absaoss/ngx-inception/reporting';
import {SchedulerTitleResolver} from '@absaoss/ngx-inception/scheduler';
import {SecurityTitleResolver} from '@absaoss/ngx-inception/security';
import {AdministrationTitleResolver} from './administration-title-resolver';
import {AdministrationComponent} from './administration.component';
import {SystemTitleResolver} from './system-title-resolver';
const routes: Routes = [{
path: '',
pathMatch: 'full',
component: AdministrationComponent,
resolve: {
title: AdministrationTitleResolver
}
}, {
path: 'security',
resolve: {
title: SecurityTitleResolver
},
loadChildren: () => import('../wrappers/security-views-wrapper.module').then(m => m.SecurityViewsWrapperModule)
}, {
path: 'system',
resolve: {
title: SystemTitleResolver
},
children: [{
path: 'code-categories',
resolve: {
title: CodeCategoriesTitleResolver
},
loadChildren: () => import('../wrappers/codes-views-wrapper.module').then(m => m.CodesViewsWrapperModule)
}, {
path: 'config',
resolve: {
title: ConfigsTitleResolver
},
loadChildren: () => import('../wrappers/config-views-wrapper.module').then(
m => m.ConfigViewsWrapperModule)
}, {
path: 'mail',
resolve: {
title: MailTitleResolver
},
loadChildren: () => import('../wrappers/mail-views-wrapper.module').then(m => m.MailViewsWrapperModule)
}, {
path: 'scheduler',
resolve: {
title: SchedulerTitleResolver
},
loadChildren: () => import('../wrappers/scheduler-views-wrapper.module').then(m => m.SchedulerViewsWrapperModule)
}, {
path: 'reporting',
resolve: {
title: ReportingTitleResolver
},
loadChildren: () => import('../wrappers/reporting-views-wrapper.module').then(m => m.ReportingViewsWrapperModule)
}
]
}
];
@NgModule({
imports: [CommonModule, RouterModule.forChild(routes)
],
declarations: [AdministrationComponent],
providers: [
// Resolvers
CodeCategoriesTitleResolver, ConfigsTitleResolver, MailTitleResolver, ReportingTitleResolver,
SchedulerTitleResolver, SecurityTitleResolver, SystemTitleResolver
]
})
export class AdministrationModule {
}
Create the src/main/frontend/src/app/views/administration/administration-title-resolver.ts file with the following contents.
import {Injectable} from '@angular/core';
import {ActivatedRouteSnapshot, Resolve, RouterStateSnapshot} from '@angular/router';
import {Observable, of} from 'rxjs';
/**
* The AdministrationTitleResolver class provides the route data resolver that resolves the
* title for the "Administration" route in the navigation hierarchy.
*/
@Injectable()
export class AdministrationTitleResolver implements Resolve<string> {
/**
* Constructs a new AdministrationTitleResolver.
*/
constructor() {
}
/**
* Resolve the title.
*
* @param activatedRouteSnapshot The activate route snapshot.
* @param routerStateSnapshot The router state snapshot.
*/
resolve(activatedRouteSnapshot: ActivatedRouteSnapshot,
routerStateSnapshot: RouterStateSnapshot): Observable<string> {
return of('Administration');
}
}
Create the src/main/frontend/src/app/views/administration/system-title-resolver.ts file with the following contents.
import {Injectable} from '@angular/core';
import {ActivatedRouteSnapshot, Resolve, RouterStateSnapshot} from '@angular/router';
import {Observable, of} from 'rxjs';
/**
* The SystemTitleResolver class provides the route data resolver that resolves the
* title for the "System" route in the navigation hierarchy.
*/
@Injectable()
export class SystemTitleResolver implements Resolve<string> {
/**
* Constructs a new SystemTitleResolver.
*/
constructor() {
}
/**
* Resolve the title.
*
* @param activatedRouteSnapshot The activate route snapshot.
* @param routerStateSnapshot The router state snapshot.
*/
resolve(activatedRouteSnapshot: ActivatedRouteSnapshot,
routerStateSnapshot: RouterStateSnapshot): Observable<string> {
return of('System');
}
}
Create the src/main/frontend/src/app/views/dashboard directory.
mkdir -p src/main/frontend/src/app/views/dashboard
Create the src/main/frontend/src/app/views/dashboard/dashboard.component.ts file with the following contents.
import {Component} from '@angular/core';
/**
* The DashboardComponent class implements the dashboard component.
*/
@Component({
template: `Dashboard
`
})
export class DashboardComponent {
constructor() {
}
}
Create the src/main/frontend/src/app/views/dashboard/dashboard.module.ts file with the following contents.
import {CommonModule} from '@angular/common';
import {NgModule} from '@angular/core';
import {RouterModule, Routes} from '@angular/router';
import {DashboardComponent} from './dashboard.component';
const routes: Routes = [{
path: '',
component: DashboardComponent
}
];
@NgModule({
imports: [CommonModule,
RouterModule.forChild(routes)
],
declarations: [DashboardComponent]
})
export class DashboardModule {
}
Replace the contents of the src/main/frontend/src/app/app-routing.module.ts file with the following.
import {NgModule} from '@angular/core';
import {RouterModule, Routes} from '@angular/router';
import {
AdminContainerComponent, CanActivateFunctionGuard, NotFoundComponent, SimpleContainerComponent
} from '@absaoss/ngx-inception/core';
import {UserProfileComponent} from '@absaoss/ngx-inception/security';
import {AdministrationTitleResolver} from './views/administration/administration-title-resolver';
export const routes: Routes = [{
path: '',
pathMatch: 'full',
redirectTo: 'dashboard',
}, {
path: '',
component: AdminContainerComponent,
children: [{
path: 'profile',
component: UserProfileComponent
},
{
path: 'dashboard',
canActivate: [CanActivateFunctionGuard
],
data: {
title: 'Dashboard',
authorities: ['ROLE_Administrator', 'FUNCTION_Dashboard.Dashboard']
},
loadChildren: () => import('./views/dashboard/dashboard.module').then(m => m.DashboardModule)
}, {
path: 'administration',
resolve: {
title: AdministrationTitleResolver
},
loadChildren: () => import('./views/administration/administration.module').then(m => m.AdministrationModule)
}
]
},
// Login route
{
path: 'login',
component: SimpleContainerComponent,
loadChildren: () => import('./views/wrappers/login-views-wrapper.module').then(m => m.LoginViewsWrapperModule)
},
// Send Error Report route
{
path: 'error',
component: SimpleContainerComponent,
loadChildren: () => import('./views/wrappers/error-views-wrapper.module').then(m => m.ErrorViewsWrapperModule)
},
// Default route for invalid paths
{
path: '**',
component: NotFoundComponent
}
];
@NgModule({
// Tracing should only be enabled for DEBUG purposes
imports: [
// Angular modules
RouterModule.forRoot(routes, {enableTracing: false})
],
exports: [RouterModule],
providers: [AdministrationTitleResolver]
})
export class AppRoutingModule {
}
Delete the src/main/frontend/src/app/app.component.html, src/main/frontend/src/app/app.component.scss, and src/main/frontend/src/app/app.component.spec.ts files.
rm -f src/main/frontend/src/app/app.component.html
rm -f src/main/frontend/src/app/app.component.scss
rm -f src/main/frontend/src/app/app.component.spec.ts
Replace the contents of the src/main/frontend/src/app/app.component.ts file with the following.
import {Component, OnDestroy, OnInit} from '@angular/core';
import {NavigationEnd, Router} from '@angular/router';
import {Subject} from 'rxjs';
import {takeUntil} from 'rxjs/operators';
@Component({
// tslint:disable-next-line
selector: 'body',
template: `
<router-outlet></router-outlet>`
})
export class AppComponent implements OnInit, OnDestroy {
// tslint:disable-next-line
private unsubscribe$: Subject<any> = new Subject();
constructor(private router: Router) {
}
ngOnDestroy(): void {
this.unsubscribe$.next();
this.unsubscribe$.complete();
}
ngOnInit(): void {
this.router.events.pipe(takeUntil(this.unsubscribe$)).subscribe((evt) => {
if (!(evt instanceof NavigationEnd)) {
return;
}
window.scrollTo(0, 0);
});
}
}
Create the src/main/frontend/src/assets/images and src/main/frontend/src/assets/scss directories.
mkdir -p src/main/frontend/src/assets/images
mkdir -p src/main/frontend/src/assets/scss
Copy the image for the logo to src/main/frontend/src/assets/images/logo.png and the image for the logo symbol to src/main/frontend/src/assets/images/logo-symbol.png.
NOTE: If you do not have application-specific logo and logo symbol images then you can copy the ones from the ngx-inception
cp src/main/frontend/node_modules/@absaoss/ngx-inception/assets/images/logo.png src/main/frontend/src/assets/images/logo.png
cp src/main/frontend/node_modules/@absaoss/ngx-inception/assets/images/logo-symbol.png src/main/frontend/src/assets/images/logo-symbol.png
Replace the contents of the src/main/frontend/src/styles.scss file with the following.
@import "~@absaoss/ngx-inception/assets/scss/default-theme.scss";
.brand-full {
display: inline-block;
width: 100px;
height: 13px;
background-image: url("assets/images/logo.png");
background-size: 100% auto;
background-repeat: no-repeat;
}
.brand-minimized {
display: inline-block;
width: 30px;
height: 30px;
background-image: url("assets/images/logo-symbol.png");
background-size: 100% auto;
background-repeat: no-repeat;
}
.logo-login {
background-image: url("assets/images/logo.png");
}
span.copyright-name:before
{
content: "My Company";
}
Replace the contents of the src/main/frontend/src/environments/environment.ts file with the following.
export const environment = {
production: false,
// Inception API endpoints
inception_oauthTokenUrl: 'http://localhost:8080/oauth/token',
inception_codesApiUrlPrefix: 'http://localhost:8080/api/codes',
inception_configApiUrlPrefix: 'http://localhost:8080/api/config',
inception_errorApiUrlPrefix: 'http://localhost:8080/api/error',
inception_mailApiUrlPrefix: 'http://localhost:8080/api/mail',
inception_partyApiUrlPrefix: 'http://localhost:8080/api/party',
inception_partyReferenceApiUrlPrefix: 'http://localhost:8080/api/party/reference',
inception_referenceApiUrlPrefix: 'http://localhost:8080/api/reference',
inception_reportingApiUrlPrefix: 'http://localhost:8080/api/reporting',
inception_schedulerApiUrlPrefix: 'http://localhost:8080/api/scheduler',
inception_securityApiUrlPrefix: 'http://localhost:8080/api/security',
};
/*
* For easier debugging in development mode, we import the following file to ignore
* zone related error stack frames such as `zone.run`, `zoneDelegate.invokeTask`.
*/
import 'zone.js/plugins/zone-error';
Replace the contents of the src/main/frontend/src/environments/environment.prod.ts file with the following.
export const environment = {
production: true,
// Inception API endpoints
inception_oauthTokenUrl: '/oauth/token',
inception_codesApiUrlPrefix: '/api/codes',
inception_configApiUrlPrefix: '/api/config',
inception_errorApiUrlPrefix: '/api/error',
inception_mailApiUrlPrefix: '/api/mail',
inception_partyApiUrlPrefix: '/api/party',
inception_partyReferenceApiUrlPrefix: '/api/party/reference',
inception_referenceApiUrlPrefix: '/api/reference',
inception_reportingApiUrlPrefix: '/api/reporting',
inception_schedulerApiUrlPrefix: '/api/scheduler',
inception_securityApiUrlPrefix: '/api/security',
};
Execute the following command under the src directory to build the new application.
mvn package
FAQs
Unknown package
We found that africa.absa:inception 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
Snyk's use of malicious npm packages for research raises ethical concerns, highlighting risks in public deployment, data exfiltration, and unauthorized testing.
Research
Security News
Socket researchers found several malicious npm packages typosquatting Chalk and Chokidar, targeting Node.js developers with kill switches and data theft.
Security News
pnpm 10 blocks lifecycle scripts by default to improve security, addressing supply chain attack risks but sparking debate over compatibility and workflow changes.