This is part three of the weblog series on the New Enhancement Framework. You get to know some more facts as to what a BADI is and when to use it, and then, at last, we start with an example that shows you how to build the BADI with the respective tools in the ABAP Workbench and how the code to instantiate and call the BADI is integrated in the ABAP language.
In the last weblogs you have learnt about the basic structure of an enhancement, the different enhancement technologies within the framework and the containers for enhancement options and enhancement implementation elements. You need to know them in order not to lose your bearings within the Enhancement Framework when we now start building a BADI. If you have a mental map of the entities in the framework, you will find it really too easy to understand what is going on in the example. So I would recommend you should take your time to recapitulate theses basics, before proceeding to the practical part of this weblog.
Some Basics about the BADI
Many of you know perhaps the classic BADI. And there is some good news for them: The basic things about the BADI itself have not changed a lot, though there are some major improvements, such as a gigantic step forward in performance. What will be totally new for those familiar with the classic BADI is the way the new BADI is embedded in the container-structure of the Enhancement Framework.
For those not acquainted with the classic BADI, let me spend some words on what a BADI is and what it is for:
The BADI is an object-oriented enhancement option. The BADI defines an interface that can be implemented by BADI-implementations that are transport objects of their own. The new BADI is fully integrated into the Enhancement Framework. Within the Enhancement Framework a BADI is an enhancement option or an anchor point for an object plug-in.
A BADI is to put it this way a controlled explicit enhancement option. It is an explicit enhancement option, which means: BADIs are not provided by the framework, but they have to be defined explicitly by the developer. Why do I call it a controlled enhancement option. To understand what this means we have to go a bit deeper into the methodology of enhancing development objects. Why is it done, and who does it? What is the typical process of enhancing a development object like?
Providing and Implementing Enhancement Options - Two Different Roles
In this process there are two different roles: There is the so-called option-provider. He is the one who builds an enhancement option that is a hook where others can attach something to. And it is essential to have such a hook: Where there is no hook you cannot attach anything. This analogy means that you can only insert an enhancement implementation where there is an enhancement option. Of course, there are implicit enhancement options, that are, so to speak, for free and always available. But there are not implicit BADIs. If you need a BADI, somebody has to define it explicitly. Let us call this role the (enhancement) option provider. In general, the option provider is identical with the developer of a development object that should offer an explicit enhancement option. Let us call this object the basic development object. Typically, the developer of the basic development object could be somebody working in the SAP core who anticipates that the customer or an industry solution might want to add some code at a particular point. In this case he creates both the development object to be enhanced at a later stage and defines an enhancement option.
An enhancement option that has no implementation does not do a lot when processed; in fact, it has no effect at all. To make something happen at an enhancement option, you need to implement it. The developer who implements an enhancement option, or to put it in the analogy, attaches something to the hook, is called the implementer. He need not care about the details of an enhancement option, but just know how he implements it. But, of course, he should have a sound knowledge of the way enhancement options function. You might compare this to somebody who uses a service class. He does not need to know the inside of this class, but just its public methods or maybe even only the respective interface he is interested in. In the same way, the implementer only needs a limited knowledge as to how to define an enhancement option.
In the strict sense, many of you are probably only interested in implementing a BADI. They suppose that SAP provides the BADI and calls the respective BADI methods. All they want to know is what they have to take heed of when writing the BADI implementation. But it is quite useful to know a bit more about defining a BADI, and once you have understood what a powerful pattern a BADI is, you might also be tempted to write your own BADI definitions. Moreover, it would be a bit long-winded to write two different series of weblogs, one for the option provider and another one for the implementer, because there is a large overlap in the background knowledge. For all these reasons, I have decided to handle both topics in this one series. But still, I treat building the BADI in this weblog and providing an BADI implementation in the next one. And you should always keep in mind: Implementing and defining a BADI are two different roles. And the same is true for source code enhancements.
The BADI as a Controlled Enhancement Option
To fully grasp in which way a BADI is a controlled enhancement option, let us first have look at source code plug-ins in contrast. This enhancement technology is not controlled at all. This means: The option provider inserts an enhancement point in the code thereby giving the possibility to attach a source code plug-in. The implementer can insert any code whatsoever at this enhancement point. The option provider has no control at all on the code that is written, because generally it is implemented at a later stage of development. Moreover the code of the source code plug-in has access to all the variables that are visible within the modularization unit that is enhanced. So the option provider cannot prevent the implementer from changing the value of some variables that should not be touched at all. In fact he cannot bar the implementer from anything at all.
With BADIs, things are completely different. The option provider has a firm grip on what the implementer does insofar as he defines the interface and confines the implementer to the class that implements the BADI. In detail this means: A BADI defines the methods with their signature, that is, which parameters the respective method imports, exports, changes or returns. The BADI implementer has to implement these methods defined in the BADI interface. We could put it this way: With a BADI the option provider defines what has to be done, but leaves it to the implementer how he implements the BADI. Moreover, the BADI implementer cannot fiddle with the basic program. It is assured that a BADI implementation can only change the parameters handed over to BADI. The implementation of a method has a context of its own and the variables of the development object using the BADI are not even visible from inside the BADI implementation.
BADI Commands in ABAP
There are two ABAP commands for the new BADI. GET BADI hugo returns a handle to all active instances of the implementations of the respective BADI. For example hugo might refer to five active BADI implementations.
You call a BADI method with the command: CALL BADI hugo->method. If a method of the BADI is called, all active implementations are selected. So this amounts to a loop with the different method calls in it.
When to Use a BADI?
As a potential option provider you should use a BADI if you already know that a particular procedure should be executed at a particular position in your program, but you want leave it to somebody else to specify the specific content of that procedure. This is typically the case if the details of the procedure vary among different potential implementers or are so specific that the developer of basic development object does not know them or want to give the possibility to add different implementations at a later stage.
So the developer of the basic development object just writes something like saying: subtract your fees and return the result. He does not care about what the different fees are and if there is a complex formula, percentages calculation or just a simple subtraction. To do this the option provider writes a BADI which contains a method:
subtract_fees CHANGING costs.
This way, a BADI can be used quite well to understand what enhancements are for from a semantic point of view. This is because the way BADIs work is pretty much like our common sense concept of enhancing something. Usually we do not think: Just add any enhancement, but add something as a means to a particular end. You can decide yourself how you want to accomplish the end, but the end is defined. This is one analogy to explain a BADI.
You might compare the usage of BADIs also to the process when some company delivers the same car body to different manufactures of multi-purpose vehicles. They all build their specific vehicle on the same car body. The producer of the car body leaves the technical details to the different manufacturer, but of course it prescribes that the car suspension has to be build in here and the engine there. In this way, the engine has to be in a particular place and has to have particular properties, but all the other details are up to the manufacturer of the multipurpose vehicle.
This process can be compared to the process of multilevel software development I have already sketched: The program logic of a SAP core program needs some value at a certain point, but the developer of this program does not know how to calculate this value. So he inserts a method call that should provide this value, but leaves the actual implementation, that is the calculation to the developer of the next development level. In terms of this framework, the developer inserts an enhancement option that is a BADI, providing the respective method in its interface, and also a call of a BADI method.
Multiple and Single Use BADIs
Concerning the programming model, there are two different BADI types: Single- and multiple use BADIs. They have a completely different semantics.
The use case with the calculation I have sketched above is a typical example of a single use BADI. The basic program needs the result of a calculation. You leave the implementation details of the calculation to subsequent levels of development, for example the developers for the particular countries. The business logic requires a result of the calculation, and it must be exactly one result. The next steps in the program need this result, so you need at least one BADI implementation. And there is no way to process many return values, so you need exactly one result. If you have a single use BADI, the system takes care that there is exactly one active implementation. This way the call of the method of a single use BADI functions like a method call.
Multiple use BADIs are quite different in this respect. Such a BADI is suitable for activities, that might be done or not and that might even be consecutively done in different ways in different implementations when a program is executed. Imagine you build a BADI additional_output that might convert data for different additional output devices such as a Blackberry, handy, etc… Many different active implementations can coexist in the program simultaneously, but there might also be no active implementation. So the call of a method of a multiple use BADI is in a way like sending a message: You do not know and do not care how many services react to the message. It corresponds to a publish and subscribe mechanism.
Building Our First BADI
The Use Case
And now let us have a look at an example in detail. Imagine that the developer of the basic program needs the VAT of different book entries. These entries should be passed to a method that calculates and returns the VAT. As this developer does not know what the VAT in a specific country is, he defines a BADI BADI_vat with the method get_vat. This method should return the VAT for a particular value that is passed to the method get_vat.
Building the Enhancement Spot
The first thing you need when creating a BADI is a container for the BADI. Up to now we have no container, so we have to create a (simple) enhancement spot. This is the container in which we develop our BADI.
We go to the ABAP workbench and navigate there to our local objects. We select:
Create->Enhancement->Enhancement-Spot
from the context menu of the local objects folder.
This is what the dialog window looks like:
We enter a name for the enhancement spot and some short text. To name a composite enhancement spot is voluntary. We are content with our enhancement spot and need no higher level container for the example. In real life programming, you probably should take advantage of the structure the composite enhancement spots offer you and always work with these complex containers.
The BADI Itself
Next we create a BADI within the new enhancement spot. We select the CREATE icon on the left:
In the dialog window that appears we enter the BADI name z_BADI_calc_vat plus a short description and confirm. Now we have a BADI in the list of our enhancement spot. We deselect the property "multiple use", because for our calculation we need a single use BADI:
The BADI Interface
More important is the fact that up now we do not have much of BADI. We need an interface. In fact, it is the interface where you define the methods that determine what use you can make of your BADI. Understood this way, the interface makes up the identity of a BADI to a large extent. After clicking the arrow in front of the BADI name and after doubleclicking Interface in the tree below we can input the name of the interface. You can either choose the name of an interface that already exists or create a new one. There are no restrictions on the name of the interface enforced by the framework. But, of course, you should conform to the normal rules for choosing speaking names in a systematic manner.
Selecting the Change icon leads you to the class builder, where you can create the methods you need for your BADI in the same way that you are used to if you are familiar with the class builder. We just type in the name of the method get_vat and enter the parameters we need.
A BADI interface has to implement the interface if_BADI_interface. But if we create a BADI interface in the way just shown this marker interface is already integrated in the BADI interface. Probably you know how to work with the class builder, but still I have inserted the following two screenshots to show you: Defining a BADI interface is just normal ABAP Objects programming and as simple as could be for anyone familiar with the Class Builder. We define one BADI method.
Next we determine the parameters of the method.
We save and activate the interface and the spot.
So let us now take a breath and consider what we have created so far:
We have created an enhancement spot and a BADI with an interface. The interface has one method so far.
Of course, just building a BADI does not suffice. It does not do anything. You need a BADI instance, and this instance must be called somewhere in the code. This is the definition part of a BADI. But as a BADI only defines an interface, you need a class that implements this interface. Remember: A BADI definition is the option to insert an object plug-in that does a job at runtime. You still need an object that is plugged in to get something done.
BADI Commands in ABAP
So it is time to write some ABAP code to use the BADI. We need a variable that can refer to the BADI and some variables which are given as actual parameters to the BADI method. Next we create a handle for that BADI and call the BADI method get_vat. The respective commands are GET BADI and CALL BADI.
If we run the program it dumps. Considering what I have told you before about the single use BADI, this is no wonder. It is mandatory, that there is exactly one active implementation for a single use BADI. Our BADI is a single use BADI. One way to handle this is to catch the respective exception cx_BADI_not_implemented.
Using and Creating a Fallback Class
The other way is the better solution, namely to use a fallback class. A fallback class for a BADI is used if there is no active BADI implementation. This means: The GET BADI command returns a handle to an instance of the fallback class and the respective CALL BADI calls the methods of the fallback class instance. As soon as there is an active BADI implementation, the fallback class is not used any longer at runtime. So using a fallback class serves two functions at once:
- The program runs with a single use BADI without raising an exception.
- It is guaranteed that the fallback class is not used any more as soon as a BADI implementation is supplied. So a fallback class is only selected conditionally. This is important, because the BADI provider does usually not know the details of some process. This is why a BADI is used.
We select the checkbox the bottom with the text: Call fallback class if no implementation is executed and insert a name for the class.
Clicking the change icon leads us to the class builder. Again the tools do a great job for you: The respective method of the BADI interface is already defined. We only have to implement it:
I skip this, because it is no news to anybody familiar with the Class Builder. Of course, we have to save and activate the class. Then we navigate back to the enhancement spot and activate it.
Running the program again leads to result:
percentage: 20 VAT: 10.
And this is the end of this weblog. Now, you know how to build an enhancement spot and a BADI including the BADI interface and a fallback class plus the ABAP commands to use the new BADI. In the next weblog you learn how to create a BADI implementation inside the respective container, that is the (simple) enhancement implementation and how to select among the different BADI implementations using the addition FILTER to the GET BADI command
No comments:
Post a Comment