If you're reading this page then you've either applied for a position on our product team, or you're thinking about it. Welcome!

This page provides detailed information about our technical stack and development environment. These details will, obviously, have a big impact on what day-to-day life on our team is like and it's only natural that you'd want to know about them in advance.

Software is an ever-changing field and some of these details may have changed by the time you read this!

Who we’re looking for

We are looking for smart people who enjoy complex problems, learning new things, and have high emotional intelligence. We like to get things done while taking care of each other and joking around in the process.

We love it when our employees find innovative ways to solve problems or to delight our users and customers.

Development Process

We like to think of ourselves as "small 'a'" agile. We believe in the core tenets of the Agile Manifesto but are not strict followers of any specific methodology, and we shift our process based on business needs.

Sometimes we follow a regular release cadence, using a form of Kanban to track work.

Other times we work in two-week iterations, influenced by Scrum.

In all cases, we believe that cross-functional teams, working in collaborative environments and with short feedback cycles, is the key to building quality software and maintaining a stable velocity over time.

Programming Languages / Platforms / Architectures

LearningBuilder consists of:

The monolith was originally written in .NET Framework using ASP.NET MVC. All new UI features are being written in Vue.js and we are in the process of upgrading to .NET 6/.Net Core.

We use:

Developers and testers can run the full stack locally, except for the cloud pieces.

Data Access

All database schema changes are fully scripted (using a homegrown utility) and are source-controlled like everything else. We maintain a shared database snapshot in a known good state so that you're only ever one command-line tool away from restoring a local database with staged data.

Our legacy features use an ORM called NHibernate to interact with the database. Queries are typically expressed using the NH LINQ provider, but when necessary we'll hand-roll a query to optimize performance. 

We also work with stored procedures a decent amount; LearningBuilder is a very configurable product, and sprocs are often used as a way of injecting client-specific business logic into pre-defined "hooks" in the app.

Automated Testing

Automated testing is a big deal on our team. We switch between TDD, "test-first", and "test-after" workflows depending on the nature of the changes we're making. Regardless, the goals are always to:

  1. Prove our code does what we expect it does

  2. Document our intent to aid other programmers

  3. Prevent regressions and aid in issue triage

As of this writing we have many thousands of unit tests and many hundreds of automated UI tests. These tests can be executed via the command line, via the NUnit test runner, or within Visual Studio using R# or TestDriven.NET. 

We have invested heavily in a library of "data helpers" that assist with setting up test data, ensuring that it remains fairly easy to write new automated tests even as the product complexity continues to grow.

Source Control / CI

We use Git and Bitbucket.

Our general pattern is that most tasks are performed in feature branches and then merged into trunk once they are stable and are code-reviewed. We maintain release branches so that critical changes can be merged across multiple lines of development if needed.

The entire test suite is automatically executed for each check-in to trunk, and we will often set up CI builds to monitor long-running feature or release branches as well.

Deployments

The process of preparing a release package is fully automated using a Ruby tool called Rake. Rake scripts will do a fresh checkout, compile the release package, run the tests, commit the package to source control, etc.

Deployments themselves are handled through Octopus Deploy and are approximately 75% automated. Our IT/Devops team manages the deployment process.

Technical Principles

In addition to the core corporate values (outlined below), the technical team believes in the following:

Core Values

Heuristic Solutions has the following core values:

These aren't just words on a motivational poster in the corner; they are things that we truly believe in, and you'll hear them frequently as we do our work.