Interpreting API as a full-fledged dependency
How not to hate the API you're working with
Intro
For a long time, I couldn’t finalize the draft of this note to publish it. I would sit down, sketch diagrams, write introductions like: “I’ve been working with APIs for a long time; it all started back then, evolved this way, and now I’m here, talking about APIs as dependencies that need proper maintenance.”
I tried to formulate justifications for a simple idea — API is a product with its own lifecycle — and now it’s time to try a new format
In this note, I want to gather feedback, refine it, and create a prototype
Motivation
There are plenty of services where I can store my API, publish documentation, create portals, and so on. But I don’t have a clear understanding of how to work with APIs
I primarily code in Java, so before I start writing anything, I open GitHub and Maven Central to find the right packages to work with. If I’m using Spring or Quarkus, I can even request a skeleton of the future service — for example, via https://code.quarkus.io
Next, I launch my IDE, import the project, and easily track all dependency changes declared in pom.xml
, for example
But when it comes to API integration, things get really messy:
First, I need to find the API. If it’s in OpenAPI or AsyncAPI format, that’s great — I can download it and generate clients (even better if there’s a dependency available, which I still need to match correctly with the API version).
If there are no clients/DTOs/other artifacts, I have to generate everything manually or use an agent, hoping the contract was written correctly and everything works on the first try
This problem isn’t new — otherwise, we wouldn’t have so many package repositories:
- NPM
- Maven Central
- etc.
In my mind, a simple concept has formed: API + artifacts = API Repository — which any vendor could implement
API Repository
An important clarification — the goal of an API Repository is not to replace existing API inventory solutions, of which there are plenty:
- https://www.eventcatalog.dev/
- https://bump.sh/
- https://www.postman.com/
- Own Git repo
- etc.
But rather to act as a glue—a standard that connects an API with its corresponding packages.
To achieve this, only two dependencies are needed:
- A URL to the API contract
- Package coordinates in various repositories
API Lifecycle
I don’t want to overcomplicate things, so I propose a familiar lifecycle
stateDiagram-v2
direction LR
[*] --> DRAFT
DRAFT --> RC
RC --> RELEASE
RELEASE --> EOL
EOL --> DEPRECATED
DEPRECATED --> [*]
States:
Draft
– A draft version of the API that is still evolving.RC
(Release Candidate) – The API structure is stable, with only minor changes expected.Release
– The API is officially released and will not change.EOL
(End of Life) – The API will soon be retired, and its use is not recommended.Deprecated
– The API has been fully decommissioned and can no longer be used.
Restrictions Based on Lifecycle:
Draft
,RC
APIs are available for:- Reading
- Writing
Release
,EOL
,Deprecated
APIs are available for:- Reading only
Now it’s clearer which phase the API is in and when it will be decommissioned
API Repository structure
Storing APIs
An API can be stored anywhere, so focusing on its storage isn’t necessary. Instead, it’s enough to define a structure that includes:
- The API contract
- The current lifecycle phase
- The availability of artifacts in package managers
How It Works in Practice
As the API contract is finalized, we generate a URN and register it in the repository
1
2
3
4
urn: com.myCompanyName:asyncapi:chat-api-gateway:2.34.0
cycle: Draft
contract:
$ref: 'https://api-hub.com/apis/asyncapi/5a0052fd-fd49-4eb1-a257-1f246906a0da'
Next, if possible, we generate the necessary packages from the contract and publish them in repositories
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
urn: com.myCompanyName:asyncapi:chat-api-gateway:2.34.0
cycle: Release
contract:
$ref: 'https://api-hub.com/apis/asyncapi/5a0052fd-fd49-4eb1-a257-1f246906a0da'
artifacts:
maven:
- language: java
framework: any # spring-boot | quarkus | micronaut
dependency: # com.github.myCompanyName:chat-api-gateway:2.34.0
groupId: com.myCompanyName
artifactId: chat-api-gateway
version: 2.34.0
npm:
- language: type-script
dependency: # "@myCompanyName/chat-api-gateway": "2.34.0"
groupId: myCompanyName
artifactId: chat-api-gateway
version: 2.34.0
When the API is ready for an update or deprecation, we move it to EOL
, indicating when this will happen and what alternatives are available
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
urn: com.myCompanyName:asyncapi:chat-api-gateway:2.34.0
cycle: EOL
deprecationNotice:
note: Current version of API will be deprecated at 2025-09-26
eliminationDate: 2025-09-26
successor:
urn: com.myCompanyName:asyncapi:chat-api-gateway:3.0.0
contract:
$ref: 'https://api-hub.com/apis/asyncapi/5a0052fd-fd49-4eb1-a257-1f246906a0da'
artifacts:
maven:
- language: java
framework: any # spring-boot | quarkus | micronaut
dependency: # com.github.myCompanyName:chat-api-gateway:2.34.0
groupId: com.myCompanyName
artifactId: chat-api-gateway
version: 2.34.0
npm:
- language: type-script
dependency: # "@myCompanyName/chat-api-gateway": "2.34.0"
groupId: myCompanyName
artifactId: chat-api-gateway
version: 2.34.0
Ultimately, we move it to the Deprecated
lifecycle phase
Working with APIs
There is a place to fetch the API from, so it’s time to figure out how to do that
It will all start with api.yaml
at the root of the project, which we will create
1
touch api.yaml
In the manifest, we will list the repositories we need and the parameters for searching and building the API packages
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
repositories:
- id: central
url: https://www.eventcatalog.dev/.well-known/api-repository
- id: myCompanyName_repo_1
url: https://github.com/myCompanyName/ai-apis/tree/main/.well-known/api-repository
- id: myCompanyName_repo_2
url: https://github.com/myCompanyName/chat-apis/tree/main/.well-known/api-repository
build:
jvm:
language: java
framework: quarkus
build-tool: maven
build-file-path: ./pom.xml
dependencies-have-priority: true
Once the repositories are listed, we can specify the APIs we are using
1
2
3
4
5
apis:
- groupId: com.github.myCompanyName
type: asyncapi
apiId: chat-api-gateway
version: 2.34.0
If we are only interested in specific operations, we have the option to specify only those
1
2
3
4
5
6
7
8
apis:
- groupId: com.github.myCompanyName
type: openapi
apiId: chat-api-ai-assistant
version: 1.3.0
operationIds:
- aiGenerateChatBlock
- aiGetChatBlockSchemas
Once the manifest is ready, you can work with it via the CLI
Reviewing dependencies
1
2
3
4
5
6
➜ git:(main) ✗ apim ls
demo-project@1.6.0 /Users/pavelbodyachevskiy/IdeaProjects/demo-project
├── com.github.myCompanyName:asyncapi:chat-api-gateway:2.34.0
├─┬ com.github.myCompanyName:openapi:chat-api-ai-assistant:1.3.0
│ ├── aiGenerateChatBlock
│ ├── aiGetChatBlockSchemas
Adding new API
1
➜ git:(main) ✗ apim install com.github.myCompanyName:asyncapi:notifications-api:1.73.25:subscribeForRejectedMessages
Building project
1
➜ git:(main) ✗ apim build
During the build process, we go to the API Repository and check the lifecycle phase of the API and whether it has the appropriate packages. In the case of Draft
, RC
, EOL
, or Deprecated
, we display a warning while continuing the project build
If the API has packages, we add the dependency
1
2
3
4
5
6
7
8
<dependencies>
<!-- Chat API AI Assistant Client -->
<dependency>
<groupId>com.github.myCompanyName</groupId>
<artifactId>chat-api-gateway</artifactId>
<version>2.34.0</version>
</dependency>
</dependencies>
If that’s not possible, we generate the necessary package locally
1
2
3
4
5
find . -type f
./generated/src/main/java/com.example/inegrations/com.github.myCompanyName/asyncapi/notifications-api/v_1_73_25/RejectedMessagesListener.java
./src/main/java/Application.java
./api.yaml
./pom.xml
Conclusion
In this note, I want to show that the introduction of a small standard can make working with APIs easier
There’s a repository you can refer to. It will tell you which APIs are current, which are not, and whether they have official packages that can be reused.
By writing an IDE extension:
- I can retrieve information in the background about APIs that are going to be deprecated, and update them in time
- Implement autocomplete and documentation display for used APIs
- Pass the necessary information to the agent via MCP for code generation or to find the required API
What do you think?