forked from jlandersen/vscode-kafka
-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Execute Produce code lens in kafka file with progress
Fixes jlandersen#117 Signed-off-by: azerr <[email protected]>
- Loading branch information
1 parent
86f7bf3
commit 65414a2
Showing
7 changed files
with
333 additions
and
39 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,219 @@ | ||
import { ProducerRecord } from "kafkajs"; | ||
import * as vscode from "vscode"; | ||
import { addQueryParameter, ClientAccessor } from "."; | ||
import { Producer as KafkaJSProducer } from "kafkajs"; | ||
|
||
|
||
export enum ProducerLaunchState { | ||
none, | ||
connecting, | ||
connected, | ||
sending, | ||
sent, | ||
disconnecting, | ||
disconnected | ||
} | ||
|
||
export interface ProducerCollectionChangedEvent { | ||
producers: Producer[]; | ||
} | ||
|
||
export class Producer implements vscode.Disposable { | ||
|
||
public state: ProducerLaunchState = ProducerLaunchState.none; | ||
|
||
private producer: KafkaJSProducer | undefined; | ||
|
||
constructor(public uri: vscode.Uri, private clientAccessor: ClientAccessor) { | ||
|
||
} | ||
|
||
async start(): Promise<void> { | ||
const [clusterId] = this.uri.path.split("/"); | ||
const client = this.clientAccessor.get(clusterId); | ||
this.producer = await client.producer(); | ||
await this.producer.connect(); | ||
} | ||
|
||
async send(record: ProducerRecord) { | ||
if (this.producer) { | ||
await this.producer.send(record); | ||
} | ||
} | ||
|
||
async dispose(): Promise<void> { | ||
if (this.producer) { | ||
await this.producer.disconnect(); | ||
this.producer = undefined; | ||
} | ||
} | ||
} | ||
|
||
/** | ||
* A collection of producers. | ||
*/ | ||
export class ProducerCollection implements vscode.Disposable { | ||
|
||
private producers: { [id: string]: Producer } = {}; | ||
|
||
private onDidChangeCollectionEmitter = new vscode.EventEmitter<ProducerCollectionChangedEvent>(); | ||
|
||
public onDidChangeCollection = this.onDidChangeCollectionEmitter.event; | ||
|
||
constructor(private clientAccessor: ClientAccessor) { | ||
|
||
} | ||
/** | ||
* Creates a new producer for a provided uri. | ||
*/ | ||
async create(uri: vscode.Uri): Promise<Producer> { | ||
|
||
// Create the producer | ||
const producer = new Producer(uri, this.clientAccessor); | ||
this.producers[uri.toString()] = producer; | ||
|
||
// Fire an event to notify that producer is connecting | ||
this.changeState(producer, ProducerLaunchState.connecting); | ||
|
||
// Start the producer | ||
try { | ||
await producer.start(); | ||
} | ||
catch (e) { | ||
this.handleProducerError(producer, e); | ||
} | ||
|
||
// Fire an event to notify that producer is connected | ||
this.changeState(producer, ProducerLaunchState.connected); | ||
|
||
return producer; | ||
} | ||
|
||
dispose(): void { | ||
this.disposeProducers(); | ||
this.onDidChangeCollectionEmitter.dispose(); | ||
} | ||
|
||
disposeProducers(): void { | ||
Object.keys(this.producers).forEach((key) => { | ||
this.producers[key].dispose(); | ||
}); | ||
|
||
this.producers = {}; | ||
} | ||
|
||
/** | ||
* Retrieve the number of active producers | ||
*/ | ||
length(): number { | ||
return Object.keys(this.producers).length; | ||
} | ||
|
||
/** | ||
* Retrieve an existing producer if exists. | ||
*/ | ||
get(uri: vscode.Uri): Producer | null { | ||
if (!this.has(uri)) { | ||
return null; | ||
} | ||
|
||
return this.producers[uri.toString()]; | ||
} | ||
|
||
/** | ||
* Retrieve all producers | ||
*/ | ||
getAll(): Producer[] { | ||
return Object.keys(this.producers).map((c) => this.producers[c]); | ||
} | ||
|
||
/** | ||
* Check whether a producer exists. | ||
*/ | ||
has(uri: vscode.Uri): boolean { | ||
return this.producers.hasOwnProperty(uri.toString()); | ||
} | ||
|
||
async send(uri: vscode.Uri, record: ProducerRecord): Promise<void> { | ||
const producer = this.get(uri); | ||
|
||
if (producer === null) { | ||
return; | ||
} | ||
|
||
// Fire an event to notify that producer message is sending | ||
this.changeState(producer, ProducerLaunchState.sending); | ||
|
||
// Send messages | ||
try { | ||
await producer.send(record); | ||
} | ||
catch (e) { | ||
this.handleProducerError(producer, e); | ||
} | ||
|
||
// Fire an event to notify that producer message is sent | ||
this.changeState(producer, ProducerLaunchState.sent); | ||
} | ||
|
||
/** | ||
* Disconnect an existing producer if exists. | ||
*/ | ||
async disconnect(uri: vscode.Uri): Promise<void> { | ||
const producer = this.get(uri); | ||
|
||
if (producer === null) { | ||
return; | ||
} | ||
|
||
// Fire an event to notify that producer is disconnecting | ||
this.changeState(producer, ProducerLaunchState.disconnecting); | ||
|
||
try { | ||
await producer.dispose(); | ||
} | ||
catch (e) { | ||
this.handleProducerError(producer, e); | ||
} | ||
|
||
// Fire an event to notify that producer is disconnected | ||
this.changeState(producer, ProducerLaunchState.disconnected); | ||
delete this.producers[uri.toString()]; | ||
|
||
} | ||
|
||
private handleProducerError(producer: Producer, e: Error) { | ||
this.changeState(producer, ProducerLaunchState.none); | ||
delete this.producers[producer.uri.toString()]; | ||
throw e; | ||
} | ||
|
||
private changeState(producer: Producer, state: ProducerLaunchState) { | ||
producer.state = state; | ||
this.onDidChangeCollectionEmitter.fire({ | ||
producers: [producer] | ||
}); | ||
} | ||
} | ||
|
||
// ---------- Producer URI utilities | ||
|
||
export interface ProducerInfoUri { | ||
clusterId: string; | ||
topicId?: string; | ||
key?: string; | ||
value: string; | ||
} | ||
|
||
const TOPIC_QUERY_PARAMETER = 'topic'; | ||
const KEY_QUERY_PARAMETER = 'key'; | ||
const VALUE_QUERY_PARAMETER = 'value'; | ||
|
||
export function createProducerUri(info: ProducerInfoUri): vscode.Uri { | ||
const path = `kafka:${info.clusterId}`; | ||
let query = ''; | ||
query = addQueryParameter(query, TOPIC_QUERY_PARAMETER, info.topicId); | ||
query = addQueryParameter(query, KEY_QUERY_PARAMETER, info.key); | ||
query = addQueryParameter(query, VALUE_QUERY_PARAMETER, info.value); | ||
return vscode.Uri.parse(path + query); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.