diff --git a/.circleci/config.yml b/.circleci/config.yml
new file mode 100644
index 0000000..91e9188
--- /dev/null
+++ b/.circleci/config.yml
@@ -0,0 +1,37 @@
+# Java Gradle CircleCI 2.0 configuration file
+#
+# Check https://circleci.com/docs/2.0/language-java/ for more details
+#
+version: 2
+jobs:
+ build:
+ docker:
+ # specify the version you desire here
+ - image: circleci/openjdk:8-jdk
+
+ # Specify service dependencies here if necessary
+ # CircleCI maintains a library of pre-built images
+ # documented at https://circleci.com/docs/2.0/circleci-images/
+ # - image: circleci/postgres:9.4
+
+ working_directory: ~/repo
+
+ steps:
+ - checkout
+
+ # Download and cache dependencies
+ - restore_cache:
+ keys:
+ - v1-dependencies-{{ checksum "build.gradle" }}
+ # fallback to using the latest cache if no exact match is found
+ - v1-dependencies-
+
+ - run: gradle dependencies
+
+ - save_cache:
+ paths:
+ - ~/.gradle
+ key: v1-dependencies-{{ checksum "build.gradle" }}
+
+ # run tests!
+ - run: gradle test
\ No newline at end of file
diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS
new file mode 100644
index 0000000..222702d
--- /dev/null
+++ b/.github/CODEOWNERS
@@ -0,0 +1,4 @@
+#
+# Team and People to notify
+#
+* @amatiushkin
\ No newline at end of file
diff --git a/.github/CODE_OF_CONDUCT.md b/.github/CODE_OF_CONDUCT.md
new file mode 100644
index 0000000..920c7d3
--- /dev/null
+++ b/.github/CODE_OF_CONDUCT.md
@@ -0,0 +1,32 @@
+Open source projects are “living.” Contributions in the form of issues and pull requests are welcomed and encouraged. When you contribute, you explicitly say you are part of the community and abide by its Code of Conduct.
+
+# The Code
+
+At Intuit, we foster a kind, respectful, harassment-free cooperative community. Our open source community works to:
+
+- Be kind and respectful;
+- Act as a global community;
+- Conduct ourselves professionally.
+
+As members of this community, we will not tolerate behaviors including, but not limited to:
+
+- Violent threats or language;
+- Discriminatory or derogatory jokes or language;
+- Public or private harassment of any kind;
+- Other conduct considered inappropriate in a professional setting.
+
+## Reporting Concerns
+
+If you see someone violating the Code of Conduct please email TechOpenSource@intuit.com
+
+## Scope
+
+This code of conduct applies to:
+
+All repos and communities for Intuit-managed projects, whether or not the text is included in a Intuit-managed project’s repository;
+
+Individuals or teams representing projects in official capacity, such as via official social media channels or at in-person meetups.
+
+## Attribution
+
+This Code of Conduct is partly inspired by and based on those of Amazon, CocoaPods, GitHub, Microsoft, thoughtbot, and on the Contributor Covenant version 1.4.1.
diff --git a/.github/CONTRIBUTING.md b/.github/CONTRIBUTING.md
new file mode 100644
index 0000000..3b962e7
--- /dev/null
+++ b/.github/CONTRIBUTING.md
@@ -0,0 +1,23 @@
+Thanks for contributing to graphql-java!
+
+
+Please be sure that you read the [Code of Conduct](CODE_OF_CONDUCT.md) before contributing to this project and please
+create a new Issue and discuss first what your are planning to do for bigger changes.
+
+
+The overall goal of traverser is to implement support for specific set of use cases in a production ready way.
+
+In order to achieve that we have a strong focus on maintainability and high test coverage:
+
+- We expect new or modified unit test for every change (written in [Spock](http://spockframework.org/)) and jUnit.
+
+- Low dependencies footprint is a must.
+
+- traverser is dedicated to specific use cases - expect that some suggestions will be out of scope
+
+
+For bug reports or specific code related topics create a new issue.
+
+Thanks!
+
+
diff --git a/.github/ISSUE_TEMPLATE.md b/.github/ISSUE_TEMPLATE.md
new file mode 100644
index 0000000..5eb2788
--- /dev/null
+++ b/.github/ISSUE_TEMPLATE.md
@@ -0,0 +1 @@
+If you are sure you have found a bug, please make sure you follow [contributing](CONTRIBUTING.md) guidelines.
diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md
new file mode 100644
index 0000000..561e004
--- /dev/null
+++ b/.github/PULL_REQUEST_TEMPLATE.md
@@ -0,0 +1,27 @@
+
+
+**1. Issue Link:**
+
+
+**2. Brief explanation of a change:**
+
+
+**3. Will it break existing clients and code in production?**
diff --git a/.github/RELEASE_TEMPLATE.md b/.github/RELEASE_TEMPLATE.md
new file mode 100644
index 0000000..c6bc359
--- /dev/null
+++ b/.github/RELEASE_TEMPLATE.md
@@ -0,0 +1,23 @@
+## Location
+
+Maven:
+```
+com.intuit.commons
+traverser
+NEXT_VERSION
+```
+
+Gradle:
+```
+implementation 'com.intuit.commons:traverser:NEXT_VERSION'
+```
+
+## Details of what's new
+
+Release notes: https://github.intuit.com/services-java/traverser/blob/NEXT_VERSION/RELEASE.md
+
+## Summary of changes
+
+- add brief one-line explanation of a change
+- ???
+- PROFIT
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..c178f38
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,17 @@
+.gradle
+gradle
+*.iml
+.idea
+*.class
+.DS_Store
+.classpath
+.project
+.settings/
+.nb-gradle/
+*/out/
+*.iws
+gradlew
+gradlew.bat
+build
+out
+gradle.properties
\ No newline at end of file
diff --git a/CHANGELOG b/CHANGELOG
new file mode 100644
index 0000000..12c253a
--- /dev/null
+++ b/CHANGELOG
@@ -0,0 +1,3 @@
+#
+# Traverser Changelog
+#
\ No newline at end of file
diff --git a/LICENSE b/LICENSE
new file mode 100644
index 0000000..7a2f164
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,202 @@
+
+ Apache License
+ Version 2.0, January 2004
+ http://www.apache.org/licenses/
+
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+ 1. Definitions.
+
+ "License" shall mean the terms and conditions for use, reproduction,
+ and distribution as defined by Sections 1 through 9 of this document.
+
+ "Licensor" shall mean the copyright owner or entity authorized by
+ the copyright owner that is granting the License.
+
+ "Legal Entity" shall mean the union of the acting entity and all
+ other entities that control, are controlled by, or are under common
+ control with that entity. For the purposes of this definition,
+ "control" means (i) the power, direct or indirect, to cause the
+ direction or management of such entity, whether by contract or
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
+ outstanding shares, or (iii) beneficial ownership of such entity.
+
+ "You" (or "Your") shall mean an individual or Legal Entity
+ exercising permissions granted by this License.
+
+ "Source" form shall mean the preferred form for making modifications,
+ including but not limited to software source code, documentation
+ source, and configuration files.
+
+ "Object" form shall mean any form resulting from mechanical
+ transformation or translation of a Source form, including but
+ not limited to compiled object code, generated documentation,
+ and conversions to other media types.
+
+ "Work" shall mean the work of authorship, whether in Source or
+ Object form, made available under the License, as indicated by a
+ copyright notice that is included in or attached to the work
+ (an example is provided in the Appendix below).
+
+ "Derivative Works" shall mean any work, whether in Source or Object
+ form, that is based on (or derived from) the Work and for which the
+ editorial revisions, annotations, elaborations, or other modifications
+ represent, as a whole, an original work of authorship. For the purposes
+ of this License, Derivative Works shall not include works that remain
+ separable from, or merely link (or bind by name) to the interfaces of,
+ the Work and Derivative Works thereof.
+
+ "Contribution" shall mean any work of authorship, including
+ the original version of the Work and any modifications or additions
+ to that Work or Derivative Works thereof, that is intentionally
+ submitted to Licensor for inclusion in the Work by the copyright owner
+ or by an individual or Legal Entity authorized to submit on behalf of
+ the copyright owner. For the purposes of this definition, "submitted"
+ means any form of electronic, verbal, or written communication sent
+ to the Licensor or its representatives, including but not limited to
+ communication on electronic mailing lists, source code control systems,
+ and issue tracking systems that are managed by, or on behalf of, the
+ Licensor for the purpose of discussing and improving the Work, but
+ excluding communication that is conspicuously marked or otherwise
+ designated in writing by the copyright owner as "Not a Contribution."
+
+ "Contributor" shall mean Licensor and any individual or Legal Entity
+ on behalf of whom a Contribution has been received by Licensor and
+ subsequently incorporated within the Work.
+
+ 2. Grant of Copyright License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ copyright license to reproduce, prepare Derivative Works of,
+ publicly display, publicly perform, sublicense, and distribute the
+ Work and such Derivative Works in Source or Object form.
+
+ 3. Grant of Patent License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ (except as stated in this section) patent license to make, have made,
+ use, offer to sell, sell, import, and otherwise transfer the Work,
+ where such license applies only to those patent claims licensable
+ by such Contributor that are necessarily infringed by their
+ Contribution(s) alone or by combination of their Contribution(s)
+ with the Work to which such Contribution(s) was submitted. If You
+ institute patent litigation against any entity (including a
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
+ or a Contribution incorporated within the Work constitutes direct
+ or contributory patent infringement, then any patent licenses
+ granted to You under this License for that Work shall terminate
+ as of the date such litigation is filed.
+
+ 4. Redistribution. You may reproduce and distribute copies of the
+ Work or Derivative Works thereof in any medium, with or without
+ modifications, and in Source or Object form, provided that You
+ meet the following conditions:
+
+ (a) You must give any other recipients of the Work or
+ Derivative Works a copy of this License; and
+
+ (b) You must cause any modified files to carry prominent notices
+ stating that You changed the files; and
+
+ (c) You must retain, in the Source form of any Derivative Works
+ that You distribute, all copyright, patent, trademark, and
+ attribution notices from the Source form of the Work,
+ excluding those notices that do not pertain to any part of
+ the Derivative Works; and
+
+ (d) If the Work includes a "NOTICE" text file as part of its
+ distribution, then any Derivative Works that You distribute must
+ include a readable copy of the attribution notices contained
+ within such NOTICE file, excluding those notices that do not
+ pertain to any part of the Derivative Works, in at least one
+ of the following places: within a NOTICE text file distributed
+ as part of the Derivative Works; within the Source form or
+ documentation, if provided along with the Derivative Works; or,
+ within a display generated by the Derivative Works, if and
+ wherever such third-party notices normally appear. The contents
+ of the NOTICE file are for informational purposes only and
+ do not modify the License. You may add Your own attribution
+ notices within Derivative Works that You distribute, alongside
+ or as an addendum to the NOTICE text from the Work, provided
+ that such additional attribution notices cannot be construed
+ as modifying the License.
+
+ You may add Your own copyright statement to Your modifications and
+ may provide additional or different license terms and conditions
+ for use, reproduction, or distribution of Your modifications, or
+ for any such Derivative Works as a whole, provided Your use,
+ reproduction, and distribution of the Work otherwise complies with
+ the conditions stated in this License.
+
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
+ any Contribution intentionally submitted for inclusion in the Work
+ by You to the Licensor shall be under the terms and conditions of
+ this License, without any additional terms or conditions.
+ Notwithstanding the above, nothing herein shall supersede or modify
+ the terms of any separate license agreement you may have executed
+ with Licensor regarding such Contributions.
+
+ 6. Trademarks. This License does not grant permission to use the trade
+ names, trademarks, service marks, or product names of the Licensor,
+ except as required for reasonable and customary use in describing the
+ origin of the Work and reproducing the content of the NOTICE file.
+
+ 7. Disclaimer of Warranty. Unless required by applicable law or
+ agreed to in writing, Licensor provides the Work (and each
+ Contributor provides its Contributions) on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+ implied, including, without limitation, any warranties or conditions
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+ PARTICULAR PURPOSE. You are solely responsible for determining the
+ appropriateness of using or redistributing the Work and assume any
+ risks associated with Your exercise of permissions under this License.
+
+ 8. Limitation of Liability. In no event and under no legal theory,
+ whether in tort (including negligence), contract, or otherwise,
+ unless required by applicable law (such as deliberate and grossly
+ negligent acts) or agreed to in writing, shall any Contributor be
+ liable to You for damages, including any direct, indirect, special,
+ incidental, or consequential damages of any character arising as a
+ result of this License or out of the use or inability to use the
+ Work (including but not limited to damages for loss of goodwill,
+ work stoppage, computer failure or malfunction, or any and all
+ other commercial damages or losses), even if such Contributor
+ has been advised of the possibility of such damages.
+
+ 9. Accepting Warranty or Additional Liability. While redistributing
+ the Work or Derivative Works thereof, You may choose to offer,
+ and charge a fee for, acceptance of support, warranty, indemnity,
+ or other liability obligations and/or rights consistent with this
+ License. However, in accepting such obligations, You may act only
+ on Your own behalf and on Your sole responsibility, not on behalf
+ of any other Contributor, and only if You agree to indemnify,
+ defend, and hold each Contributor harmless for any liability
+ incurred by, or claims asserted against, such Contributor by reason
+ of your accepting any such warranty or additional liability.
+
+ END OF TERMS AND CONDITIONS
+
+ APPENDIX: How to apply the Apache License to your work.
+
+ To apply the Apache License to your work, attach the following
+ boilerplate notice, with the fields enclosed by brackets "[]"
+ replaced with your own identifying information. (Don't include
+ the brackets!) The text should be enclosed in the appropriate
+ comment syntax for the file format. We also recommend that a
+ file or class name and description of purpose be included on the
+ same "printed page" as the copyright notice for easier
+ identification within third-party archives.
+
+ Copyright 2019 Intuit Inc.
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
diff --git a/README.md b/README.md
new file mode 100644
index 0000000..fd9a880
--- /dev/null
+++ b/README.md
@@ -0,0 +1,221 @@
+# Traverser: java library to walk object graph
+
+
+
+Traverser solves a one of the most common tasks to operate on tree or graph data structure:
+
+- flatten into collection or stream
+- perform an action during traversal
+- control traversal flow
+
+It exposes rich and fine level of capabilities like:
+- iterators
+- both depth- and breadth- first search (DFS/BFS)
+- visitors
+- local / global context
+
+It helps in several areas:
+- speed up implementation by re-using generic solution (stable, tested, well-performant solution)
+- reduce codebase
+- expand and adjust use cases with simple changes
+- decouples traversing from data structure, which boosts maintenability
+
+## Getting Started
+
+Add dependency on this module and traverse any complex data structure.
+Gradle:
+```
+ compile 'com.intuit.commons:traverser:alfa-0.1.0'
+```
+Maven:
+```
+
+ com.intuit.commons
+ traverser
+ alfa-0.1.0
+ pom
+
+```
+
+## Learn by example
+Represnting hierarchy of a medium-sized company is good example to illustrate key features of the traverser.
+```
+Company
+ Bussiness Entity
+ Teams
+ Members
+```
+
+Snapshot of a virtual company staff:
+
+```
+Company
+ BU-Pacific
+ Product Development
+ John Smith
+ Lucy Gold
+ Sylvester Stallone
+ Sales Department
+ Nick Becker
+ Ruby Klingemann
+ Management
+ Anna Schulze
+```
+
+## Explore
+
+### Integration: children provider
+
+Once of benefits of using generic traversing mechanism is to decouple actual traversing from data structure it is beeing operated on.
+Regardles of the way data is organized, core algorithms does not change. But data, root element(s) as well as immediate children for a given elements must be be known to traverser in some way, so the algorithm can move to next iteration.
+
+Children proivder or simply *\* indicates such function, which feeds traverser with children of current given element.
+It could be passed as lamda-function or method reference or any other applicable means.
+
+### Iterator
+Iteration is build on top of graph/tree traversal and shares common API.
+There are 2 flavors (go deep or go broad) of direction and 2 flavours (before or after) of invocation.
+It give 4 total possible combinations of how iteration can be performed.
+
+
+
+Iterators are useful to "flatten" object structure into stream of objects.
+
+#### Depth-First traversing
+
+Depth-first always follows reference to child, when next item on current level:
+
+
+##### Perform an action, when move to next:
+```java
+Company c = new Company(); //
+// ...
+TraversingIterator i = Traverser.depthFirst().preOrderIterator(c);
+// ...
+```
+
+Output:
+```
+Company
+BU-Pacific
+PD=Product Development
+Sylvester Stallone
+John Smith
+Lucy Gold
+SL=Sales Department
+Nick Becker
+Ruby Klingemann
+MG=Management
+Anna Schulze
+```
+
+##### Move to next, perform an action:
+```java
+Company c = new Company(); //
+// ...
+TraversingIterator i = Traverser.depthFirst().postOrderIterator(c);
+// ...
+```
+
+Output:
+```
+Sylvester Stallone
+John Smith
+Lucy Gold
+PD=Product Development
+Nick Becker
+Ruby Klingemann
+SL=Sales Department
+Anna Schulze
+MG=Management
+BU-Pacific
+Company
+```
+
+
+#### Breadth-First traversing
+
+Breadth-first always follows next item on current level, when goes to reference to child.
+##### Perform an action, when move to next:
+```java
+Company c = new Company(); //
+// ...
+TraversingIterator i = Traverser.breadthFirst().preOrderIterator(c);
+// ...
+```
+Output:
+```
+Company
+BU-Pacific
+PD=Product Development
+SL=Sales Department
+MG=Management
+Sylvester Stallone
+John Smith
+Lucy Gold
+Nick Becker
+Ruby Klingemann
+Anna Schulze
+```
+
+
+##### Move to next, perform an action:
+```java
+Company c = new Company(); //
+// ...
+TraversingIterator i = Traverser.breadthFirst().postOrderIterator(c);
+// ...
+```
+Output
+```
+Company
+BU-Pacific
+PD=Product Development
+SL=Sales Department
+MG=Management
+Sylvester Stallone
+John Smith
+Lucy Gold
+Nick Becker
+Ruby Klingemann
+Anna Schulze
+```
+### Manipulating traversal flow
+Traveser accepts [visitors](https://en.wikipedia.org/wiki/Visitor_pattern) which can be used to take action on a given node.
+
+
+#### Configuration
+
+
+It is possible to override `version` and `group` of the artifact:
+```
+./gradlew clean -Pproject.version=1.0.0-SNAPSHOT -Pproject.group=com.intuit.commons publishToMavenLocal
+```
+where `project.version` and `project.group` are used to control desired group and version.
+Artifact name is configured with `rootProject.name` property in *settings.gradle*.
+
+## Technologies Used
+
+Minimum required: Java 8
+
+This library depends on Apache `commons-collections` and `slf4j`.
+
+## Contributing Guidelines
+
+Welcome contributors!
+ [CONTRIBUTING.md](.github/CONTRIBUTING.md).
+
+## Local Development
+
+```
+./gradlew test
+```
+
+
+## Support
+
+About [opensource](https://opensource.intuit.com) at Intuit.
+
+## Legal
+
+Read more about license for this software [License](LICENSE).
diff --git a/RELEASE.md b/RELEASE.md
new file mode 100644
index 0000000..407cfe0
--- /dev/null
+++ b/RELEASE.md
@@ -0,0 +1,23 @@
+# Release notes
+Browse all [releases](https://github.intuit.com/services-java/traverser/releases).
+
+
+
+
+# alfa-0.1.0
+## Release Notes
+
+### Extract traverser code from original codebase
+
+Standalone release of the traverser as oneself artifact.
+
+----
\ No newline at end of file
diff --git a/build.gradle b/build.gradle
new file mode 100644
index 0000000..0124ab7
--- /dev/null
+++ b/build.gradle
@@ -0,0 +1,77 @@
+plugins {
+ id "java"
+ id "org.sonarqube" version "2.7"
+ id 'maven-publish'
+}
+
+
+group = findProperty("project.group") ?: "com.intuit.commons"
+version = findProperty("project.version") ?: "${new Date().format('yyyy_MMdd_HHmmss')}-SNAPSHOT"
+
+def PUBLISH_URI = findProperty("PUBLISH_URL") ?: ""
+
+allprojects {
+ repositories {
+ mavenLocal()
+ maven {
+ url "https://oss.sonatype.org/content/repositories/snapshots"
+ }
+
+ mavenCentral()
+ }
+}
+
+
+dependencies {
+
+ compile 'org.slf4j:slf4j-api:1.7.13'
+
+ // For Collections, Iterators, Predicates, etc.
+ compile 'org.apache.commons:commons-collections4:4.1'
+
+ testCompile 'org.codehaus.groovy:groovy-all:2.3.7'
+ testCompile 'org.jmockit:jmockit:1.20'
+ testCompile 'junit:junit:4.12'
+ testCompile 'org.slf4j:slf4j-simple:1.7.13'
+
+}
+
+sonarqube {
+ properties {
+ property "sonar.host.url", System.getenv("SONARQUBE_SERVER_URL")
+ property "sonar.login", System.getenv("SONARQUBE_USERNAME")
+ property "sonar.password", System.getenv("SONARQUBE_PASSWORD")
+ }
+}
+
+
+task sourcesJar(type: Jar) {
+ from sourceSets.main.allJava
+ classifier = 'sources'
+}
+
+task javadocJar(type: Jar) {
+ from javadoc
+ classifier = 'javadoc'
+}
+
+publishing {
+ publications {
+ mavenJava(MavenPublication) {
+ from components.java
+ artifact sourcesJar
+ artifact javadocJar
+ }
+
+ }
+ repositories {
+ maven {
+ url(uri(PUBLISH_URI))
+ credentials {
+ username = System.getenv("PUBLISH_REPO_USERNAME")
+ password = System.getenv("PUBLISH_REPO_PASSWORD")
+ }
+ }
+ }
+}
+
diff --git a/logo.png b/logo.png
new file mode 100644
index 0000000..eacf489
Binary files /dev/null and b/logo.png differ
diff --git a/settings.gradle b/settings.gradle
new file mode 100644
index 0000000..037cb9f
--- /dev/null
+++ b/settings.gradle
@@ -0,0 +1,3 @@
+rootProject.name = 'traverser'
+// Gradle 5.0 compatibility
+// enableFeaturePreview('STABLE_PUBLISHING')
\ No newline at end of file
diff --git a/src/main/java/com/intuit/commons/Comparables.java b/src/main/java/com/intuit/commons/Comparables.java
new file mode 100644
index 0000000..d6662fa
--- /dev/null
+++ b/src/main/java/com/intuit/commons/Comparables.java
@@ -0,0 +1,59 @@
+/**
+ * Copyright 2019 Intuit Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package com.intuit.commons;
+
+import java.util.Comparator;
+import java.util.Objects;
+
+/**
+ *
+ * @author gkesler
+ */
+public class Comparables {
+ // disable instantiation
+ private Comparables () {
+ }
+
+ /**
+ * Compares two enum values and returns one with the highest ordinal value
+ *
+ * @param type of Comparable objects in this method
+ *
+ * @param left LHS of comparison
+ * @param right RHS of comparison
+ * @return left argument if its ordinal value is highest, or right otherwise
+ */
+ public static > E max (E left, E right) {
+ return Objects.compare(left, right, Comparator.naturalOrder()) > 0
+ ? left
+ : right;
+ }
+
+ /**
+ * Compares two enum values and returns one with the lowest ordinal value
+ *
+ * @param type of Comparable objects in this method
+ *
+ * @param left LHS of comparison
+ * @param right RHS of comparison
+ * @return left argument if its ordinal value is lowest, or right otherwise
+ */
+ public static > E min (E left, E right) {
+ return Objects.compare(left, right, Comparator.naturalOrder()) < 0
+ ? left
+ : right;
+ }
+}
diff --git a/src/main/java/com/intuit/commons/TriConsumer.java b/src/main/java/com/intuit/commons/TriConsumer.java
new file mode 100644
index 0000000..9d381f1
--- /dev/null
+++ b/src/main/java/com/intuit/commons/TriConsumer.java
@@ -0,0 +1,36 @@
+/**
+ * Copyright 2019 Intuit Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package com.intuit.commons;
+
+import java.util.Objects;
+
+/**
+ *
+ * @author gkesler
+ */
+@FunctionalInterface
+public interface TriConsumer {
+ void accept (X x, Y y, Z z);
+
+ default TriConsumer andThen (TriConsumer super X, ? super Y, ? super Z> after) {
+ Objects.requireNonNull(after);
+
+ return (X x, Y y, Z z) -> {
+ accept(x, y, z);
+ after.accept(x, y, z);
+ };
+ }
+}
diff --git a/src/main/java/com/intuit/commons/package-info.java b/src/main/java/com/intuit/commons/package-info.java
new file mode 100644
index 0000000..ce82ad6
--- /dev/null
+++ b/src/main/java/com/intuit/commons/package-info.java
@@ -0,0 +1,6 @@
+/**
+ * Commons - a place for generic software components and solutions.
+ *
+ * These components are decoupled from any business logic.
+ */
+package com.intuit.commons;
\ No newline at end of file
diff --git a/src/main/java/com/intuit/commons/traverser/AbstractTraverseContextQueue.java b/src/main/java/com/intuit/commons/traverser/AbstractTraverseContextQueue.java
new file mode 100644
index 0000000..3e38cdf
--- /dev/null
+++ b/src/main/java/com/intuit/commons/traverser/AbstractTraverseContextQueue.java
@@ -0,0 +1,45 @@
+/**
+ * Copyright 2019 Intuit Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package com.intuit.commons.traverser;
+
+import java.util.Objects;
+import java.util.Queue;
+
+/**
+ *
+ * @author gkesler
+ * @param traversable type
+ * @param implementing type of a memory structure (stack ({@link java.util.Deque}) or queue)
+ */
+
+abstract public class AbstractTraverseContextQueue>> implements TraverseContextQueue {
+ AbstractTraverseContextQueue(Q impl, Traverser super T, ? super TraverseContext> outer) {
+ this.impl = Objects.requireNonNull(impl);
+ this.outer = Objects.requireNonNull((Traverser>)outer);
+ }
+
+ @Override
+ public final boolean isEmpty() {
+ return impl.isEmpty();
+ }
+
+ protected final TraverseContext newPostOrderContext(TraverseContext super T> preOrder) {
+ return outer.newPostOrderContext(preOrder);
+ }
+
+ protected final Q impl;
+ protected final Traverser> outer;
+}
diff --git a/src/main/java/com/intuit/commons/traverser/BreadthFirstQueue.java b/src/main/java/com/intuit/commons/traverser/BreadthFirstQueue.java
new file mode 100644
index 0000000..586e5c4
--- /dev/null
+++ b/src/main/java/com/intuit/commons/traverser/BreadthFirstQueue.java
@@ -0,0 +1,58 @@
+/**
+ * Copyright 2019 Intuit Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package com.intuit.commons.traverser;
+
+
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Optional;
+import java.util.Queue;
+import java.util.stream.Stream;
+
+/**
+ *
+ * @author gkesler
+ */
+class BreadthFirstQueue extends AbstractTraverseContextQueue>> {
+ public BreadthFirstQueue(Traverser super T, ? super TraverseContext> outer) {
+ super(new LinkedList<>(), outer);
+ }
+
+ @Override
+ public TraverseContext pop() {
+ return impl.remove();
+ }
+
+ @Override
+ public TraverseContext peek() {
+ return impl.peek();
+ }
+
+ @Override
+ public void pushAll(Optional> parent, Stream> contexts) {
+ // let parent have children: {child1, child2, ..., childN}
+ // let stack have state: {ctx1, ctx2, ctx3, ..., ctxN}
+ // BFS algorithm implies that the most recent nodes are visited last
+ // therefore we need to append children contexts followed by the parent post-order context
+ // at the tail of the queue and let the older content remain closer to the head of the queue
+ List> tail = (List>)impl;
+ contexts
+ .forEach(tail::add);
+ parent
+ .map(this::newPostOrderContext)
+ .ifPresent(tail::add);
+ }
+}
diff --git a/src/main/java/com/intuit/commons/traverser/DepthFirstQueue.java b/src/main/java/com/intuit/commons/traverser/DepthFirstQueue.java
new file mode 100644
index 0000000..e485eb2
--- /dev/null
+++ b/src/main/java/com/intuit/commons/traverser/DepthFirstQueue.java
@@ -0,0 +1,57 @@
+/**
+ * Copyright 2019 Intuit Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package com.intuit.commons.traverser;
+
+import java.util.Deque;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Optional;
+import java.util.stream.Stream;
+
+/**
+ *
+ * @author gkesler
+ */
+class DepthFirstQueue extends AbstractTraverseContextQueue>> {
+ public DepthFirstQueue(Traverser super T, ? super TraverseContext> outer) {
+ super(new LinkedList<>(), outer);
+ }
+
+ @Override
+ public TraverseContext pop() {
+ return impl.pop();
+ }
+
+ @Override
+ public TraverseContext peek() {
+ return impl.peek();
+ }
+
+ @Override
+ public void pushAll(Optional> parent, Stream> contexts) {
+ // let parent have children: {child1, child2, ..., childN}
+ // let stack have state: {ctx1, ctx2, ctx3, ..., ctxN}
+ // DFS algorithm implies that the most recent nodes are visited first
+ // therefore we need to insert children contexts followed by the parent post-order context
+ // at the head of the queue and push the older content further to the end of the queue
+ List> head = ((List>)impl).subList(0, 0);
+ contexts
+ .forEach(head::add);
+ parent
+ .map(this::newPostOrderContext)
+ .ifPresent(head::add);
+ }
+}
diff --git a/src/main/java/com/intuit/commons/traverser/Markers.java b/src/main/java/com/intuit/commons/traverser/Markers.java
new file mode 100644
index 0000000..b4859d1
--- /dev/null
+++ b/src/main/java/com/intuit/commons/traverser/Markers.java
@@ -0,0 +1,31 @@
+/**
+ * Copyright 2019 Intuit Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package com.intuit.commons.traverser;
+
+/**
+ * Mark special results from Visitor
+ */
+public enum Markers {
+
+ /**
+ * Tells the traversal process to skip children
+ */
+ SKIP,
+ /**
+ * Tells the traversal process to quit traversing
+ */
+ QUIT
+}
diff --git a/src/main/java/com/intuit/commons/traverser/TraverseContext.java b/src/main/java/com/intuit/commons/traverser/TraverseContext.java
new file mode 100644
index 0000000..90c27e1
--- /dev/null
+++ b/src/main/java/com/intuit/commons/traverser/TraverseContext.java
@@ -0,0 +1,261 @@
+/**
+ * Copyright 2019 Intuit Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package com.intuit.commons.traverser;
+
+import java.util.Iterator;
+import java.util.Map;
+import java.util.NoSuchElementException;
+import java.util.Objects;
+import java.util.Optional;
+import java.util.Spliterators;
+import java.util.function.Function;
+import java.util.stream.Stream;
+import java.util.stream.StreamSupport;
+
+/**
+ * Encapsulates current state of traversal.
+ * Context wraps the current graph element being entered (pre-order) or left (post-order)
+ * along with the convenient methods to navigate through the parent contexts all the way
+ * to the root/source graph node, store traversal result, maintain traversal internal variables and more.
+ *
+ * @param type of the graph node being wrapped
+ *
+ * @author gkesler
+ */
+public interface TraverseContext {
+ /**
+ * Returns the graph node associated with this Context instance
+ *
+ * @return wrapped graph node
+ */
+ T thisNode();
+ /**
+ * Returns previous context.
+ * Contexts are organized as a linked lists that start from the Context for
+ * wrapped the graph element used to start traversing all the way down to the
+ * current Context.
+ *
+ * @return parent Context
+ */
+ TraverseContext parentContext();
+ /**
+ * A shortcut to obtain result of immediate parent.
+ * For performance and usability reasons, traverser guarantees that for every
+ * Context being traversed there always be a parent, so null check for parent Context is not needed.
+ *
+ * @param expected type of result
+ *
+ * @return parent result
+ */
+ default U getParentResult() {
+ return parentContext().getResult();
+ }
+ /**
+ * A shortcut to set result of immediate parent.
+ * For performance and usability reasons, traverser guarantees that for every
+ * Context being traversed there always be a parent, so null check for parent Context is not needed.
+ *
+ * @param expected type of result
+ * @param o parent result value
+ */
+ default void setParentResult(U o) {
+ parentContext().setResult(o);
+ }
+ /**
+ * Sets result for the current context.
+ * Depending on the Context implementation, results could be shared among all
+ * Contexts in a single traversal or they could be stored per each Context/graph element
+ *
+ * Default: all Contexts share the same result {@link TraverseContextBuilder#ROOT_STRATEGY}.
+ *
+ * @see TraverseContextBuilder.ContextStrategy
+ *
+ * @param expected type of result
+ * @param o result value
+ */
+ void setResult(U o);
+ /**
+ * Obtains result of this Context
+ *
+ * @param expected type of result
+ * @return result value
+ */
+ U getResult();
+ /**
+ * Retrieves local result stored in this context.
+ * Difference between this method and {@link #getResult() } is that the latter
+ * is not guaranteed to return locally stored result. It rather represents the
+ * overall centralized result acquired during entire graph traversal.
+ *
+ * @param expected type of result
+ * @return result value
+ */
+ U getContextResult ();
+ /**
+ * Obtains initial data provided when traversal was requested.
+ * Serves as the default result value in case it wasn't overwritten
+ * during traversal.
+ *
+ * @return initial result value provided by the client
+ */
+ Object initialData();
+ /**
+ * Context local variables used during traversal.
+ * Some visitors can declare their own local variables, while others can
+ * leverage this common variables map.
+ *
+ * @return variables map.
+ */
+ Map, Object> getContextVars();
+ /**
+ * A fluent API method to set parent result
+ *
+ * @param expected type of result
+ * @param o parent result value
+ * @return this instance to allow method chaining
+ */
+ default TraverseContext parentResult(U o) {
+ setParentResult(o);
+ return this;
+ }
+ /**
+ * A fluent API method to set context result
+ *
+ * @param expected type of result
+ * @param o result to save in context
+ * @return this instance to allow method chaining
+ */
+ default TraverseContext result(U o) {
+ setResult(o);
+ return this;
+ }
+ /**
+ * Shortcut to read a variable associated with the provided key
+ *
+ * @param expected type of result
+ * @param valueClass used as a key to access the variable
+ * @return variable value
+ */
+ default U getVar (Class super U> valueClass) {
+ return (U)getContextVars().get(valueClass);
+ }
+ /**
+ * Shortcut to associate a variable the provided key
+ *
+ * @param expected type of result
+ * @param valueClass used as a key to access the variable
+ * @param value variable value
+ * @return this instance to allow method chaining
+ */
+ default U setVar(Class super U> valueClass, U value) {
+ return (U)getContextVars().put(valueClass, value);
+ }
+ /**
+ * A fluent API method to set a variable value
+ *
+ * @param type of a variable
+ * @param valueClass implementing class
+ * @param value value
+ * @return current context
+ */
+ default TraverseContext var (Class super U> valueClass, U value) {
+ setVar(valueClass, value);
+ return this;
+ }
+ /**
+ * Checks if this context wraps one of the roots passed to traverse method
+ *
+ * @see Traverser
+ *
+ * @return {@code true} in case of root context
+ */
+ default boolean isRoot () {
+ return !Optional
+ .ofNullable(parentContext())
+ .flatMap(ctx -> Optional.ofNullable(ctx.thisNode()))
+ .isPresent();
+ }
+ /**
+ * Indicates that this Context is accessed after all its immediate children
+ * have been visited. This allows for post-order operations on the associated
+ * graph-element.
+ *
+ * @return {@code true} if the Context is a post-order context
+ */
+ boolean isPostOrder();
+ /**
+ * Indicates that this Context wraps a graph node that had been already seen
+ * during traversal process. This allows the traversal to prevent infinite
+ * cyclic traversals and inform the client code about detected cycle.
+ *
+ * @param visitedTracker lookup method to check if this context has been already seen
+ * @return {@code true} is the graph node had been already seen
+ */
+ boolean isBackRef(Function super TraverseContext, ? extends Optional