Hey there! Do you want to build the next generation of edge services? Do you want to work on cutting-edge technology with a great team full of talented and motivated individuals? StackPath is hiring backend developers! StackPath's backend powers not just our customer portal and API but is the core behind many of our amazing products.
We love golang at StackPath. Most of our services are written in go, and we are always looking to add bright and awesome people to our ranks. If you think this sounds great and if you have what it takes then apply for one of our backend service positions. If being a backend engineer isn't your thing then we have many open positions to go for.
We employ a lot of modern technology and processes at StackPath. These three exercises are intended to demonstrate your basic knowledge of go and its applications. These problems have many solutions. Our managers are most interested in how you choose to solve them and why. There are no wrong answers, as long as these examples compile and work in at least go 1.14 and use go modules for package management.
We're pretty serious about testing at StackPath. In addition to QA, all of our code includes unit tests that are run in builds. Builds must pass before code can go live.
The unit-testing
folder contains a FizzBuzz
example implemented in go. It takes three optional command line arguments:
- The number of numbers to iterate over
- The multiple to "Fizz" on
- The multiple to "Buzz" on
For instance:
$ go run main.go 16 3 5
SP// Backend Developer Test - FizzBuzz
FizzBuzzing 16 number(s), fizzing at 3 and buzzing at 5:
1
2
Fizz
4
Buzz
Fizz
7
8
Fizz
Buzz
11
Fizz
13
14
FizzBuzz
16
Done
It works well enough, but doesn't have any unit tests. Without tests, we can't prove to everyone that the code works and that it doesn't affect other services.
Look in the unit-testing/pkg/fizzbuzz/fizzbuzz_test.go
file and implement unit
tests that flex the FizzBuzz()
function in unit-testing/pkg/fizzbuzz.go
. Think
of the different kinds of inputs that can be passed to the function and how it
should act when common and edge cases are used against it. Did your tests find
any bugs in FizzBuzz()
? If so then fix 'em up! Can you get greater than 80%
code coverage with your tests?
Run go test -v -cover ./...
from the unit-testing
directory and let us know
how you did.
Web services are our bread and butter. Our services talk to each other over
gRPC and REST.
The rest-service
directory contains a simple Person
model and a set of
sample data that needs a REST service in front of it. This service should:
- Respond with JSON output
- Respond to
GET /people
with a 200 OK response containing all people in the system - Respond to
GET /people/:id
with a 200 OK response containing the requested person or a 404 Not Found response if the:id
doesn't exist - Respond to
GET /people?first_name=:first_name&last_name=:last_name
with a 200 OK response containing the people with that first and last name or an empty array if no people were found - Respond to
GET /people?phone_number=:phone_number
with a 200 OK response containing the people with that phone number or an empty array if no people were found
You can implement the service with go's built-in routines or import a framework
or router if you like. The Person
model and all of the backend code is in the
rest-service/pkg/models/person.go
file, the service should be initialized in
rest-service/main.go
, and should run by running go run main.go
from the
rest-service
directory.
Implementing the service is a good start, but are there any extras you can throw in? How would you test this service? How would you audit it? How would an ops person audit it?
StackPath operates a gigantic worldwide network to power our edge services. These nodes are in constant communication with each other and various central systems. Our services have to be robust enough to handle this communication at scale.
The third programming test in the input-processing
directory contains a
program that reads STDIN and should output every line that contains the word "error"
to STDOUT. We've taken care of most of the boilerplate, but the rest is up to you.
Consider scale when implementing this. How well will this work if a line is
hundreds of megabytes long? What if 10 gigabytes of information is passed to it?
What if entries are streamed to it? How would you differentiate between errors read
from the stream vs program errors? How would you test this? Assume that \n
ends a
line of input. Was with the REST service test you're free to use any built-ins or
import any frameworks you like to do this.
In some advanced situations, StackPath relies on Go's concurrency primitives to
perform asynchronous tasks. The concurrency
directory contains two interfaces for
asynchronous task pools, SimplePool
and AdvancedPool
, whose implementation
requirements are documented on the interface. Please write an implementation for
NewSimplePool
. As an extra bonus challenge, but by no means required or necessary,
feel free to also write an implementation for NewAdvancedPool
.
While third party libraries can assist, many times our unique needs rely on channels directly, so please refrain from using anything but the Go standard library for this exercise. Unit tests are not necessary, but feel free to write them if it helps during the development.
What did you think of these? Are these too easy? Too hard? We're open to change and accept issues and pull requests for these tests. See our contributing guide for more information. Thanks for giving these a try, and we hope to hear from you soon!