Apple Pay is nearly perfect. They managed to make payment both simpler and more secure than using a credit card. But Apple left out one critical ingredient:
Most stores have these. Consumers trade some privacy (allowing the store to profile their purchases) in exchange for lower prices. While Google and others may support scanning of loyalty cards, Apple can do it better: support loyalty cards by eliminating them.
When you use Apple Pay at a merchant, the payment screen (as the confirmation appears) could include an option to enroll in the merchant’s loyalty program. If the user opts in, Apple would create a random key for the merchant and share that same key with the merchant each time you make a purchase at their store. The loyalty key would be unique to each consumer/merchant pairing, so different merchants would get a different key for you, but a single merchant could easily track your purchasing behaviors (even if they don’t know who you are). In exchange for opting in, you would enjoy all the merchant’s loyalty benefits.
This would be a win-win-win scenario:
Apple could even facilitate a process where merchants optionally offer greater savings for more information (e.g., within a list of your merchants in the Apple Pay app, you could not only manage your loyalty enrollments, but also share your gender for 5% more discount or share full demographics and get 10% more of the marked price).
Update 15-June-2015: Apple announced that iOS 9 will support loyalty cards. So far, it looks like loyalty cards that are credit/debit cards will be supported, but what about the myriad of loyalty cards that are used in addition to paying (i.e., swipe your loyalty card to get the discounts, then use your credit card to pay). Will Apple support these and, if so, will they be automatically passed in a single payment transaction? Or will they be handled separately requiring two transactions? Currently, many merchants have you swipe your loyalty card up front (before payment) in order to include discounts in the total price. A better approach (for both consumers & merchants) would be to avoid the loyalty card step and present the consumer with two totals: a total price and then the total for loyalty customers. Apply Pay could transmit the loyalty number along with payment to get the discounts and consumers could get their discounts with a single transaction. By associating the loyalty card number (or Apple auto-generating one), consumers would no longer have to carry the extra loyalty card or even deal without it within Apple Wallet after enrollment.
Update 10-January-2016: In November 2015, Walgreens announced it would be the first to incorporate loyalty cards in Apple Pay. Unfortunately, the loyalty card is entered into the phone and used separately, meaning that the user is left to do the work that the phone (i.e., Apple) could be doing. Scanning the loyalty card into Apple Wallet is be a very simple & intuitive way to consent to the loyalty program; that’s perfectly fine. However, Apple should recognize when you use Apple Pay at a merchant for which you’ve registered a loyalty card and automatically send the loyalty card number within the payment process. Instead, the user is forced to make two transactions: first selecting the loyalty card and then selecting the card for payment.
I’ve seen countless presentations. The best ones use PowerPoint with lots of words. The more words, the better. Recently, the number of people trying to make only a few takeaway points during a talk or distracting us with clever images has increased. I thought it was time to offer a concise example of best practices. Fortunately, this can be done in a single PowerPoint slide. Here it is:
In the OpenMRS community, we often hear reference to “core devs.” Who are these people? What make them “core devs” anyway?
OpenMRS is used all over the world with more than 115,000 downloads across more than 200 countries. As of February 2015, there are 1263 subscriptions to the OpenMRS Developers Mailing List. The initial release of OpenMRS 1.9 was thanks to substantive contributions from more than 70 devs. GitHub shows 930 forks and over 140 contributors to OpenMRS Core for openmrs-core. As of the OpenMRS Implementers Meeting in Maputo (#MOZ15), we introduced Developer Stages, both to recognize & empower developers based on their level of expertise and community engagement and to adopt a more scalable approach. If you listen to Yehuda Katz’s great discussion Indie OSS, it’s not healthy to make a distinction between “the core team” and community. So, why are we referring to a few developers as “core devs” and labeling the 99.9% as non-core?
As a community, we need to evolve beyond using distinctions like “core” vs. “non-core” developers toward a more scalable approach: Developer Stages. So, if you see me (or anyone in the OpenMRS community who agrees with me), calling someone out in the future for using the term “core devs,” please understand it’s not for a lack of respect & appreciation for the awesome contributions from those who have been attributed as such; rather, it’s out of an appreciation for all the awesome developers around the world in past & future, including the “core devs,” who have contributed and will contribute to saving lives through coding for OpenMRS.
From now on when you feel the urge to say “core devs,” try substituting it with “/dev/5’s” or “available /dev/4’s & /dev/5’s.” As a community, we will be working to make this attribution easier to see and understand, so, in the future, when someone refers to the “available /dev/5’s”, they’ll be referring to far more than a three people. 🙂
During the OpenMRS Leadership Camp 2014, we talked about ways to empower and scale the OpenMRS Developer Community. While our approach to collaborative development has gotten us far, we don’t have a clear process for developers to grow in responsibilities. The fact that OpenMRS has approximately the same number of developers doing code review or pushing to core as it had five years ago is a significant failure on our part. We’ve been talking about ways to be more inclusive for a while, but haven’t put these desires into something actionable… until now.
With the help from several folks in the community, I was able to come up with a draft process for recognizing the stages of a developer within the OpenMRS community:
Stage | Description |
---|---|
/dev/null
“Community”
|
Criteria
Expectations
Privileges
|
/dev/1
“Learning”
|
Criteria
Expectations
Privileges
|
/dev/2
“Contributing”
|
Criteria
Expectations
Privileges
|
/dev/3
“Cooperating”
|
Criteria
Expectations
Privileges
|
/dev/4
“Collaborating”
|
Criteria
Expectations
Privileges
|
/dev/5
“Leading”
|
Criteria
Expectations
Privileges
|
The goals of defining a process like this are:
Our goal would be to fully automate the transition from /dev/null to /dev/1 – i.e., anyone in the community should be able to transition to /dev/1 without requiring manual review from anyone else in the community. Realistically, transitioning through later stages of development would require some manual review, but our hope would be to keep things as objective as possible, so any developer would know what she needed to do in order to advance to the next stage of development.
Next steps:
While I could spend more time revising this blog post, it’s primary purpose is to share the work in progress, so I’m going to stop editing and let it go warts ‘n’ all. 🙂
You can see the actual OpenMRS Developer Stages wiki page here.
I recently sent an email to the OpenMRS Implements Mailing List describing the evolution of OpenMRS between 2012 and 2015. While writing the email, I thought it might be easier to describe in an image:
Our goal is to get to a UI-less platform (providing API & web services) used by an OpenMRS with a new and agile UI framework that meets or exceeds the needs of existing implementations. Currently, we have several implementations working in the 2.x UI, but the majority of implementations are still using the the 1.x UI. To achieve our goal, we will not only need to reach OpenMRS 2.3 with comparable or greater functionality than OpenMRS 1.9, but also find a way to ease the burden for implementations to migrate from 1.x to 2.x (e.g., migration tools, converting key modules to run in 2.x, possibly find a way to run most 1.x modules within the 2.x framework, etc.).
There are many ways to capture feedback from users & testers, from feedback buttons built into the app to issue trackers and tools like JIRA Capture. Another method we have been using, especially for upcoming releases or new features or widgets, within the OpenMRS community is a side-by-side feedback page.
The application is on the left and an etherpad on the right. While I am not suggesting this as an approach for issue tracking, but we have found it to be a quick & easy way of collecting community feedback. The combination of a link taking them directly to the product to be tested along with the near-zero activation energy required by etherpad makes it a handy combination. It’s also nice to be able to throw a brief intro into the etherpad to direct people on what to test and how to report feedback. And lastly, there’s a nice side effect of people seeing each other’s activity in real time. When combined with a developer responding to feedback and re-deploying fixes in real time, it can be incredibly powerful.
Anyway, the main reason I decided to blog on this is because I tweaked our side-by-side tool a bit and wanted to throw my one-page feedback HTML in here for the next time I need it. Here it is:
Feedback
Ever since the beginnings of OpenMRS, we’ve used the data model as a reference and as a teaching tool. As the number of tables has grown, it has become harder to keep the data model diagram updated. I also wanted an easy way to search for tables, columns, or foreign keys. So, I created dbtohtml to generate an easily browsable, standalone, singe-page HTML view of the data model.
I recently described the @should taglet created by OpenMRS that helped the community adopt and sustain better testing practices. Mário asked a good question about test-driven development (TDD):
While I don’t think we’re doing much TDD in the OpenMRS Community at this point, it would be great to evolve this direction. The real question is: will the @should tags that helped us start testing our code become an impediment to TDD? I don’t think so.
Let’s try a simple example to see how we could be TDD-ish with @should tags. Imagine that we want to be able to get the age in years of a person:
class Person {
Integer getAge(Date onDate) {
return 0; // TODO: return age
}
}
Before we write any code, we describe the expected behavior. To keep the example brief, I’ll just describe a couple expected behaviors:
class Person {
/**
* Returns person's age in years.
* @should return null for date before birthdate
* @should not round up age
*/
Integer getAge(Date onDate) {
return 0; // TODO: return age
}
}
Next, we invoke the Behavior Test Generator plugin to automatically do the busy work of generating the skeleton for our unit tests.
class PersonTest {
void getAge_shouldReturnNullForDateBeforeBirthdate() {
// TODO: write unit test
}
void getAge_shouldNotRoundUpAge() {
// TODO: write unit test
}
}
So, now we can write our unit tests and see them fail, like any newborn tests in TDD would do. Granted, in this example, you don’t technically start with the test code, but you can start with describing behavior (using @should tags) prior to writing code and using those tests to drive development. So, yes, we start with @should tags; however, @should tags, can precede any actual code, since they are effectively shorthand for the tests we are writing before coding.
In 2008, when OpenMRS was struggling to adopt better test-driven development practices, I was lucky enough to read Dan North’s Introducing BDD. As Dan says:
How true! For example, it’s common to see something like this when you start creating unit tests:
public class PatientTest {
public void testPatient() {
// test stuff here
}
}
The next question is, what gets tested in a method called “testPatient”? I suppose the only wrong answer is “nothing.” But the problem is there are an infinite number of right answers… because “testPatient” doesn’t say anything about the behavior. As Dan points out, simply replacing the word “test” with the word “should” is a game changer. Let’s try again, except this time we will use “should” in our method name:
public class PatientTest {
public void addIdentifier_shouldNotAddIdentifierThatIsInListAlready() {
// make sure an identifier isn't duplicated
}
}
It’s much easier to guess what will be tested inside that unit test’s method. That’s good… but it gets better. Dan’s suggestion of “should” not only places the focus on behavior, it also automagically forces testing to be scoped to a specific behavior, since any developer who sees a method name wrapping onto its third line instantly knows she is going about testing the wrong way and will look for help. Dan gives a great justification for this approach… but he had me at should.
Given Dan’s insight into using “should” instead of “test” to drive BDD, the trick was figuring out how we could engrain this approach within the OpenMRS community. After some discussion, we came up with an idea that I’m still proud of today and I believe has helped us adopt a better testing culture. Here’s what we did…
Testing is often filled with cookie-cutter code and requires additional effort that is difficult to sustain. We wanted to find a way to overcome both of these challenges. What we needed was a trivially easy way to generate behavior-focused tests. So, we invented the @should Javadoc tag to allow developers to describe expected behaviors within the Javadoc and then we paid someone to develop an IDE plugin to auto-generate the test methods from existing method names.
Now that we have the @should tag, let’s take on more stab at testing. Imagine you are writing some code for the Patient object…
public class Patient {
public void addIdentifier(PatientIdentifier patientIdentifier) {
// ...
}
}
You know that an identifier shouldn’t be added twice for the same patient, so you simply state that behavior in the Javadoc:
public class Patient {
/**
* @should not add identifier that is in list already
*/
public void addIdentifier(PatientIdentifier patientIdentifier) {
// ...
}
}
That’s it. You’re already doing BDD! Now, you tell your IDE to generate any missing unit tests for Patient and it automatically generates this method stub for you in the appropriate location:
public class PatientTest {
public void addIdentifier_shouldNotAddIdentifierThatIsInListAlready) {
// write your test here
}
}
The IDE plugin automatically derives the proper location and method name from your @should tag and the associated method. Now you can focus on testing that specific behavior without having to worry about any cookie-cutter code and adopting BDD is as simple as writing a Javadoc comment.
We still have a long way to go down the road to full BDD, but I was very happy with our first step. Over the years, the @should tag has become a handy tool for establishing a behavior-driven culture of testing; in fact, it has helped us adopt testing in general. For any Java-shop that is wondering “How do we get our developers to start testing their code?”, I would strongly encourage you to read Dan North’s writings and consider adopting the @should Javadoc tag.
Very cool… of course, it’s all Chinese too me. Here’s Google’s translation back to English:
Thanks to Yang & team, Harsha, and all who contributed!