Adaptoring: Adapter Generation to Provide an Alternative API for a Library
Lars Reimann, Günter Kniesel-Wünsche
TL;DR
Adaptoring introduces a wrapper-based approach to provide an alternative API for existing libraries without modifying the original code, preserving compatibility while improving learnability and usability. It formalizes adaptorable API transformations, automates adapter generation for Python, and combines usage data with docstring-driven inferences to suggest deletions, parameter optionality, and preconditions. A GUI-assisted editing workflow enables human review and extension of automated suggestions, and an evolution framework addresses maintaining adapters as the original library evolves. Empirical results on scikit-learn and other libraries demonstrate meaningful API size reductions, competitive inference precision, and efficient end-to-end adapter generation times, indicating practical viability for large-scale libraries.
Abstract
Third-party libraries are a cornerstone of fast application development. To enable efficient use, libraries must provide a well-designed API. An obscure API instead slows down the learning process and can lead to erroneous use. The usual approach to improve the API of a library is to edit its code directly, either keeping the old API but deprecating it (temporarily increasing the API size) or dropping it (introducing breaking changes). If maintainers are unwilling to make such changes, others need to create a hard fork, which they can refactor. But then it is difficult to incorporate changes to the original library, such as bug fixes or performance improvements. In this paper, we instead explore the use of the adapter pattern to provide a new API as a new library that calls the original library internally. This allows the new library to leverage all implementation changes to the original library, at no additional cost. We call this approach adaptoring. To make the approach practical, we identify API transformations for which adapter code can be generated automatically, and investigate which transformations can be inferred automatically, based on the documentation and usage patterns of the original library. For cases where automated inference is not possible, we present a tool that lets developers manually specify API transformations. Finally, we consider the issue of migrating the generated adapters if the original library introduces breaking changes. We implemented our approach for Python, demonstrating its effectiveness to quickly provide an alternative API even for large libraries.
