-
Notifications
You must be signed in to change notification settings - Fork 59
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: Autoscaler #1077
feat: Autoscaler #1077
Changes from 30 commits
c0630df
b1b898d
ea8860c
d62bb01
8f58902
96c71eb
9eafb76
5ef31ef
ce608de
13097f5
3498c9f
bd780e5
4270a6a
67f6645
370d41c
72d54ea
152d6e3
059aa4e
51ec882
bc32192
822bd6d
4798915
19ac452
931b25c
9ea29bb
61eb533
b4ad269
8ec8232
a95bf13
961697d
749ac86
d759f43
c7cf8a2
b4025ed
232cc0d
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -22,6 +22,7 @@ const pumpify = require('pumpify'); | |
import {google} from '../protos/protos'; | ||
import {Bigtable} from '.'; | ||
import {Instance} from './instance'; | ||
import {ClusterUtils} from './utils/cluster'; | ||
|
||
import { | ||
Backup, | ||
|
@@ -75,7 +76,10 @@ export type GetClustersCallback = ( | |
apiResponse?: google.bigtable.admin.v2.IListClustersResponse | ||
) => void; | ||
export interface SetClusterMetadataOptions { | ||
nodes: number; | ||
nodes?: number; | ||
minServeNodes?: number; | ||
maxServeNodes?: number; | ||
cpuUtilizationPercent?: number; | ||
} | ||
export type SetClusterMetadataCallback = GenericOperationCallback< | ||
Operation | null | undefined | ||
|
@@ -84,8 +88,11 @@ export interface BasicClusterConfig { | |
encryption?: google.bigtable.admin.v2.Cluster.IEncryptionConfig; | ||
key?: string; | ||
location: string; | ||
nodes: number; | ||
nodes?: number; | ||
storage?: string; | ||
minServeNodes?: number; | ||
maxServeNodes?: number; | ||
cpuUtilizationPercent?: number; | ||
} | ||
|
||
export interface CreateBackupConfig extends ModifiableBackupFields { | ||
|
@@ -690,35 +697,42 @@ Please use the format 'my-cluster' or '${instance.name}/clusters/my-cluster'.`); | |
gaxOptionsOrCallback?: CallOptions | SetClusterMetadataCallback, | ||
cb?: SetClusterMetadataCallback | ||
): void | Promise<SetClusterMetadataResponse> { | ||
ClusterUtils.validateMetadata(metadata); | ||
const callback = | ||
typeof gaxOptionsOrCallback === 'function' ? gaxOptionsOrCallback : cb!; | ||
const gaxOptions = | ||
typeof gaxOptionsOrCallback === 'object' | ||
? gaxOptionsOrCallback | ||
: ({} as CallOptions); | ||
|
||
const reqOpts: ICluster = Object.assign( | ||
{}, | ||
{ | ||
name: this.name, | ||
serveNodes: metadata.nodes, | ||
}, | ||
metadata | ||
); | ||
// eslint-disable-next-line @typescript-eslint/no-explicit-any | ||
delete (reqOpts as any).nodes; | ||
|
||
this.bigtable.request<Operation>( | ||
{ | ||
client: 'BigtableInstanceAdminClient', | ||
method: 'updateCluster', | ||
reqOpts, | ||
gaxOpts: gaxOptions, | ||
}, | ||
(err, resp) => { | ||
callback(err, resp); | ||
} | ||
); | ||
const setMetadataWithLocation = (location: string, name: string) => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I would remove this logic if we think it's currently not needed based on your testing. |
||
const reqOpts = ClusterUtils.getRequestFromMetadata( | ||
metadata, | ||
location, | ||
name | ||
); | ||
this.bigtable.request<Operation>( | ||
{ | ||
client: 'BigtableInstanceAdminClient', | ||
method: 'partialUpdateCluster', | ||
reqOpts: reqOpts, | ||
gaxOpts: gaxOptions, | ||
}, | ||
(err, resp) => { | ||
callback(err, resp); | ||
} | ||
); | ||
}; | ||
if (this.metadata && this.metadata.location) { | ||
setMetadataWithLocation(this.metadata.location, this.name); | ||
} else { | ||
this.getMetadata(gaxOptions, (err, res) => { | ||
if (err || !this.metadata || !this.metadata.location) { | ||
callback(err, null); | ||
} else { | ||
setMetadataWithLocation(this.metadata.location, this.name); | ||
kolea2 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
} | ||
}); | ||
} | ||
} | ||
} | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,154 @@ | ||
// Copyright 2022 Google LLC | ||
// | ||
// Licensed under the Apache License, Version 2.0 (the "License"); | ||
// you may not use this file except in compliance with the License. | ||
// You may obtain a copy of the License at | ||
// | ||
// https://www.apache.org/licenses/LICENSE-2.0 | ||
// | ||
// Unless required by applicable law or agreed to in writing, software | ||
// distributed under the License is distributed on an "AS IS" BASIS, | ||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
// See the License for the specific language governing permissions and | ||
// limitations under the License. | ||
|
||
import * as protos from '../../protos/protos'; | ||
import { | ||
BasicClusterConfig, | ||
ICluster, | ||
SetClusterMetadataOptions, | ||
} from '../cluster'; | ||
import {google} from '../../protos/protos'; | ||
|
||
export class ClusterUtils { | ||
static noConfigError = | ||
'Must specify either serve_nodes or all of the autoscaling configurations (min_serve_nodes, max_serve_nodes, and cpu_utilization_percent).'; | ||
static allConfigError = | ||
'Cannot specify both serve_nodes and autoscaling configurations (min_serve_nodes, max_serve_nodes, and cpu_utilization_percent).'; | ||
static incompleteConfigError = | ||
'All of autoscaling configurations must be specified at the same time (min_serve_nodes, max_serve_nodes, and cpu_utilization_percent).'; | ||
|
||
static validateMetadata( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I would rename this There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. How about |
||
metadata: SetClusterMetadataOptions | BasicClusterConfig | ||
): void { | ||
if (metadata.nodes) { | ||
if ( | ||
metadata.minServeNodes || | ||
metadata.maxServeNodes || | ||
metadata.cpuUtilizationPercent | ||
) { | ||
throw new Error(this.allConfigError); | ||
} | ||
} else { | ||
if ( | ||
metadata.minServeNodes || | ||
metadata.maxServeNodes || | ||
metadata.cpuUtilizationPercent | ||
) { | ||
if ( | ||
!( | ||
metadata.minServeNodes && | ||
metadata.maxServeNodes && | ||
metadata.cpuUtilizationPercent | ||
) | ||
) { | ||
throw new Error(this.incompleteConfigError); | ||
} | ||
} else { | ||
throw new Error(this.noConfigError); | ||
} | ||
} | ||
} | ||
static getUpdateMask(metadata: SetClusterMetadataOptions): string[] { | ||
const updateMask: string[] = []; | ||
if (metadata.nodes) { | ||
updateMask.push('serve_nodes'); | ||
if ( | ||
!( | ||
metadata.minServeNodes || | ||
metadata.maxServeNodes || | ||
metadata.cpuUtilizationPercent | ||
) | ||
) { | ||
updateMask.push('cluster_config.cluster_autoscaling_config'); | ||
} | ||
} | ||
if (metadata.minServeNodes) { | ||
updateMask.push( | ||
'cluster_config.cluster_autoscaling_config.autoscaling_limits.min_serve_nodes' | ||
); | ||
} | ||
if (metadata.maxServeNodes) { | ||
updateMask.push( | ||
'cluster_config.cluster_autoscaling_config.autoscaling_limits.max_serve_nodes' | ||
); | ||
} | ||
if (metadata.cpuUtilizationPercent) { | ||
updateMask.push( | ||
'cluster_config.cluster_autoscaling_config.autoscaling_targets.cpu_utilization_percent' | ||
); | ||
} | ||
return updateMask; | ||
} | ||
|
||
static getClusterBaseConfig( | ||
metadata: SetClusterMetadataOptions | BasicClusterConfig, | ||
location: string | undefined, | ||
name: string | undefined | ||
): google.bigtable.admin.v2.ICluster { | ||
let clusterConfig; | ||
if ( | ||
metadata.cpuUtilizationPercent || | ||
metadata.minServeNodes || | ||
metadata.maxServeNodes | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. what happens if only one of these are set? Let's say There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. When we use this in There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. When creating instances we now validate inputs. |
||
) { | ||
clusterConfig = { | ||
clusterAutoscalingConfig: { | ||
autoscalingTargets: { | ||
cpuUtilizationPercent: metadata.cpuUtilizationPercent, | ||
}, | ||
autoscalingLimits: { | ||
minServeNodes: metadata.minServeNodes, | ||
maxServeNodes: metadata.maxServeNodes, | ||
}, | ||
}, | ||
}; | ||
} | ||
return Object.assign( | ||
{}, | ||
name ? {name} : null, | ||
location ? {location} : null, | ||
clusterConfig ? {clusterConfig} : null, | ||
metadata.nodes ? {serveNodes: metadata.nodes} : null | ||
); | ||
} | ||
|
||
static getClusterFromMetadata( | ||
metadata: SetClusterMetadataOptions, | ||
location: string, | ||
name: string | ||
): google.bigtable.admin.v2.ICluster { | ||
const cluster: ICluster = Object.assign( | ||
{}, | ||
this.getClusterBaseConfig(metadata, location, name), | ||
metadata | ||
); | ||
// eslint-disable-next-line @typescript-eslint/no-explicit-any | ||
delete (cluster as any).nodes; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't think you should have to case to |
||
delete (cluster as any).minServeNodes; | ||
delete (cluster as any).maxServeNodes; | ||
delete (cluster as any).cpuUtilizationPercent; | ||
return cluster; | ||
} | ||
|
||
static getRequestFromMetadata( | ||
metadata: SetClusterMetadataOptions, | ||
location: string, | ||
name: string | ||
): protos.google.bigtable.admin.v2.IPartialUpdateClusterRequest { | ||
return { | ||
cluster: this.getClusterFromMetadata(metadata, location, name), | ||
updateMask: {paths: this.getUpdateMask(metadata)}, | ||
}; | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this considered breaking if we change something from required to optional? I'm guessing it would be the other way around, but want to confirm :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here we are changing something from required to optional so that would make it not a breaking change since users can still pass just
nodes
in. This is where we makenodes
optional because instead the user can provide autoscaling parameters. These parameters will be validated against a new validation function and an error will be thrown if an invalid combination of parameters are provided.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would not consider switching from required to optional for TypeScript breaking 👍