Member since
05-11-2016
11
Posts
16
Kudos Received
1
Solution
My Accepted Solutions
Title | Views | Posted |
---|---|---|
7088 | 03-27-2018 08:10 PM |
03-27-2018
08:10 PM
hmmm....it looks like it is an issue with the version of npm with this distro. I recently submitted a PR to update the version of node and npm in NiFi: https://github.com/apache/nifi/pull/2571. Please cherry-pick this commit and rebuild and see if this solves your problem.
... View more
02-12-2018
06:56 PM
1 Kudo
One
of the really cool features of the new Apache NiFi Registry UI are the deep
links. This capability enhances collaboration by providing shareable URL’s to various
perspectives within the NiFi Registry web application. The following
deep links are available within
the ‘administration’ perspective: /nifi-registry/administration/workflow
–
This ‘administration’ perspective displays a sortable table of all buckets the
current user is authorized to view. /nifi-registry/administration/workflow(sidenav:manage/bucket/:bucketId)
– This administration’ perspective side panel displays the requested bucket’s details
and a sortable table of all of the policies created for this :bucketId. /nifi-registry/administration/users ** –
This ‘administration’ perspective displays a sortable table of all users and
groups the current user is authorized to view. /nifi-registry/administration/users(sidenav:manage/user/:userId) **
– This administration’ perspective side panel displays the requested user’s details
and a sortable table of all of the groups to which this :usertId belongs. /nifi-registry/administration/users(sidenav:manage/group/:groupId) **
– This administration’ perspective side panel displays the requested group’s details
and a sortable table of all of the users in the :groupId. In
the ‘explorer’ perspective deep linking provides the user with the ability to
query for items (flows) for which they are authorized to view. The following deep
links are available within the ‘explorer’ perspective: /nifi-registry/explorer/grid-list – This ‘explorer’
perspective displays all items (flows) across all buckets that the current user
is authorized to view. /nifi-registry/explorer/grid-list/buckets/:bucketId – This ‘explorer’
perspective displays ALL of the items in a specified bucket that the current
user is authorized to view. /nifi-registry/explorer/grid-list/buckets/:bucketId/flows/:flowId – This ‘explorer’
perspective displays a single item (flow). ** - This deep link is unavailable when the NiFi registry is unsecured.
... View more
Labels:
12-05-2017
03:46 PM
1 Kudo
Awesome! Thanks @kdoran this was exactly what was required!
... View more
12-05-2017
03:24 PM
1 Kudo
I was following the blog by Bryan Bende https://bryanbende.com/development/2016/08/31/apache-nifi-1.0.0-kerberos-authentication and I was able to successfully setup a docker-machine boot2docker and KDC and have configured my browsers to authenticate via SPNEGO. However, when I restarted my OSX I could no longer get NiFi to authenticate via SPNEGO... So I tried: docker-machine start boot2docker
eval "$(docker-machine env boot2docker)"
But this gave me an error: Error checking TLS connection: Error checking and/or regenerating the certs: There was an error validating certificates for host "192.168.99.100:2376": dial tcp 192.168.99.100:2376: i/o timeoutYou can attempt to regenerate them using 'docker-machine regenerate-certs [name]'.Be advised that this will trigger a Docker daemon restart which will stop running containers. So I tried: docker-machine rm boot2docker
docker-machine create --driver virtualbox boot2docker
docker-machine env boot2docker
eval $(docker-machine env boot2docker)
./kdc stop
./kdc clean
./kdc build
./kdc start
$(./kdc shellinit)
kinit bbende@NIFI.APACHE.ORG When I run klist: klist
Credentials cache: API:5D669ADD-9028-441F-A6E0-C2C78DA04CC5
Principal: bbende@NIFI.APACHE.ORG
Issued Expires Principal
Dec 4 10:12:29 2017 Dec 31 20:12:27 2017 krbtgt/NIFI.APACHE.ORG@NIFI.APACHE.ORG So I think I have everything running correctly now but the browsers will not authenticate. Does anyone have any suggestions?
... View more
Labels:
- Labels:
-
Apache NiFi
08-25-2017
01:11 PM
3 Kudos
This article is the first in a series of articles discussing various topics regarding frontend UI/UX development on the Apache NiFi project (versions 1.1.2+). This series will explore some of the architecture and bootstrapping of the client side JavaScript as well as various "recipes", case studies, and examples on how to accomplish certain tasks. Introduction
Keeping up with modern web development best practices, frameworks, and anti-patterns can be intimidating. Frameworks like AngularJS, Ember.js, Knockout, Backbone.js turned the JavaScript world inside out just 3-5 years ago with their declarative styles and TDD approach. Couple that with all the ever changing ECMAscript standard, differing support of said standard in various browsers, and the fact that as memory and storage capabilities have continued to increase and at lower price points more and more functionality is expected from, and placed on, the client and you start to understand that web development is still maturing and ever changing. The tooling around web development has also added to the insanity. Polyfills to add new features/capabilities to old browsers and transpilers that offer future ECMSscript6 functionality in today's "modern" browsers. Terminology like “module bundlers vs. module loaders,” “RequireJS vs. SystemJS”, and “AMD vs. CommonJS vs. UMD vs. NativeJS(ES5 vs. ES6)”. It can quickly become overwhelming but understanding these terms is vital for web developers who desire to contribute client side code in the Apache NiFi project.
In this article we will discuss a few of the topics mentioned above and then walk through a couple of case studies complete with in depth examples and appropriate exercises. Once you have read it you should feel comfortable with:
The modular nature of NiFi's JavaScript components
How the canvas application is bootstrapped
How to add events to components on NiFi's canvas
How to add actions to NiFi's context menu NiFi's JavaScript Modules
The Apache NiFi project strives to deliver a maintainable, testable, extensible, and reusable code base. For the client the next step in working towards these long term goals was to remove any global references to NiFi's already conceptually modular JavaScript components and then resolve any circular references. This does a couple of things
Provides better encapsulation, clear dependency graph, and more maintainable code
Enables client side unit testing (decoupled modules are easily mocked)
Positions NiFi to take advantage of modern JavaScript tooling like module loaders/bundlers
Previous to Apache NiFi version 1.1.2 the `nf` namespace was added to the browser's global `window` object as a container for all NiFi client modules. As each module was loaded they would add themselves to the `nf` object. Whenever a module wanted to access methods in another module:
//Previous to Apache NiFi version 1.1.2 the way a developer would access methods from one module in a different module was to gain access to it from the global scope `nf` object.
nf.someModule.someFunction()
The issue with this approach is that there is nothing stopping developers from having two modules depend on each other and creating circular references. YUCK!
Starting with Apache NiFi 1.1.2, in order to prevent developers from inadvertently creating circular references each JavaScript module is now wrapped in an IIFE (Immediately Invoked Function Expression that runs as soon as it is defined). Once invoked, a Universal Module Definition (UMD) pattern is leveraged to detect, accept, and inject JavaScript modules loaded via
ANY module loader (like RequireJS, SystemJS, WebPack, ..., or even script tags in the document head!). Finally, the module factory returns and the module is added to the global `nf` for storage and retrieval. The reason the UMD pattern was leveraged was because it supports all of the existing JavaScript module loader/bundlers and as of this writing the nifi-web-ui client is not utilizing any of them (RequireJS, SystemJs, WebPack, ...). Eventually a module loader may be introduced (and subsequently a module bundler to replace our custom bundler...but that is a separate effort another post!) but until then modules are still being loaded via the script tags in the document head with one major difference. The order is strictly enforced.
Let's consider the following example of a JavaScript module that has jquery injected into its scope:
// Note: `root` is the browser global `window` object.
(function (root, factory) {
if (typeof define === 'function' && define.amd) {
//If an AMD module loader is requesting this module.
//NOTE: The NiFi canvas application does not use an AMD module loader(see[5]).
define(['jquery'],
function ($) {
return (root.nf.ModuleFactory = factory($));
});
} else if (typeof exports === 'object' && typeof module === 'object') {
//If CommonJs module loader is requesting this module.
//NOTE: The NiFi canvas application does not use an AMD module loader(see[5]).
module.exports = (root.nf.ModuleFactory =
factory(require('jquery')));
} else {
//If no module loader is being used then inject jquery from the global `$` namespace.
//NOTE: This is the way that the NiFi canvas application loads JS modules
root.nf.ModuleFactory = factory(root.$);
}
}(this, function ($) {
//Use the injected jquery '$'
var body = $('body');
// private method
function notHelloOrGoodbye(){
//Use the injected jquery
var body = $('body');
};
// public methods (see below)
function hi(){};
function bye(){};
// Exposed public methods
return {
hello: hi,
goodbye: bye
};
}));
Now, the experienced JavaScript developer may notice:
There is a lot of boilerplate code.
Ultimately the module was added to the global `nf` so how is this approach any different than accessing JavaScript modules from the global scope?
Each module needs the module(s) it depends on to be injected into its scope but NiFi JavaScript modules are reusable and need to be agnostic to the module loading strategy being used. Module loaders like RequireJS and SystemJS provide a type of cache which allows developers to store and retrieve JavaScript modules and UMD supports that use case. What about the case when a developer simply wants to load a NiFi JavaScript module with a script tag? It turns out that our `nf` namespace (remember the `nf` object?) is already acting as a type of cache and so by leveraging the UMD pattern and existing browser capabilities we are able to 'roll our own' JavaScript module loader in the case that our NiFi modules are being NOT being loaded with a module loader. Bootstrapping the NiFi canvas application The nf-canvas-bootstrap.js module is the last module loaded but the first piece of code that runs when the NiFi canvas starts up. It waits for the DOM ready event to be fired, then it configures and bootstraps the angular application (more on that in a follow up article), and then is responsible for initializing the rest of the modules (many of which are dynamically manipulating the DOM) . During this startup phase control of a few Javascript modules are inverted. As a matter of convenience the injected modules are listed below:
Module
Injected with a reference to:
nf-canvas-utils.js
nf-graph.js
nf-birdseye.js
nf-actions.js
nf-snippet.js
nf-canvas.js [1]
nf-quickselect.js
nf-actions.js
nf-shell.js
nf-context-menu.js
nf-component-version.js
nf-settings.js
nf-connection-configuration.js
nf-birdseye.js
nf-graph.js
nf-controller-service.js
nf-controller-services.js
nf-reporting-task.js
nf-settings.js
nf-context-menu.js
nf-actions.js
[Table 1] Ok...but why?
There are four main strategical advantages gained by leveraging the UMD pattern, decoupling circular references, and leveraging inversion of control during the NiFi bootstrapping process:
Module loading order is enforced preventing future circular references.
With the previous circular references and global variables creating test mocks was difficult at best. Now, client side test mocks and unit testing will be straight forward.
The NiFi build process was ahead of its time. It leverages maven to bundle all the JavaScript modules into a single file, minify it, and gzip compress it in order to minimize the number of http requests (as well as the size of the files requested) necessary to load the application. Today there are many, feature rich, JavaScript module loaders and bundlers that offer built in dependency management, caching (module storage and retrieval), bundling of resources (complete with dead code removal), etc. These changes to the the NiFi JavaScript modules opens up the possibility of removing some of the custom maven build processes and allows the introduction of a modern module loader/bundler in order to better align with industry best practices and to remove the burden of learning and maintaining a custom build process that web developers will not be familiar with.
NiFi JavaScript modules are reusable and agnostic to the module loading strategy being used. [5] Cool huh? Case Study: Adding event listeners to components on the canvas
In this section we will explore the
nf-context-menu.js [2] module and how the "right-click" event listener is bound to each component displayed on the NiFi canvas. Below are links to each of the NiFi canvas component types available:
nf-processor.js
nf-funnel.js
nf-label.js
nf-process-group.js
nf-remote-process-group.js
nf-port.js
Each of these NiFi canvas component type modules are responsible for several things:
Rendering the components type on the canvas
Maintaining a map of each rendered instance of their respective component types (e.g. the nf-label.js module maintains a map of all rendered instances of a label component type.)
Persisting each rendered component position
(...there are few other responsibilities that I will leave to the reader to investigate for themselves)
However, for this case study there are a couple responsibilities that deserve a special mention:
Inverting control of any injected modules and maintaining references to them (see example). This process takes place during the initialization of the nf-graph.js module [3] which is responsible for injecting the nf-context.menu.js module reference into the appropriate NiFi canvas components.
Attaching event listeners to each rendered instance of the respective component type. Let's examine how a 'contextmenu.selection' [4] listener is attached to a NiFi canvas label component and how the NiFi canvas context menu is requested:
Example: NiFi canvas 'Label' component:
The nf-graph.js module
inverts control of the nf-context-menu.js module to the nf-label.js module during its initialization (which, consequently, is the last module initialized in the bootstrapping process). When the nf-label.js module renders a 'Label' component it leverages the D3 3.x library to attach a listener to that particular component for the 'contextmenu.selection' event.
Exercise 1:
One of the Apache NiFi community members recently submitted a PR to add a 'quick select' capability to components on the canvas that listens for a 'dblclick' event on each component and then opens the configuration dialog for that component. This is a great example of how to implement event listeners and I encourage you to review the PR
here and then attempt to add your own. Case Study: Adding actions to the context menu
At this point we have a good understanding of how the NiFi canvas context menu is requested for a single component on the canvas but what about when multiple components on the canvas are selected? Let's examine how a requested NiFi canvas context menu determines what menu items are available and how the nf-context-menu.js module is able to invoke actions in the application:
Example: How NiFi context menu items are determined based on components selected on the canvas:
No matter how many components on the canvas are selected the context menu is actually requested by the component on the canvas that is right clicked. However, the list of actions available in the requested context menu are based on the selected components. So, how do we get the collection of selected components? Well, good news, the nf-canvas-utils.js module is capable of
returning a list of the currently selected components on the canvas. Ok, so now that we know how to get the selection where is it determined which menu items to include in the context menu. The way this process works is pretty straight forward. Each menu item is an object in a list. Most of the properties of a menu item object are for things like the display text and the displayed icon for a context menu item. However, there are two properties that deserve some special attention:
The `condition` function. When the show() method of the context menu is called the list of ALL possible menu item objects are processed and each condition tested and if it passes then the item is added to the menu.
The `menuItem` object's `action` property. Notice that this property is a string. Also notice (see Table [1]) that control of the nf-actions.js module has been inverted to the nf.context.menu.js module. Since JavaScript provides access to public properties on an object via bracket notation (see all property accessors here) what this `action` property string really represents is any public property of the nf-actions.js module (you can find all available public nf-action.js properties on this object here).
Exercise 2:
Add a new menu item to the context menu. (Hint: you will need to reuse an existing private condition function from the nf-context-meu.js module or you will need to add a new one. You will also need to reuse an existing publicly available nf-actions.js module function or you will need to add a new one.) Conclusion
Whew! That was a lot to cover and a lot of the information here is pretty dense. I hope you found this informative and helpful in your quest to contribute to the client side code for Apache NiFi. I look forward to addressing any questions and thanks for reading!
NOTES:
[1] The nf-canvas.js should not be injected in any other module except for nf-canvas-utils.js and any direct interaction with the nf-canvas.js should be avoided. Use the nf-canvas-utils.js module to interface with the nf-canvas.js module.
[2] The nf-context-menu.js, as you may suspect, is responsible for rendering and maintaining the available menu options listed for the rendered NiFi canvas context menu.
[3] The nf-graph.js module depends on and acts as an interface to of each of the NiFi canvas component type modules throughout the NiFi canvas application.
[4] The D3 JavaScript Library is used extensively throughout the NiFi canvas application. This article does not focus on best practices nor does it offer any in depth explanation on how D3 is implemented or its expected behavior. A nice blog discussing D3 and context menus can be found here.
[5] If a developer is building a custom processor UI and wants to use RequireJs to load nf-common.js that functionality is there. If some other developer wants to use SystemJS to load nf-common.js that functionality is supported by the JavaScript modules provided by Apache NiFi.
... View more
Labels:
05-31-2017
01:05 PM
Do you see any javascript errors in the dev tools console of your web browser?
... View more
05-11-2017
03:46 PM
4 Kudos
With the release of Apache Nifi 1.0 data flows became a multi-tenancy environment with teams of data flow "developers" working simultaneously. In these situations team members may or may not have read and write permission to all components in the flow. One pattern that has emerged to aid in separating work spaces and grouping together components for these different teams is the use of process groups (PG). The "pain point" with this type of approach is that when the Nifi canvas is loaded (or reloaded) it always shows the root group. This is frustrating for team members who want to edit a flow but only have permissions to edit components in a PG that may be nested within several other PGs. In this scenario the team member will need to load the root group, pan or search to find the PG, pan or search to find the PG, pan or.....you get the idea. It is a manual effort to find the nested PG that contains the few components a user has permissions to edit. To ease this the latest Apache Nifi 1.2 release includes the ability to link to any component(s)* anywhere in a flow. This new feature (along with multi-tenancy) allows users to easily collaborate on a data flow just by by sharing a link. Let's take a look at how to use this new feature: Example: In the screen shot below notice the URL as well as the breadcrumbs. You see that we have the "Execute Script" processor selected within the "ddd" PG (which is nested inside the root PG) and that this state is reflected in the URL. To share this processor (in this nested PG) with another (authorized) user simply copy the URL from the browser, share it with another user, and have them paste it in their browser and go! *NOTE: There is a browser and app server limit on the number of characters in a URL. As such, users may link to multiple components as long as this limit is not exceeded.
... View more
Labels:
02-24-2017
04:07 PM
2 Kudos
One of the earliest improvement requests for the Apache NiFi project was to allow a user the ability to easily align components on the canvas. Well, in the upcoming 1.2 release it's HERE! In this article I will demonstrate how to align multiple selected components either vertically or horizontally. Previously, when building a flow a user would drop components on the canvas into a layout. Typically users will attempt to organize their data flow into rows and columns. This was a tedious task since aligning these components was a manual effort of dragging and dropping components to line them up 'just right'. Let's look at a crude example of such a layout: Let's see how we can organize this flow with the new 'click to align' features. First, let's align the rows: Ok great! Now let's align the columns: BOOM! And there you have it! A nicely aligned layout for an organized data flow. I hope you have found this tutorial useful and I look forward to your feedback.
... View more
Labels:
12-12-2016
09:21 PM
4 Kudos
Status, Icons, and Color: The Apache Nifi 1.0 release brought its users a welcome refresh of the UI with emphasis on a flatter more modern design that included muting the color palette. While these updates have been well received the community around Apache Nifi requested that we "bring some color back". Well we listened and in Apache Nifi 1.1 we have introduced more color in the UI for easier recognition of status changes and necessary user actions! One area (identified by the community) where color makes sense are the icons. In many cases in the Nifi UI icons provide valuable feedback worthy of drawing the attention from the user. However not all icons meet this requirement. To differentiate it was decided that wherever icons are reporting status or require the user to take an action then the icon should receive color and if an icon invokes an action in the UI then it should not. Let's take a look at the canvas: At first glance you will notice the red bulletins and yellow warning icons "jump out" to the user. These icons report status and require user action so they have been given a brighter color to draw the user's eye. Looking a bit closer you will notice the flow status bar as well as the status icons on the face of the components (and connections) on the canvas. Notice that whenever the values are 0 the icons are still a muted blue color but when something is running, stopped, enabled, or transmitting the icons are now colored green, red, or light blue (for both enabled and transmitting) respectively. This subtle change allows the user to quickly gain context as to what is going on in their flow. Finally, we see that the icons in the 'Operate pallet' and the 'Context Menu' have retained a dark blue color because these invoke an action and do not report status. Let's also check out the Summary shell: Again you can see that the action invoking icons are a dark blue while the status reporting icons have much brighter colors allowing the user to quickly gain context about their flow. In this article we have talked about icon color, what it means, and when it is appropriate. Users will see that this design concept is implemented throughout the Nifi UI and should be followed for any future features to come. I hope you have found this article helpful and thanks for reading!
... View more
Labels: