If you are planning to build API and make them available as micro services and your code assets are C#/.NET Framework/Windows then you have some choices :
ASP.NET Core comes with an internal web server called Kestrel : kestrel claims to be a light and fast web framework which runs ASP.NET Core applications without the need for IIS in front (you might decide that IIS is usefull for other things). Kestrel allows for middleware modules (like IIS ISAPI filters) that allow for adding features to the basic server.
Katana (OWIN Microsoft implementation) is a collection of NuGET packages for building OWIN applications.
Plain old IIS + ASP.NET REST API code
Some good information can be found here from uShip guys
I’m basically writing this for myself as a reminder for the new .NET Core Microsoft environment.
Good Coders don’t want to break their code because their code is like a creature and nobody wants to harm a creature. Take a note, this is how it is. Good Coders tests are just ‘combing’ the code they should test. But this creature has bugs, no matter how good the Coder is, so good Coders are bad at writing tests on their own code.
A good test tries to tear code apart, to abuse it doing the nastiest things, at the maximum possible speed, in the worst possible way : this is why you should consider making another set of tests by another coder which has not written the code itself.
But with TDD this is difficult : I like the idea that a Coder writes it’s software use cases before writing the software itself but I’m missing how to include in this process a second head to develop some really nasty tests on that piece of code and I’m not a great fan of pair programming (for many reasons).
So open issue for me here : how to include development by a second head of integration tests.
(In the picture : good coders and good testers 🙂 )
Over engineering code is a common plague that leads to a number of unwanted side effects in your software ranch. Consider it just a form of bad software design.
This post does not want to be an exhaustive guide on over engineering but with the help of many quotes from different sources, I’ll try to summarize the impacts, when you can have evidence of over engineered code and the causes (to try to avoid them).
Impacts
unneccessary complexity => less agility in software
more code, more bugs
more code, more unit tests, more development time
performance impacts
longer catch up time for newcomers
difficult maintenance, fixing bugs requires more time than it should
small changes require big efforts and are prone to having many bugs
unwanted features are present even though there is not reference in requirements/backlogs and so you have to maintain them even if they are not used
false perception of complexity in the code that leads to over inflated estimates for fixes, changes.
extending an over engineered project makes extensions more complex than necessary
dismantling of over engineered code takes way longer than it took to over engineer it
Need for complete rewrite over time to reduce unneeded complexity when maintenance times become higher than rewrite time This is an interesting topic itself and needs a separate post)
Concrete symptons of over engineered code
Really difficult here to point out some easy and simple guidelines to recognize over engineered code. Often it is an individual perception and this does not help. I’ll try :
When a well experienced coder, which has been working on that piece of code for at least an year, still takes way more than necessary/affordable to figure out where/how a specific feature works. I’m trying to find a better explanation for this concept
In code : using a factory even if it is making type of objects
In code : use an interface even if it is actually going to be implemented by just one class
“This is really going to be a clean framework. I’ll make an abstract class out of this part so that folks can subclass it later, and I’ll put in a bunch of well-commented overridable hooks in the concrete subclasses so that folks can use them as templates, and just in case somebody ever needs to build special debug subclasses, I’ll put in extra stubs over there (somebody will thank me for ’em one of these days). Yeah, this is really going to be neat.Thus is bloated software produced when our artistic sense gets the better of us. — DaveSmith “
Some other signs of over engineering taken from stackoverflow :
One very strong warning sign of over engineering is when everything goes through so much indirection that it’s hard to find the piece of code that actually implements some concrete, domain-level piece of functionality. If you find that most of your functions do very little concrete work and just call other virtual functions, you may have a problem.
[..] Make factories where the factories make more then one type of object. Use dependency injection, where it immediately shows benefits. Make interfaces that are actually going to be implemented by more then one class [..] What I see too often in “true OO” is that advanced techniques are used to solve really simple problems in an overly complex way [..]
Software should behave predictably and accomplish its goals without too many surprises (that is, outages in production). The number of surprises directly correlates with the amount of unnecessary complexity found in a project. It’s therefore crucial to think about accidental complexity and essential complexity:
Accidental complexity relates to problems which engineers create and can fix, [whereas] essential complexity is caused by the problem to be solved, and nothing can remove it — Fred Brooks in his seminal “No Silver Bullet” essay
Another interesting point on when a coder start to write unnecessary code :
Boredom is good precursor to over-engineered code. I’ll admit, when I got my first job, I felt so underutilized. I was just bored. And when I got bored, I wrote code. Not just any code — CATHEDRALS OF CODE.
No seriously, I had a mental picture of my code and abstractions as large towers with golden jutting spires, flying buttresses of glassy onyx, a wonderful vault supporting by arched domes topped with beautiful geometrical tracery, etc etc etc.
It was really fascinating to see the patterns working together for myself, but in retrospect, I am completely ashamed of the ungodly mess I left behind.
If you’re writing your own frameworks and DSLs code to while away the less stimulating hours at work, just stop. Time is better spent reading Wards Wiki, or writing an open source book, or you may just want to ask management for more work.
Some theory on why/when over engineering happens
Second system effect : you wrote a first good, simple, performant version of a product and it is getting traction so you decide to write 2.0 version of it which does everything.
feature creep : you just keep adding features that are used by less than 1 % of the user base/use cases.
inner-platform effect : you are making your system so customizable to become a replica of the features provided by the sw component of dev environment you are using.
I’m sure you found yourself in the situation of having to fix a piece of code that :
was not written by you
has not a clear set of requirements or the requirements are clear but the code is unnecessarily complex (at a first glance looks like you encountered the dreaded “big ball of mud”)
Nobody knows exactly the requirements anymore so you’ll have to reverse engineer from the code the behavior
Tests are missing or trivial and don’t help in understanding what that piece of code should do
My first question here is to understand where is the break even between fix and rewrite. I tend to prefer rewriting code if :
besides needing fixes there are also performance problems
code needs to be fixed/changed often
code is a company asset
code complexity is exposed to customers
coders turnover is high
Remember that :
Debugging is twice as hard as writing the code in the first place. Therefore, if you write the code as cleverly as possible, you are, by definition, not smart enough to debug it. — Kernighan and Plauger in “The Elements of Programming Style”
Let’s try to add some information. Pros and Cons for rewriting instead of fixing :
Cons :
Time/cost : rewriting takes time, probably more than a “quick and dirty fix”. The real cost analysis here is “how much faster would we make fixes if we had a clean code base?”. How many fixes we perform every year ?
Pros :
Coders will not have to work on a “big ball of mud” : this causes frustration and in the end lowers productivity (Time/Cost–)
New code base will probably be smaller (requirements sediments will get cleaned up), so less tests need to be maintained (Time/Cost –)
New code base will probably be faster (Cost–)
Newcomers will have short catch up time on the new codebase ( old codebase Time/Cost –)
I recently had a chance to try out SmartOS after my last experience in application porting on it (2012) and I was impressed by the virtualization features that are made available directly by the os. One set of commands allows downloading images, running either vm like container or namespace like containers. Zones also allow you to run debian/centos applications on SmartOS inside a lx zone with system call translation … Awesome.
If your interested in digging deeper take a look at this post from Tim Boudreau.